mirror of
				https://github.com/johrpan/musicus.git
				synced 2025-10-26 03:47:23 +01:00 
			
		
		
		
	library: Add export functionality
This commit is contained in:
		
							parent
							
								
									d49b9a9efe
								
							
						
					
					
						commit
						14416d49d2
					
				
					 11 changed files with 893 additions and 16 deletions
				
			
		
							
								
								
									
										293
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							
							
						
						
									
										293
									
								
								Cargo.lock
									
										
									
										generated
									
									
									
								
							|  | @ -2,6 +2,23 @@ | ||||||
| # It is not intended for manual editing. | # It is not intended for manual editing. | ||||||
| version = 4 | version = 4 | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "adler2" | ||||||
|  | version = "2.0.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "512761e0bb2578dd7380c6baaa0f4ce03e84f95e960231d1dec8bf4d7d6e2627" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "aes" | ||||||
|  | version = "0.8.4" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b169f7a6d4742236a0a00c541b845991d0ac43e546831af1249753ab4c3aa3a0" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  |  "cipher", | ||||||
|  |  "cpufeatures", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "aho-corasick" | name = "aho-corasick" | ||||||
| version = "1.1.3" | version = "1.1.3" | ||||||
|  | @ -32,6 +49,15 @@ version = "1.0.95" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04" | checksum = "34ac096ce696dc2fcabef30516bb13c0a68a11d30131d3df6f04711467681b04" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "arbitrary" | ||||||
|  | version = "1.4.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "dde20b3d026af13f561bdd0f15edf01fc734f0dafcedbaf42bba506a9517f223" | ||||||
|  | dependencies = [ | ||||||
|  |  "derive_arbitrary", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "async-broadcast" | name = "async-broadcast" | ||||||
| version = "0.7.2" | version = "0.7.2" | ||||||
|  | @ -239,6 +265,25 @@ version = "1.5.0" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" | checksum = "1fd0f2584146f6f2ef48085050886acf353beff7305ebd1ae69500e27c67f64b" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "bzip2" | ||||||
|  | version = "0.5.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "49ecfb22d906f800d4fe833b6282cf4dc1c298f5057ca0b5445e5c209735ca47" | ||||||
|  | dependencies = [ | ||||||
|  |  "bzip2-sys", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "bzip2-sys" | ||||||
|  | version = "0.1.13+1.0.8" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "225bff33b2141874fe80d71e07d6eec4f85c5c216453dd96388240f96e1acc14" | ||||||
|  | dependencies = [ | ||||||
|  |  "cc", | ||||||
|  |  "pkg-config", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "cairo-rs" | name = "cairo-rs" | ||||||
| version = "0.20.7" | version = "0.20.7" | ||||||
|  | @ -268,6 +313,8 @@ version = "1.2.9" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "c8293772165d9345bdaaa39b45b2109591e63fe5e6fbc23c6ff930a048aa310b" | checksum = "c8293772165d9345bdaaa39b45b2109591e63fe5e6fbc23c6ff930a048aa310b" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  |  "jobserver", | ||||||
|  |  "libc", | ||||||
|  "shlex", |  "shlex", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | @ -307,6 +354,16 @@ dependencies = [ | ||||||
|  "windows-targets", |  "windows-targets", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "cipher" | ||||||
|  | version = "0.4.4" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "773f3b9af64447d2ce9850330c473515014aa235e6a783b02db81ff39e4a3dad" | ||||||
|  | dependencies = [ | ||||||
|  |  "crypto-common", | ||||||
|  |  "inout", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "concurrent-queue" | name = "concurrent-queue" | ||||||
| version = "2.5.0" | version = "2.5.0" | ||||||
|  | @ -316,6 +373,12 @@ dependencies = [ | ||||||
|  "crossbeam-utils", |  "crossbeam-utils", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "constant_time_eq" | ||||||
|  | version = "0.3.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "7c74b8349d32d297c9134b8c88677813a227df8f779daa29bfc29c183fe3dca6" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "core-foundation-sys" | name = "core-foundation-sys" | ||||||
| version = "0.8.7" | version = "0.8.7" | ||||||
|  | @ -331,6 +394,30 @@ dependencies = [ | ||||||
|  "libc", |  "libc", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "crc" | ||||||
|  | version = "3.2.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "69e6e4d7b33a94f0991c26729976b10ebde1d34c3ee82408fb536164fa10d636" | ||||||
|  | dependencies = [ | ||||||
|  |  "crc-catalog", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "crc-catalog" | ||||||
|  | version = "2.4.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "19d374276b40fb8bbdee95aef7c7fa6b5316ec764510eb64b8dd0e2ed0d7e7f5" | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "crc32fast" | ||||||
|  | version = "1.4.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "a97769d94ddab943e4510d138150169a2758b5ef3eb191a9ee688de3e23ef7b3" | ||||||
|  | dependencies = [ | ||||||
|  |  "cfg-if", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "crossbeam-utils" | name = "crossbeam-utils" | ||||||
| version = "0.8.21" | version = "0.8.21" | ||||||
|  | @ -382,6 +469,12 @@ dependencies = [ | ||||||
|  "syn", |  "syn", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "deflate64" | ||||||
|  | version = "0.1.9" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "da692b8d1080ea3045efaab14434d40468c3d8657e42abddfffca87b428f4c1b" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "deranged" | name = "deranged" | ||||||
| version = "0.3.11" | version = "0.3.11" | ||||||
|  | @ -391,6 +484,17 @@ dependencies = [ | ||||||
|  "powerfmt", |  "powerfmt", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "derive_arbitrary" | ||||||
|  | version = "1.4.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "30542c1ad912e0e3d22a1935c290e12e8a29d704a420177a31faad4a601a0800" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "diesel" | name = "diesel" | ||||||
| version = "2.2.6" | version = "2.2.6" | ||||||
|  | @ -444,6 +548,18 @@ checksum = "9ed9a281f7bc9b7576e61468ba615a66a5c8cfdff42420a70aa82701a3b1e292" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "block-buffer", |  "block-buffer", | ||||||
|  "crypto-common", |  "crypto-common", | ||||||
|  |  "subtle", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "displaydoc" | ||||||
|  | version = "0.2.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "97369cbbc041bc366949bc74d34658d6cda5621039731c6310521892a3a20ae0" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
|  | @ -546,6 +662,16 @@ dependencies = [ | ||||||
|  "rustc_version", |  "rustc_version", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "flate2" | ||||||
|  | version = "1.1.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "11faaf5a5236997af9848be0bef4db95824b1d534ebc64d0f0c6cf3e67bd38dc" | ||||||
|  | dependencies = [ | ||||||
|  |  "crc32fast", | ||||||
|  |  "miniz_oxide", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "fnv" | name = "fnv" | ||||||
| version = "1.0.7" | version = "1.0.7" | ||||||
|  | @ -1084,6 +1210,15 @@ version = "0.4.3" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" | checksum = "7f24254aa9a54b5c858eaee2f5bccdb46aaf0e486a595ed5fd8f86ba55232a70" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "hmac" | ||||||
|  | version = "0.12.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "6c49c37c09c17a53d937dfbb742eb3a961d65a994e6bcdcf37e7399d0cc8ab5e" | ||||||
|  | dependencies = [ | ||||||
|  |  "digest", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "iana-time-zone" | name = "iana-time-zone" | ||||||
| version = "0.1.61" | version = "0.1.61" | ||||||
|  | @ -1123,6 +1258,15 @@ dependencies = [ | ||||||
|  "hashbrown", |  "hashbrown", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "inout" | ||||||
|  | version = "0.1.4" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "879f10e63c20629ecabbb64a8010319738c66a5cd0c29b02d63d272b03751d01" | ||||||
|  | dependencies = [ | ||||||
|  |  "generic-array", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "itertools" | name = "itertools" | ||||||
| version = "0.13.0" | version = "0.13.0" | ||||||
|  | @ -1138,6 +1282,15 @@ version = "1.0.14" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674" | checksum = "d75a2a4b1b190afb6f5425f10f6a8f959d2ea0b9c2b1d79553551850539e4674" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "jobserver" | ||||||
|  | version = "0.1.32" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "48d1dbcbbeb6a7fec7e059840aa538bd62aaccf972c7346c4d9d2059312853d0" | ||||||
|  | dependencies = [ | ||||||
|  |  "libc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "js-sys" | name = "js-sys" | ||||||
| version = "0.3.77" | version = "0.3.77" | ||||||
|  | @ -1220,12 +1373,28 @@ dependencies = [ | ||||||
|  "winapi", |  "winapi", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "lockfree-object-pool" | ||||||
|  | version = "0.1.6" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "9374ef4228402d4b7e403e5838cb880d9ee663314b0a900d5a6aabf0c213552e" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "log" | name = "log" | ||||||
| version = "0.4.25" | version = "0.4.25" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "04cbf5b083de1c7e0222a7a51dbfdba1cbe1c6ab0b15e29fff3f6c077fd9cd9f" | checksum = "04cbf5b083de1c7e0222a7a51dbfdba1cbe1c6ab0b15e29fff3f6c077fd9cd9f" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "lzma-rs" | ||||||
|  | version = "0.3.0" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "297e814c836ae64db86b36cf2a557ba54368d03f6afcd7d947c266692f71115e" | ||||||
|  | dependencies = [ | ||||||
|  |  "byteorder", | ||||||
|  |  "crc", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "malloc_buf" | name = "malloc_buf" | ||||||
| version = "0.0.6" | version = "0.0.6" | ||||||
|  | @ -1271,6 +1440,15 @@ dependencies = [ | ||||||
|  "quote", |  "quote", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "miniz_oxide" | ||||||
|  | version = "0.8.5" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8e3e04debbb59698c15bacbb6d93584a8c0ca9cc3213cb423d31f760d8843ce5" | ||||||
|  | dependencies = [ | ||||||
|  |  "adler2", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "mpris-server" | name = "mpris-server" | ||||||
| version = "0.8.1" | version = "0.8.1" | ||||||
|  | @ -1295,6 +1473,7 @@ name = "musicus" | ||||||
| version = "0.1.0" | version = "0.1.0" | ||||||
| dependencies = [ | dependencies = [ | ||||||
|  "anyhow", |  "anyhow", | ||||||
|  |  "async-channel", | ||||||
|  "chrono", |  "chrono", | ||||||
|  "diesel", |  "diesel", | ||||||
|  "diesel_migrations", |  "diesel_migrations", | ||||||
|  | @ -1313,6 +1492,7 @@ dependencies = [ | ||||||
|  "serde_json", |  "serde_json", | ||||||
|  "tracing-subscriber", |  "tracing-subscriber", | ||||||
|  "uuid", |  "uuid", | ||||||
|  |  "zip", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
| [[package]] | [[package]] | ||||||
|  | @ -1468,6 +1648,16 @@ version = "1.0.15" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" | checksum = "57c0d7b74b563b49d38dae00a0c37d4d6de9b432382b2892f0574ddcae73fd0a" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "pbkdf2" | ||||||
|  | version = "0.12.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f8ed6a7761f76e3b9f92dfb0a60a6a6477c61024b775147ff0973a02653abaf2" | ||||||
|  | dependencies = [ | ||||||
|  |  "digest", | ||||||
|  |  "hmac", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "pin-project-lite" | name = "pin-project-lite" | ||||||
| version = "0.2.16" | version = "0.2.16" | ||||||
|  | @ -1740,6 +1930,12 @@ dependencies = [ | ||||||
|  "libc", |  "libc", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "simd-adler32" | ||||||
|  | version = "0.3.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "d66dc143e6b11c1eddc06d5c423cfc97062865baf299914ab64caa38182078fe" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "slab" | name = "slab" | ||||||
| version = "0.4.9" | version = "0.4.9" | ||||||
|  | @ -1767,6 +1963,12 @@ version = "0.11.1" | ||||||
| source = "registry+https://github.com/rust-lang/crates.io-index" | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
| checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" | checksum = "7da8b5736845d9f2fcb837ea5d9e2628564b3b043a70948a3f0b778838c5fb4f" | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "subtle" | ||||||
|  | version = "2.6.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "13c2bddecc57b384dee18652358fb23172facb8a2c51ccc10d74c157bdea3292" | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "syn" | name = "syn" | ||||||
| version = "2.0.96" | version = "2.0.96" | ||||||
|  | @ -2315,6 +2517,97 @@ dependencies = [ | ||||||
|  "syn", |  "syn", | ||||||
| ] | ] | ||||||
| 
 | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zeroize" | ||||||
|  | version = "1.8.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ced3678a2879b30306d323f4542626697a464a97c0a07c9aebf7ebca65cd4dde" | ||||||
|  | dependencies = [ | ||||||
|  |  "zeroize_derive", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zeroize_derive" | ||||||
|  | version = "1.4.2" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "ce36e65b0d2999d2aafac989fb249189a141aee1f53c612c1f37d72631959f69" | ||||||
|  | dependencies = [ | ||||||
|  |  "proc-macro2", | ||||||
|  |  "quote", | ||||||
|  |  "syn", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zip" | ||||||
|  | version = "2.2.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "b280484c454e74e5fff658bbf7df8fdbe7a07c6b2de4a53def232c15ef138f3a" | ||||||
|  | dependencies = [ | ||||||
|  |  "aes", | ||||||
|  |  "arbitrary", | ||||||
|  |  "bzip2", | ||||||
|  |  "constant_time_eq", | ||||||
|  |  "crc32fast", | ||||||
|  |  "crossbeam-utils", | ||||||
|  |  "deflate64", | ||||||
|  |  "displaydoc", | ||||||
|  |  "flate2", | ||||||
|  |  "hmac", | ||||||
|  |  "indexmap", | ||||||
|  |  "lzma-rs", | ||||||
|  |  "memchr", | ||||||
|  |  "pbkdf2", | ||||||
|  |  "rand", | ||||||
|  |  "sha1", | ||||||
|  |  "thiserror", | ||||||
|  |  "time", | ||||||
|  |  "zeroize", | ||||||
|  |  "zopfli", | ||||||
|  |  "zstd", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zopfli" | ||||||
|  | version = "0.8.1" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e5019f391bac5cf252e93bbcc53d039ffd62c7bfb7c150414d61369afe57e946" | ||||||
|  | dependencies = [ | ||||||
|  |  "bumpalo", | ||||||
|  |  "crc32fast", | ||||||
|  |  "lockfree-object-pool", | ||||||
|  |  "log", | ||||||
|  |  "once_cell", | ||||||
|  |  "simd-adler32", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zstd" | ||||||
|  | version = "0.13.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "e91ee311a569c327171651566e07972200e76fcfe2242a4fa446149a3881c08a" | ||||||
|  | dependencies = [ | ||||||
|  |  "zstd-safe", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zstd-safe" | ||||||
|  | version = "7.2.3" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "f3051792fbdc2e1e143244dc28c60f73d8470e93f3f9cbd0ead44da5ed802722" | ||||||
|  | dependencies = [ | ||||||
|  |  "zstd-sys", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
|  | [[package]] | ||||||
|  | name = "zstd-sys" | ||||||
|  | version = "2.0.14+zstd.1.5.7" | ||||||
|  | source = "registry+https://github.com/rust-lang/crates.io-index" | ||||||
|  | checksum = "8fb060d4926e4ac3a3ad15d864e99ceb5f343c6b34f5bd6d81ae6ed417311be5" | ||||||
|  | dependencies = [ | ||||||
|  |  "cc", | ||||||
|  |  "pkg-config", | ||||||
|  | ] | ||||||
|  | 
 | ||||||
| [[package]] | [[package]] | ||||||
| name = "zvariant" | name = "zvariant" | ||||||
| version = "4.2.0" | version = "4.2.0" | ||||||
|  |  | ||||||
|  | @ -6,6 +6,7 @@ edition = "2021" | ||||||
| [dependencies] | [dependencies] | ||||||
| adw = { package = "libadwaita", version = "0.7", features = ["v1_6"] } | adw = { package = "libadwaita", version = "0.7", features = ["v1_6"] } | ||||||
| anyhow = "1" | anyhow = "1" | ||||||
|  | async-channel = "2.3" | ||||||
| chrono = "0.4" | chrono = "0.4" | ||||||
| diesel = { version = "2.2", features = ["chrono", "sqlite"] } | diesel = { version = "2.2", features = ["chrono", "sqlite"] } | ||||||
| diesel_migrations = "2.2" | diesel_migrations = "2.2" | ||||||
|  | @ -23,3 +24,4 @@ serde = { version = "1", features = ["derive"] } | ||||||
| serde_json = "1" | serde_json = "1" | ||||||
| tracing-subscriber = "0.3" | tracing-subscriber = "0.3" | ||||||
| uuid = { version = "1", features = ["v4"] } | uuid = { version = "1", features = ["v4"] } | ||||||
|  | zip = "2.2" | ||||||
|  | @ -62,6 +62,27 @@ template $MusicusLibraryManager: Adw.NavigationPage { | ||||||
|               activated => $export_archive() swapped; |               activated => $export_archive() swapped; | ||||||
|             } |             } | ||||||
|           } |           } | ||||||
|  | 
 | ||||||
|  |           Gtk.Label { | ||||||
|  |             label: _("Progress"); | ||||||
|  |             visible: bind process_list.visible; | ||||||
|  |             xalign: 0; | ||||||
|  |             margin-top: 24; | ||||||
|  | 
 | ||||||
|  |             styles [ | ||||||
|  |               "heading", | ||||||
|  |             ] | ||||||
|  |           } | ||||||
|  | 
 | ||||||
|  |           Gtk.ListBox process_list { | ||||||
|  |             selection-mode: none; | ||||||
|  |             margin-top: 12; | ||||||
|  |             visible: false; | ||||||
|  | 
 | ||||||
|  |             styles [ | ||||||
|  |               "boxed-list-separate", | ||||||
|  |             ] | ||||||
|  |           } | ||||||
|         } |         } | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|  |  | ||||||
							
								
								
									
										65
									
								
								data/ui/process_row.blp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								data/ui/process_row.blp
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,65 @@ | ||||||
|  | using Gtk 4.0; | ||||||
|  | 
 | ||||||
|  | template $MusicusProcessRow: Gtk.ListBoxRow { | ||||||
|  |   activatable: false; | ||||||
|  | 
 | ||||||
|  |   Gtk.Box { | ||||||
|  |     orientation: vertical; | ||||||
|  |     spacing: 12; | ||||||
|  |     margin-top: 12; | ||||||
|  |     margin-bottom: 12; | ||||||
|  |     margin-start: 12; | ||||||
|  |     margin-end: 12; | ||||||
|  | 
 | ||||||
|  |     Gtk.Box { | ||||||
|  |       spacing: 12; | ||||||
|  | 
 | ||||||
|  |       Gtk.Box { | ||||||
|  |         orientation: vertical; | ||||||
|  |         hexpand: true; | ||||||
|  | 
 | ||||||
|  |         Gtk.Label description_label { | ||||||
|  |           wrap: true; | ||||||
|  |           xalign: 0.0; | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         Gtk.Label success_label { | ||||||
|  |           label: _("Process finished"); | ||||||
|  |           wrap: true; | ||||||
|  |           xalign: 0.0; | ||||||
|  |           visible: false; | ||||||
|  | 
 | ||||||
|  |           styles [ | ||||||
|  |             "success", | ||||||
|  |             "caption" | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         Gtk.Label error_label { | ||||||
|  |           wrap: true; | ||||||
|  |           visible: false; | ||||||
|  |           xalign: 0.0; | ||||||
|  | 
 | ||||||
|  |           styles [ | ||||||
|  |             "error", | ||||||
|  |             "caption" | ||||||
|  |           ] | ||||||
|  |         } | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  |       Gtk.Button remove_button { | ||||||
|  |         icon-name: "window-close-symbolic"; | ||||||
|  |         tooltip-text: _("Remove from list"); | ||||||
|  |         valign: start; | ||||||
|  |         visible: false; | ||||||
|  |         clicked => $remove() swapped; | ||||||
|  | 
 | ||||||
|  |         styles [ | ||||||
|  |           "flat", | ||||||
|  |         ] | ||||||
|  |       } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     Gtk.ProgressBar progress_bar {} | ||||||
|  |   } | ||||||
|  | } | ||||||
|  | @ -1,8 +1,10 @@ | ||||||
| use std::{ | use std::{ | ||||||
|     cell::{OnceCell, RefCell}, |     cell::{OnceCell, RefCell}, | ||||||
|     ffi::OsString, |     ffi::OsString, | ||||||
|     fs, |     fs::{self, File}, | ||||||
|  |     io::{BufWriter, Read, Write}, | ||||||
|     path::{Path, PathBuf}, |     path::{Path, PathBuf}, | ||||||
|  |     thread, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| use adw::{ | use adw::{ | ||||||
|  | @ -14,6 +16,7 @@ use anyhow::{anyhow, Result}; | ||||||
| use chrono::prelude::*; | use chrono::prelude::*; | ||||||
| use diesel::{dsl::exists, prelude::*, QueryDsl, SqliteConnection}; | use diesel::{dsl::exists, prelude::*, QueryDsl, SqliteConnection}; | ||||||
| use once_cell::sync::Lazy; | use once_cell::sync::Lazy; | ||||||
|  | use zip::{write::SimpleFileOptions, ZipWriter}; | ||||||
| 
 | 
 | ||||||
| use crate::{ | use crate::{ | ||||||
|     db::{self, models::*, schema::*, tables, TranslatedString}, |     db::{self, models::*, schema::*, tables, TranslatedString}, | ||||||
|  | @ -72,6 +75,39 @@ impl Library { | ||||||
|             .build() |             .build() | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  |     /// Import from a library archive.
 | ||||||
|  |     pub fn import(&self, _path: impl AsRef<Path>) -> Result<()> { | ||||||
|  |         Ok(()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     /// Export the whole music library to an archive at `path`. If `path` already exists, it will
 | ||||||
|  |     /// be overwritten. The work will be done in a background thread.
 | ||||||
|  |     pub fn export( | ||||||
|  |         &self, | ||||||
|  |         path: impl AsRef<Path>, | ||||||
|  |     ) -> Result<async_channel::Receiver<LibraryProcessMsg>> { | ||||||
|  |         let mut binding = self.imp().connection.borrow_mut(); | ||||||
|  |         let connection = &mut *binding.as_mut().unwrap(); | ||||||
|  | 
 | ||||||
|  |         let path = path.as_ref().to_owned(); | ||||||
|  |         let library_folder = PathBuf::from(&self.folder()); | ||||||
|  |         let tracks = tracks::table.load::<tables::Track>(connection)?; | ||||||
|  | 
 | ||||||
|  |         let (sender, receiver) = async_channel::unbounded::<LibraryProcessMsg>(); | ||||||
|  |         thread::spawn(move || { | ||||||
|  |             if let Err(err) = sender.send_blocking(LibraryProcessMsg::Result(write_zip( | ||||||
|  |                 path, | ||||||
|  |                 library_folder, | ||||||
|  |                 tracks, | ||||||
|  |                 &sender, | ||||||
|  |             ))) { | ||||||
|  |                 log::error!("Failed to send library action result: {err}"); | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         Ok(receiver) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|     pub fn search(&self, query: &LibraryQuery, search: &str) -> Result<LibraryResults> { |     pub fn search(&self, query: &LibraryQuery, search: &str) -> Result<LibraryResults> { | ||||||
|         let search = format!("%{}%", search); |         let search = format!("%{}%", search); | ||||||
|         let mut binding = self.imp().connection.borrow_mut(); |         let mut binding = self.imp().connection.borrow_mut(); | ||||||
|  | @ -1582,3 +1618,55 @@ impl LibraryResults { | ||||||
|             && self.albums.is_empty() |             && self.albums.is_empty() | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | fn write_zip( | ||||||
|  |     zip_path: impl AsRef<Path>, | ||||||
|  |     library_folder: impl AsRef<Path>, | ||||||
|  |     tracks: Vec<tables::Track>, | ||||||
|  |     sender: &async_channel::Sender<LibraryProcessMsg>, | ||||||
|  | ) -> Result<()> { | ||||||
|  |     let mut zip = zip::ZipWriter::new(BufWriter::new(fs::File::create(zip_path)?)); | ||||||
|  | 
 | ||||||
|  |     // Start with the database:
 | ||||||
|  |     add_file_to_zip(&mut zip, &library_folder, "musicus.db")?; | ||||||
|  | 
 | ||||||
|  |     let n_tracks = tracks.len(); | ||||||
|  | 
 | ||||||
|  |     // Include all tracks that are part of the library.
 | ||||||
|  |     for (index, track) in tracks.into_iter().enumerate() { | ||||||
|  |         add_file_to_zip(&mut zip, &library_folder, &track.path)?; | ||||||
|  | 
 | ||||||
|  |         // Ignore if the reveiver has been dropped.
 | ||||||
|  |         let _ = sender.send_blocking(LibraryProcessMsg::Progress( | ||||||
|  |             (index + 1) as f64 / n_tracks as f64, | ||||||
|  |         )); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     zip.finish()?; | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | // TODO: Cross-platform paths?
 | ||||||
|  | fn add_file_to_zip( | ||||||
|  |     zip: &mut ZipWriter<BufWriter<File>>, | ||||||
|  |     library_folder: impl AsRef<Path>, | ||||||
|  |     library_path: &str, | ||||||
|  | ) -> Result<()> { | ||||||
|  |     let file_path = library_folder.as_ref().join(PathBuf::from(library_path)); | ||||||
|  | 
 | ||||||
|  |     let mut file = File::open(file_path)?; | ||||||
|  |     let mut buffer = Vec::new(); | ||||||
|  |     file.read_to_end(&mut buffer)?; | ||||||
|  | 
 | ||||||
|  |     zip.start_file(library_path, SimpleFileOptions::default())?; | ||||||
|  |     zip.write_all(&buffer)?; | ||||||
|  | 
 | ||||||
|  |     Ok(()) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[derive(Debug)] | ||||||
|  | pub enum LibraryProcessMsg { | ||||||
|  |     Progress(f64), | ||||||
|  |     Result(Result<()>), | ||||||
|  | } | ||||||
|  |  | ||||||
|  | @ -1,10 +1,14 @@ | ||||||
| use std::{cell::OnceCell, ffi::OsStr, path::Path}; | use std::{cell::OnceCell, ffi::OsStr, path::Path}; | ||||||
| 
 | 
 | ||||||
| use adw::{prelude::*, subclass::prelude::*}; | use adw::{prelude::*, subclass::prelude::*}; | ||||||
|  | use formatx::formatx; | ||||||
| use gettextrs::gettext; | use gettextrs::gettext; | ||||||
| use gtk::glib; | use gtk::glib::{self, clone}; | ||||||
| 
 | 
 | ||||||
| use crate::{library::Library, window::Window}; | use crate::{ | ||||||
|  |     library::Library, process::Process, process_manager::ProcessManager, process_row::ProcessRow, | ||||||
|  |     window::Window, | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| mod imp { | mod imp { | ||||||
|     use super::*; |     use super::*; | ||||||
|  | @ -14,9 +18,12 @@ mod imp { | ||||||
|     pub struct LibraryManager { |     pub struct LibraryManager { | ||||||
|         pub navigation: OnceCell<adw::NavigationView>, |         pub navigation: OnceCell<adw::NavigationView>, | ||||||
|         pub library: OnceCell<Library>, |         pub library: OnceCell<Library>, | ||||||
|  |         pub process_manager: OnceCell<ProcessManager>, | ||||||
| 
 | 
 | ||||||
|         #[template_child] |         #[template_child] | ||||||
|         pub library_path_row: TemplateChild<adw::ActionRow>, |         pub library_path_row: TemplateChild<adw::ActionRow>, | ||||||
|  |         #[template_child] | ||||||
|  |         pub process_list: TemplateChild<gtk::ListBox>, | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[glib::object_subclass] |     #[glib::object_subclass] | ||||||
|  | @ -47,16 +54,28 @@ glib::wrapper! { | ||||||
| 
 | 
 | ||||||
| #[gtk::template_callbacks] | #[gtk::template_callbacks] | ||||||
| impl LibraryManager { | impl LibraryManager { | ||||||
|     pub fn new(navigation: &adw::NavigationView, library: &Library) -> Self { |     pub fn new( | ||||||
|  |         navigation: &adw::NavigationView, | ||||||
|  |         library: &Library, | ||||||
|  |         process_manager: &ProcessManager, | ||||||
|  |     ) -> Self { | ||||||
|         let obj: Self = glib::Object::new(); |         let obj: Self = glib::Object::new(); | ||||||
| 
 | 
 | ||||||
|         obj.imp().navigation.set(navigation.to_owned()).unwrap(); |         for process in process_manager.processes() { | ||||||
|         obj.imp().library.set(library.to_owned()).unwrap(); |             obj.add_process(&process); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         if let Some(Some(filename)) = Path::new(&library.folder()).file_name().map(OsStr::to_str) { |         if let Some(Some(filename)) = Path::new(&library.folder()).file_name().map(OsStr::to_str) { | ||||||
|             obj.imp().library_path_row.set_subtitle(filename); |             obj.imp().library_path_row.set_subtitle(filename); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|  |         obj.imp().navigation.set(navigation.to_owned()).unwrap(); | ||||||
|  |         obj.imp().library.set(library.to_owned()).unwrap(); | ||||||
|  |         obj.imp() | ||||||
|  |             .process_manager | ||||||
|  |             .set(process_manager.to_owned()) | ||||||
|  |             .unwrap(); | ||||||
|  | 
 | ||||||
|         obj |         obj | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | @ -85,8 +104,107 @@ impl LibraryManager { | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     #[template_callback] |     #[template_callback] | ||||||
|     fn import_archive(&self) {} |     async fn import_archive(&self) { | ||||||
|  |         let dialog = gtk::FileDialog::builder() | ||||||
|  |             .title(gettext("Import from library archive")) | ||||||
|  |             .modal(true) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         let root = self.root(); | ||||||
|  |         let window = root | ||||||
|  |             .as_ref() | ||||||
|  |             .and_then(|r| r.downcast_ref::<gtk::Window>()) | ||||||
|  |             .and_then(|w| w.downcast_ref::<Window>()) | ||||||
|  |             .unwrap(); | ||||||
|  | 
 | ||||||
|  |         match dialog.open_future(Some(window)).await { | ||||||
|  |             Err(err) => { | ||||||
|  |                 if !err.matches(gtk::DialogError::Dismissed) { | ||||||
|  |                     log::error!("File selection failed: {err}"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             Ok(path) => { | ||||||
|  |                 if let Some(path) = path.path() { | ||||||
|  |                     if let Err(err) = self.imp().library.get().unwrap().import(path) { | ||||||
|  |                         log::error!("Failed to import library from archive: {err}"); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     #[template_callback] |     #[template_callback] | ||||||
|     fn export_archive(&self) {} |     async fn export_archive(&self) { | ||||||
|  |         let dialog = gtk::FileDialog::builder() | ||||||
|  |             .title(gettext("Export library")) | ||||||
|  |             .modal(true) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         let root = self.root(); | ||||||
|  |         let window = root | ||||||
|  |             .as_ref() | ||||||
|  |             .and_then(|r| r.downcast_ref::<gtk::Window>()) | ||||||
|  |             .and_then(|w| w.downcast_ref::<Window>()) | ||||||
|  |             .unwrap(); | ||||||
|  | 
 | ||||||
|  |         match dialog.save_future(Some(window)).await { | ||||||
|  |             Err(err) => { | ||||||
|  |                 if !err.matches(gtk::DialogError::Dismissed) { | ||||||
|  |                     log::error!("File selection failed: {err}"); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |             Ok(path) => { | ||||||
|  |                 if let Some(path) = path.path() { | ||||||
|  |                     match self.imp().library.get().unwrap().export(&path) { | ||||||
|  |                         Ok(receiver) => { | ||||||
|  |                             let process = Process::new( | ||||||
|  |                                 &formatx!( | ||||||
|  |                                     gettext("Exporting music library to {}"), | ||||||
|  |                                     path.file_name() | ||||||
|  |                                         .map(|f| f.to_string_lossy().into_owned()) | ||||||
|  |                                         .unwrap_or(gettext("archive")) | ||||||
|  |                                 ) | ||||||
|  |                                 .unwrap(), | ||||||
|  |                                 receiver, | ||||||
|  |                             ); | ||||||
|  | 
 | ||||||
|  |                             self.imp() | ||||||
|  |                                 .process_manager | ||||||
|  |                                 .get() | ||||||
|  |                                 .unwrap() | ||||||
|  |                                 .add_process(&process); | ||||||
|  | 
 | ||||||
|  |                             self.add_process(&process); | ||||||
|  |                         } | ||||||
|  |                         Err(err) => log::error!("Failed to export library: {err}"), | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn add_process(&self, process: &Process) { | ||||||
|  |         let row = ProcessRow::new(process); | ||||||
|  | 
 | ||||||
|  |         row.connect_remove(clone!( | ||||||
|  |             #[weak(rename_to = obj)] | ||||||
|  |             self, | ||||||
|  |             move |row| { | ||||||
|  |                 obj.imp() | ||||||
|  |                     .process_manager | ||||||
|  |                     .get() | ||||||
|  |                     .unwrap() | ||||||
|  |                     .remove_process(&row.process()); | ||||||
|  | 
 | ||||||
|  |                 obj.imp().process_list.remove(row); | ||||||
|  | 
 | ||||||
|  |                 if obj.imp().process_list.first_child().is_none() { | ||||||
|  |                     obj.imp().process_list.set_visible(false); | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         )); | ||||||
|  | 
 | ||||||
|  |         self.imp().process_list.append(&row); | ||||||
|  |         self.imp().process_list.set_visible(true); | ||||||
|  |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | @ -4,7 +4,6 @@ mod application; | ||||||
| mod config; | mod config; | ||||||
| mod db; | mod db; | ||||||
| mod editor; | mod editor; | ||||||
| mod search_page; |  | ||||||
| mod library; | mod library; | ||||||
| mod library_manager; | mod library_manager; | ||||||
| mod player; | mod player; | ||||||
|  | @ -12,9 +11,13 @@ mod player_bar; | ||||||
| mod playlist_item; | mod playlist_item; | ||||||
| mod playlist_page; | mod playlist_page; | ||||||
| mod playlist_tile; | mod playlist_tile; | ||||||
|  | mod process; | ||||||
|  | mod process_manager; | ||||||
|  | mod process_row; | ||||||
| mod program; | mod program; | ||||||
| mod program_tile; | mod program_tile; | ||||||
| mod recording_tile; | mod recording_tile; | ||||||
|  | mod search_page; | ||||||
| mod search_tag; | mod search_tag; | ||||||
| mod selector; | mod selector; | ||||||
| mod tag_tile; | mod tag_tile; | ||||||
|  |  | ||||||
							
								
								
									
										67
									
								
								src/process.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										67
									
								
								src/process.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,67 @@ | ||||||
|  | use std::cell::{Cell, OnceCell, RefCell}; | ||||||
|  | 
 | ||||||
|  | use gtk::{ | ||||||
|  |     glib::{self, Properties}, | ||||||
|  |     prelude::*, | ||||||
|  |     subclass::prelude::*, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | use crate::library::LibraryProcessMsg; | ||||||
|  | 
 | ||||||
|  | mod imp { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[derive(Properties, Default, Debug)] | ||||||
|  |     #[properties(wrapper_type = super::Process)] | ||||||
|  |     pub struct Process { | ||||||
|  |         #[property(get, construct_only)] | ||||||
|  |         pub description: OnceCell<String>, | ||||||
|  |         #[property(get, set)] | ||||||
|  |         pub progress: Cell<f64>, | ||||||
|  |         #[property(get, set)] | ||||||
|  |         pub finished: Cell<bool>, | ||||||
|  |         #[property(get, set)] | ||||||
|  |         pub error: RefCell<Option<String>>, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[glib::object_subclass] | ||||||
|  |     impl ObjectSubclass for Process { | ||||||
|  |         const NAME: &'static str = "MusicusProcess"; | ||||||
|  |         type Type = super::Process; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[glib::derived_properties] | ||||||
|  |     impl ObjectImpl for Process {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | glib::wrapper! { | ||||||
|  |     pub struct Process(ObjectSubclass<imp::Process>); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Process { | ||||||
|  |     pub fn new(description: &str, receiver: async_channel::Receiver<LibraryProcessMsg>) -> Self { | ||||||
|  |         let obj: Self = glib::Object::builder() | ||||||
|  |             .property("description", description) | ||||||
|  |             .build(); | ||||||
|  | 
 | ||||||
|  |         let obj_clone = obj.clone(); | ||||||
|  |         glib::spawn_future_local(async move { | ||||||
|  |             while let Ok(msg) = receiver.recv().await { | ||||||
|  |                 match msg { | ||||||
|  |                     LibraryProcessMsg::Progress(fraction) => { | ||||||
|  |                         obj_clone.set_progress(fraction); | ||||||
|  |                     } | ||||||
|  |                     LibraryProcessMsg::Result(result) => { | ||||||
|  |                         if let Err(err) = result { | ||||||
|  |                             obj_clone.set_error(err.to_string()); | ||||||
|  |                         } | ||||||
|  | 
 | ||||||
|  |                         obj_clone.set_finished(true); | ||||||
|  |                     } | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         }); | ||||||
|  | 
 | ||||||
|  |         obj | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										57
									
								
								src/process_manager.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/process_manager.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,57 @@ | ||||||
|  | use std::cell::RefCell; | ||||||
|  | 
 | ||||||
|  | use gtk::{ | ||||||
|  |     glib::{self}, | ||||||
|  |     subclass::prelude::*, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | use crate::process::Process; | ||||||
|  | 
 | ||||||
|  | mod imp { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[derive(Debug, Default)] | ||||||
|  |     pub struct ProcessManager { | ||||||
|  |         pub processes: RefCell<Vec<Process>>, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[glib::object_subclass] | ||||||
|  |     impl ObjectSubclass for ProcessManager { | ||||||
|  |         const NAME: &'static str = "MusicusProcessManager"; | ||||||
|  |         type Type = super::ProcessManager; | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     impl ObjectImpl for ProcessManager {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | glib::wrapper! { | ||||||
|  |     pub struct ProcessManager(ObjectSubclass<imp::ProcessManager>); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl ProcessManager { | ||||||
|  |     pub fn new() -> Self { | ||||||
|  |         glib::Object::new() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn add_process(&self, process: &Process) { | ||||||
|  |         self.imp().processes.borrow_mut().push(process.to_owned()); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn processes(&self) -> Vec<Process> { | ||||||
|  |         self.imp().processes.borrow().clone() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn any_ongoing(&self) -> bool { | ||||||
|  |         self.imp().processes.borrow().iter().any(|p| !p.finished()) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     pub fn remove_process(&self, process: &Process) { | ||||||
|  |         self.imp().processes.borrow_mut().retain(|p| p != process); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Default for ProcessManager { | ||||||
|  |     fn default() -> Self { | ||||||
|  |         Self::new() | ||||||
|  |     } | ||||||
|  | } | ||||||
							
								
								
									
										128
									
								
								src/process_row.rs
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										128
									
								
								src/process_row.rs
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,128 @@ | ||||||
|  | use std::cell::OnceCell; | ||||||
|  | 
 | ||||||
|  | use formatx::formatx; | ||||||
|  | use gettextrs::gettext; | ||||||
|  | use gtk::{ | ||||||
|  |     glib::{self, subclass::Signal, Properties}, | ||||||
|  |     prelude::*, | ||||||
|  |     subclass::prelude::*, | ||||||
|  | }; | ||||||
|  | use once_cell::sync::Lazy; | ||||||
|  | 
 | ||||||
|  | use crate::process::Process; | ||||||
|  | 
 | ||||||
|  | mod imp { | ||||||
|  |     use super::*; | ||||||
|  | 
 | ||||||
|  |     #[derive(Properties, Debug, Default, gtk::CompositeTemplate)] | ||||||
|  |     #[properties(wrapper_type = super::ProcessRow)] | ||||||
|  |     #[template(file = "data/ui/process_row.blp")] | ||||||
|  |     pub struct ProcessRow { | ||||||
|  |         #[property(get, construct_only)] | ||||||
|  |         pub process: OnceCell<Process>, | ||||||
|  | 
 | ||||||
|  |         #[template_child] | ||||||
|  |         pub description_label: TemplateChild<gtk::Label>, | ||||||
|  |         #[template_child] | ||||||
|  |         pub success_label: TemplateChild<gtk::Label>, | ||||||
|  |         #[template_child] | ||||||
|  |         pub error_label: TemplateChild<gtk::Label>, | ||||||
|  |         #[template_child] | ||||||
|  |         pub remove_button: TemplateChild<gtk::Button>, | ||||||
|  |         #[template_child] | ||||||
|  |         pub progress_bar: TemplateChild<gtk::ProgressBar>, | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[glib::object_subclass] | ||||||
|  |     impl ObjectSubclass for ProcessRow { | ||||||
|  |         const NAME: &'static str = "MusicusProcessRow"; | ||||||
|  |         type Type = super::ProcessRow; | ||||||
|  |         type ParentType = gtk::ListBoxRow; | ||||||
|  | 
 | ||||||
|  |         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(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[glib::derived_properties] | ||||||
|  |     impl ObjectImpl for ProcessRow { | ||||||
|  |         fn signals() -> &'static [Signal] { | ||||||
|  |             static SIGNALS: Lazy<Vec<Signal>> = | ||||||
|  |                 Lazy::new(|| vec![Signal::builder("remove").build()]); | ||||||
|  | 
 | ||||||
|  |             SIGNALS.as_ref() | ||||||
|  |         } | ||||||
|  | 
 | ||||||
|  |         fn constructed(&self) { | ||||||
|  |             self.parent_constructed(); | ||||||
|  | 
 | ||||||
|  |             self.description_label | ||||||
|  |                 .set_label(&self.obj().process().description()); | ||||||
|  | 
 | ||||||
|  |             self.obj() | ||||||
|  |                 .process() | ||||||
|  |                 .bind_property("progress", &*self.progress_bar, "fraction") | ||||||
|  |                 .build(); | ||||||
|  | 
 | ||||||
|  |             let obj = self.obj().to_owned(); | ||||||
|  |             self.obj().process().connect_finished_notify(move |_| { | ||||||
|  |                 obj.update(); | ||||||
|  |             }); | ||||||
|  | 
 | ||||||
|  |             self.obj().update(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     impl WidgetImpl for ProcessRow {} | ||||||
|  |     impl ListBoxRowImpl for ProcessRow {} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | glib::wrapper! { | ||||||
|  |     pub struct ProcessRow(ObjectSubclass<imp::ProcessRow>) | ||||||
|  |         @extends gtk::Widget, gtk::ListBoxRow; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[gtk::template_callbacks] | ||||||
|  | impl ProcessRow { | ||||||
|  |     pub fn new(process: &Process) -> Self { | ||||||
|  |         glib::Object::builder().property("process", process).build() | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     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 | ||||||
|  |         }) | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     #[template_callback] | ||||||
|  |     fn remove(&self) { | ||||||
|  |         self.emit_by_name::<()>("remove", &[]); | ||||||
|  |     } | ||||||
|  | 
 | ||||||
|  |     fn update(&self) { | ||||||
|  |         if !self.process().finished() { | ||||||
|  |             self.imp() | ||||||
|  |                 .progress_bar | ||||||
|  |                 .set_fraction(self.process().progress()); | ||||||
|  |         } else { | ||||||
|  |             self.imp().progress_bar.set_visible(false); | ||||||
|  |             self.imp().remove_button.set_visible(true); | ||||||
|  | 
 | ||||||
|  |             if let Some(error) = self.process().error() { | ||||||
|  |                 self.imp() | ||||||
|  |                     .error_label | ||||||
|  |                     .set_label(&formatx!(gettext("Process failed: {}"), error).unwrap()); | ||||||
|  |                 self.imp().error_label.set_visible(true); | ||||||
|  |             } else { | ||||||
|  |                 self.imp().success_label.set_visible(true); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | @ -11,11 +11,15 @@ use crate::{ | ||||||
|     player::Player, |     player::Player, | ||||||
|     player_bar::PlayerBar, |     player_bar::PlayerBar, | ||||||
|     playlist_page::PlaylistPage, |     playlist_page::PlaylistPage, | ||||||
|  |     process_manager::ProcessManager, | ||||||
|     search_page::SearchPage, |     search_page::SearchPage, | ||||||
|     welcome_page::WelcomePage, |     welcome_page::WelcomePage, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| mod imp { | mod imp { | ||||||
|  |     use adw::prelude::{AlertDialogExt, AlertDialogExtManual}; | ||||||
|  |     use gettextrs::gettext; | ||||||
|  | 
 | ||||||
|     use super::*; |     use super::*; | ||||||
| 
 | 
 | ||||||
|     #[derive(Debug, Default, gtk::CompositeTemplate)] |     #[derive(Debug, Default, gtk::CompositeTemplate)] | ||||||
|  | @ -23,6 +27,7 @@ mod imp { | ||||||
|     pub struct Window { |     pub struct Window { | ||||||
|         pub library: RefCell<Option<Library>>, |         pub library: RefCell<Option<Library>>, | ||||||
|         pub player: Player, |         pub player: Player, | ||||||
|  |         pub process_manager: ProcessManager, | ||||||
| 
 | 
 | ||||||
|         #[template_child] |         #[template_child] | ||||||
|         pub stack: TemplateChild<gtk::Stack>, |         pub stack: TemplateChild<gtk::Stack>, | ||||||
|  | @ -72,8 +77,11 @@ mod imp { | ||||||
|             let library_action = gio::ActionEntry::builder("library") |             let library_action = gio::ActionEntry::builder("library") | ||||||
|                 .activate(move |_, _, _| { |                 .activate(move |_, _, _| { | ||||||
|                     if let Some(library) = &*obj.imp().library.borrow() { |                     if let Some(library) = &*obj.imp().library.borrow() { | ||||||
|                         let library_manager = |                         let library_manager = LibraryManager::new( | ||||||
|                             LibraryManager::new(&obj.imp().navigation_view, library); |                             &obj.imp().navigation_view, | ||||||
|  |                             library, | ||||||
|  |                             &obj.imp().process_manager, | ||||||
|  |                         ); | ||||||
|                         obj.imp().navigation_view.push(&library_manager); |                         obj.imp().navigation_view.push(&library_manager); | ||||||
|                     } |                     } | ||||||
|                 }) |                 }) | ||||||
|  | @ -135,6 +143,32 @@ mod imp { | ||||||
| 
 | 
 | ||||||
|     impl WindowImpl for Window { |     impl WindowImpl for Window { | ||||||
|         fn close_request(&self) -> glib::signal::Propagation { |         fn close_request(&self) -> glib::signal::Propagation { | ||||||
|  |             if self.process_manager.any_ongoing() { | ||||||
|  |                 let dialog = adw::AlertDialog::builder() | ||||||
|  |                     .heading(&gettext("Close window?")) | ||||||
|  |                     .body(&gettext( | ||||||
|  |                         "There are ongoing processes that will be canceled.", | ||||||
|  |                     )) | ||||||
|  |                     .build(); | ||||||
|  | 
 | ||||||
|  |                 dialog.add_responses(&[ | ||||||
|  |                     ("cancel", &gettext("Keep open")), | ||||||
|  |                     ("close", &gettext("Close window")), | ||||||
|  |                 ]); | ||||||
|  | 
 | ||||||
|  |                 dialog.set_response_appearance("close", adw::ResponseAppearance::Destructive); | ||||||
|  |                 dialog.set_close_response("cancel"); | ||||||
|  |                 dialog.set_default_response(Some("cancel")); | ||||||
|  | 
 | ||||||
|  |                 let obj = self.obj().to_owned(); | ||||||
|  |                 glib::spawn_future_local(async move { | ||||||
|  |                     if dialog.choose_future(&obj).await == "close" { | ||||||
|  |                         obj.destroy(); | ||||||
|  |                     } | ||||||
|  |                 }); | ||||||
|  | 
 | ||||||
|  |                 glib::signal::Propagation::Stop | ||||||
|  |             } else { | ||||||
|                 if let Err(err) = self.obj().save_window_state() { |                 if let Err(err) = self.obj().save_window_state() { | ||||||
|                     log::warn!("Failed to save window state: {err}"); |                     log::warn!("Failed to save window state: {err}"); | ||||||
|                 } |                 } | ||||||
|  | @ -142,6 +176,7 @@ mod imp { | ||||||
|                 glib::signal::Propagation::Proceed |                 glib::signal::Propagation::Proceed | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
|     impl ApplicationWindowImpl for Window {} |     impl ApplicationWindowImpl for Window {} | ||||||
|     impl AdwApplicationWindowImpl for Window {} |     impl AdwApplicationWindowImpl for Window {} | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue