mirror of
https://github.com/immich-app/immich.git
synced 2026-03-06 08:53:43 -05:00
* feat(mobile): add playbackStyle to native sync API Adds a `playbackStyle` field to `PlatformAsset` in the pigeon sync API so native platforms can communicate the asset's playback style (image, video, animated, livePhoto) to Flutter during sync. - Add `playbackStyleValue` computed property to `PHAsset` extension (iOS) - Populate `playbackStyle` in `toPlatformAsset()` and the full-sync path - Update generated Dart/Kotlin/Swift files * fix(tests): add playbackStyle to local asset test cases * fix(tests): update playbackStyle to use integer values in local sync tests * feat(mobile): extend playbackStyle enum to include videoLooping * Update PHAssetExtensions.swift Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * fix(playback): simplify playbackStyleValue implementation by removing iOS version check * feat(android): implement proper playbackStyle detection * add PlatformAssetPlaybackStyle enum * linting --------- Co-authored-by: Mert <101130780+mertalev@users.noreply.github.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com>
100 lines
2.7 KiB
Swift
100 lines
2.7 KiB
Swift
import Photos
|
|
|
|
extension PHAsset {
|
|
var platformPlaybackStyle: PlatformAssetPlaybackStyle {
|
|
switch playbackStyle {
|
|
case .image: return .image
|
|
case .imageAnimated: return .imageAnimated
|
|
case .livePhoto: return .livePhoto
|
|
case .video: return .video
|
|
case .videoLooping: return .videoLooping
|
|
@unknown default: return .unknown
|
|
}
|
|
}
|
|
|
|
func toPlatformAsset() -> PlatformAsset {
|
|
return PlatformAsset(
|
|
id: localIdentifier,
|
|
name: title,
|
|
type: Int64(mediaType.rawValue),
|
|
createdAt: creationDate.map { Int64($0.timeIntervalSince1970) },
|
|
updatedAt: modificationDate.map { Int64($0.timeIntervalSince1970) },
|
|
width: Int64(pixelWidth),
|
|
height: Int64(pixelHeight),
|
|
durationInSeconds: Int64(duration),
|
|
orientation: 0,
|
|
isFavorite: isFavorite,
|
|
adjustmentTime: adjustmentTimestamp,
|
|
latitude: location?.coordinate.latitude,
|
|
longitude: location?.coordinate.longitude,
|
|
playbackStyle: platformPlaybackStyle
|
|
)
|
|
}
|
|
|
|
var title: String {
|
|
return filename ?? originalFilename ?? "<unknown>"
|
|
}
|
|
|
|
var filename: String? {
|
|
return value(forKey: "filename") as? String
|
|
}
|
|
|
|
var adjustmentTimestamp: Int64? {
|
|
if let date = value(forKey: "adjustmentTimestamp") as? Date {
|
|
return Int64(date.timeIntervalSince1970)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// This method is expected to be slow as it goes through the asset resources to fetch the originalFilename
|
|
var originalFilename: String? {
|
|
return getResource()?.originalFilename
|
|
}
|
|
|
|
func getResource() -> PHAssetResource? {
|
|
let resources = PHAssetResource.assetResources(for: self)
|
|
|
|
let filteredResources = resources.filter { $0.isMediaResource && isValidResourceType($0.type) }
|
|
|
|
guard !filteredResources.isEmpty else {
|
|
return nil
|
|
}
|
|
|
|
if filteredResources.count == 1 {
|
|
return filteredResources.first
|
|
}
|
|
|
|
if let currentResource = filteredResources.first(where: { $0.isCurrent }) {
|
|
return currentResource
|
|
}
|
|
|
|
if let fullSizeResource = filteredResources.first(where: { isFullSizeResourceType($0.type) }) {
|
|
return fullSizeResource
|
|
}
|
|
|
|
return nil
|
|
}
|
|
|
|
private func isValidResourceType(_ type: PHAssetResourceType) -> Bool {
|
|
switch mediaType {
|
|
case .image:
|
|
return [.photo, .alternatePhoto, .fullSizePhoto].contains(type)
|
|
case .video:
|
|
return [.video, .fullSizeVideo, .fullSizePairedVideo].contains(type)
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
|
|
private func isFullSizeResourceType(_ type: PHAssetResourceType) -> Bool {
|
|
switch mediaType {
|
|
case .image:
|
|
return type == .fullSizePhoto
|
|
case .video:
|
|
return type == .fullSizeVideo
|
|
default:
|
|
return false
|
|
}
|
|
}
|
|
}
|