* Good start but some missing items: - Upload image isn't working - Only a single image is shown per section. Need to make this the HCollection of all images for the group * Upload still failing but now update and set are 2 different processes because I think that's better. Spacing on the add screen is still all wrong but we're getting closer * ~70% Complete TODO: - Spacing for remote portrait images is wrong & cramped - Upload image from file browser never works & produces 400 error - Show all images for an item.imageType opposed to just the first - Setting image works but produces a 400 error - Error alert looks bad * Merge with Main * URL Changes * Updating logic and confirmation screen * Lots of changes: Selecting a Remote image is now working without error and works consistently! Upload a local file is still broken Item types with multiple images is working as intended now! Overriding an image on index doesn't seem to work but it doesn't work for Web either so........ UI is way more jank but the hard parts are getting solved! * Breaking this even more with the hopes of a better tomorrow. * Getting better? * Refreshing is working but I might need to make this work mroe effiently... * 90% There! * Ability to cancel the update * Still no luck uploading images? * Stop reordering on deletion/addition * 2025 disclaimers * Uploading finally works! * Functional but messy. TODO: - Figure out better resizing if too big? - Upload from Photos - Move upload logic to imageViewModel and make RemtoeImageViewModel PagingLibraryViewModel conformant - Create a ImageInfoView for Selection & Deletion. * Now conforms to PagingLIbraryViewModel but everything else is a mess * Close! * First no all appears * Fix double pop/routerdismiss * Uploading from Photos is (Finally) Ready! * wip * Reuse PhotoPicker and Crop code. * 4/6 of the codefactor changes * Pass around the URL NOT the UIImage * Clean up ItemImageDetails types. * Make sure the ImageView mirrors the real shape of the image. Posters should be uniform but this is the selection for the image so the dimensions are important to demonstrate. * Rating Type label. * Delete confirmation dialog. * Remove double sizing. Remove Unused ViewModel. Change PhotoPicker to a checkmark instead a 1. Since there is only ever one picture selected, no need to count the images. * Get the image URL as needed. No more Truples. Localize ImageTypes. * Remove attempt at ImageInfo Poster Comformance. * Even more cleanup * Delete vs Save flip * Hide delete button * Even more cleanup * Fix tvOS build issues. * Reduce delay & remove unused comment. Should finally be ready again. * wip * Update ItemImagesView.swift * Event Only on upload failures. * Remove unnecessary ViewModel's from tvOS. * Add dismiss action to RemoteSearchResultView. While I am doing this here, fix it there. * Move From Coordinator -> .Sheet. This fixes the popping issue / delay requirement! * wip * wip * wip * wip --------- Co-authored-by: Ethan Pippin <ethanpippin2343@gmail.com>
		
			
				
	
	
		
			70 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
			
		
		
	
	
			70 lines
		
	
	
		
			2.4 KiB
		
	
	
	
		
			Swift
		
	
	
	
	
	
| //
 | |
| // Swiftfin is subject to the terms of the Mozilla Public
 | |
| // License, v2.0. If a copy of the MPL was not distributed with this
 | |
| // file, you can obtain one at https://mozilla.org/MPL/2.0/.
 | |
| //
 | |
| // Copyright (c) 2025 Jellyfin & Jellyfin Contributors
 | |
| //
 | |
| 
 | |
| import Foundation
 | |
| import Nuke
 | |
| 
 | |
| extension ImagePipeline {
 | |
| 
 | |
|     enum Swiftfin {}
 | |
| 
 | |
|     static func cacheKey(for url: URL) -> String? {
 | |
|         guard var components = url.components else { return nil }
 | |
| 
 | |
|         var maxWidthValue: String?
 | |
| 
 | |
|         if let maxWidth = components.queryItems?.first(where: { $0.name == "maxWidth" }) {
 | |
|             maxWidthValue = maxWidth.value
 | |
|             components.queryItems = components.queryItems?.filter { $0.name != "maxWidth" }
 | |
|         }
 | |
| 
 | |
|         guard let newURL = components.url, let urlSHA = newURL.pathAndQuery?.sha1 else { return nil }
 | |
| 
 | |
|         if let maxWidthValue {
 | |
|             return urlSHA + "-\(maxWidthValue)"
 | |
|         } else {
 | |
|             return urlSHA
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     func removeItem(for url: URL) {
 | |
|         let request = ImageRequest(url: url)
 | |
|         cache.removeCachedImage(for: request)
 | |
|         cache.removeCachedData(for: request)
 | |
| 
 | |
|         guard let dataCacheKey = Self.cacheKey(for: url) else { return }
 | |
|         configuration.dataCache?.removeData(for: dataCacheKey)
 | |
|     }
 | |
| }
 | |
| 
 | |
| extension ImagePipeline.Swiftfin {
 | |
| 
 | |
|     /// The default `ImagePipeline` to use for images that are typically posters
 | |
|     /// or server user images that should be presentable with an active connection.
 | |
|     static let posters: ImagePipeline = ImagePipeline(delegate: SwiftfinImagePipelineDelegate()) {
 | |
|         $0.dataCache = DataCache.Swiftfin.posters
 | |
|     }
 | |
| 
 | |
|     /// The `ImagePipeline` used for images that should have longer lifetimes and usable
 | |
|     /// without a connection, likes local user profile images and server splashscreens.
 | |
|     static let local: ImagePipeline = ImagePipeline(delegate: SwiftfinImagePipelineDelegate()) {
 | |
|         $0.dataCache = DataCache.Swiftfin.local
 | |
|     }
 | |
| 
 | |
|     /// An `ImagePipeline` for images to prevent more important images from losing their cache.
 | |
|     static let other: ImagePipeline = ImagePipeline(configuration: .withURLCache)
 | |
| }
 | |
| 
 | |
| final class SwiftfinImagePipelineDelegate: ImagePipelineDelegate {
 | |
| 
 | |
|     func cacheKey(for request: ImageRequest, pipeline: ImagePipeline) -> String? {
 | |
|         guard let url = request.url else { return nil }
 | |
|         return ImagePipeline.cacheKey(for: url)
 | |
|     }
 | |
| }
 |