From 5fc881156ec3f4c5b08f999c37c330225a448cbe Mon Sep 17 00:00:00 2001 From: eriedaberrie Date: Tue, 12 Nov 2024 11:04:34 -0800 Subject: [PATCH] feat: sync-music: copy album covers --- sync-music/package.lisp | 2 ++ sync-music/sync-music.lisp | 32 +++++++++++++++++++++++++++++++- sync-music/utils.lisp | 4 ++++ 3 files changed, 37 insertions(+), 1 deletion(-) diff --git a/sync-music/package.lisp b/sync-music/package.lisp index 4a1f83e..af41221 100644 --- a/sync-music/package.lisp +++ b/sync-music/package.lisp @@ -75,6 +75,8 @@ #:*dumb-cue-copy-p* #:*general-extensions* #:*meta-extensions* + #:*image-extensions* + #:*album-cover-names* #:*action-perform-output* #:action-describe #:sync-music)) diff --git a/sync-music/sync-music.lisp b/sync-music/sync-music.lisp index a7201a9..1e8bebc 100644 --- a/sync-music/sync-music.lisp +++ b/sync-music/sync-music.lisp @@ -110,6 +110,33 @@ extension equality." :do (fset:adjoinf actions current-action) :finally (return actions))))))) +(defun get-image-copy-action (origin-files target &optional (test #'string-equal)) + "Determine which image file to copy from ORIGIN-FILES, if any." + (when-let ((image-files (remove-if (lambda (f) + (let ((f-type (pathname-type f))) + (notany (lambda (type) + (funcall test f-type type)) + *image-extensions*))) + origin-files))) + (let ((image-file (or (find-if (lambda (f) + (let ((f-name (pathname-name f))) + (some (lambda (name) + (funcall test f-name name)) + *album-cover-names*))) + image-files) + (first image-files)))) + (make-copy-action :origin image-file + :target (pathname-normalize-unicode + (make-pathname + :name "cover" + :type (let ((type (string-downcase + (pathname-type image-file)))) + ;; NOTE: Auxio ignores .jpeg + (if (equal type "jpeg") + "jpg" + type)) + :defaults target)))))) + (defun get-all-copy-actions (origin target &optional (depth 0)) "Recursively iterate ORIGIN for files to copy. @@ -125,7 +152,10 @@ actions that need performing in order to store the targets as second value." (fset:unionf files cue-copy-actions) (dolist (file (uiop:directory-files origin)) (when-let ((action (make-file-copy-action file target))) - (fset:adjoinf files action)))))) + (fset:adjoinf files action)))) + (unless (fset:empty? files) + (when-let ((image-action (get-image-copy-action origin-files target))) + (fset:adjoinf files image-action))))) (dolist (directory (uiop:subdirectories origin)) (let ((target-directory (make-pathname :directory (append (pathname-directory target) diff --git a/sync-music/utils.lisp b/sync-music/utils.lisp index 04597b7..c4cb6b2 100644 --- a/sync-music/utils.lisp +++ b/sync-music/utils.lisp @@ -17,6 +17,10 @@ "General file extensions that should be copied over.") (defparameter *meta-extensions* '("cue" "m3u" "m3u8") "File extensions where *.flac text needs to be replaced.") +(defparameter *image-extensions* '("png" "jpg" "jpeg" "webp") + "File extensions for image files.") +(defparameter *album-cover-names* '("cover" "front" "folder") + "File names of album cover files.") (defparameter *invalid-char-scanner* (ppcre:create-scanner "[\"*/:<>?\\\\|]") "Scan for any characters not allowed in an Android/NTFS file name.")