jellyflood/Swiftfin/Views/AdminDashboardView/ServerDevices/DeviceDetailsView/DeviceDetailsView.swift
Joe Kribs 0025422634
[iOS & tvOS] Upgrade SDK to 10.10 (#1463)
* Buildable!

* Update file names.

* Default sort to sort name NOT name.

* SessionInfoDto vs SessionInfo

* Targetting

* Fix many invalid `ItemSortBy` existing. Will need to revisit later to see which can still be used!

* ExtraTypes Patch.

* Move from Binding to OnChange. Tested and Working.

* Update README.md

Update README to use 10.10.6. Bumped up from 10.8.13

* Update to Main on https://github.com/jellyfin/jellyfin-sdk-swift.git

* Now using https://github.com/jellyfin/jellyfin-sdk-swift.git again!

* Paths.getUserViews() userId moved to parameters

* Fix ViewModels where -Dto suffixes were removed by https://github.com/jellyfin/Swiftfin/pull/1465 auto-merge.

* SupportedCaseIterable

* tvOS supportedCases fixes for build issue.

* cleanup

* update API to 0.5.1 and correct VideoRangeTypes.

* Remove deviceProfile.responseProfiles = videoPlayer.responseProfiles

* Second to last adjustment:
Resolved: // TODO: 10.10 - Filter to only valid SortBy's for each BaseItemKind.
Last outstanding item: // TODO: 10.10 - What should authenticationProviderID & passwordResetProviderID be?

* Trailers itemID must precede userID

* Force User Policy to exist.

---------

Co-authored-by: Ethan Pippin <ethanpippin2343@gmail.com>
2025-04-06 23:42:47 -04:00

113 lines
3.0 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 Defaults
import JellyfinAPI
import SwiftUI
struct DeviceDetailsView: View {
// MARK: - Current Date
@CurrentDate
private var currentDate: Date
// MARK: - State & Environment Objects
@EnvironmentObject
private var router: AdminDashboardCoordinator.Router
@StateObject
private var viewModel: DeviceDetailViewModel
// MARK: - Custom Name Variable
@State
private var temporaryCustomName: String
// MARK: - Dialog State
@State
private var isPresentingSuccess: Bool = false
// MARK: - Error State
@State
private var error: Error?
// MARK: - Initializer
init(device: DeviceInfoDto) {
_viewModel = StateObject(wrappedValue: DeviceDetailViewModel(device: device))
self.temporaryCustomName = device.customName ?? device.name ?? ""
}
// MARK: - Body
var body: some View {
List {
if let userID = viewModel.device.lastUserID,
let userName = viewModel.device.lastUserName
{
let user = UserDto(id: userID, name: userName)
AdminDashboardView.UserSection(
user: user,
lastActivityDate: viewModel.device.dateLastActivity
) {
router.route(to: \.userDetails, user)
}
}
CustomDeviceNameSection(customName: $temporaryCustomName)
AdminDashboardView.DeviceSection(
client: viewModel.device.appName,
device: viewModel.device.name,
version: viewModel.device.appVersion
)
CapabilitiesSection(device: viewModel.device)
}
.navigationTitle(L10n.device)
.onReceive(viewModel.events) { event in
switch event {
case let .error(eventError):
UIDevice.feedback(.error)
error = eventError
case .setCustomName:
UIDevice.feedback(.success)
isPresentingSuccess = true
}
}
.topBarTrailing {
if viewModel.backgroundStates.contains(.updating) {
ProgressView()
}
Button(L10n.save) {
UIDevice.impact(.light)
if viewModel.device.id != nil {
viewModel.send(.setCustomName(temporaryCustomName))
}
}
.buttonStyle(.toolbarPill)
.disabled(temporaryCustomName == viewModel.device.customName)
}
.alert(
L10n.success.text,
isPresented: $isPresentingSuccess
) {
Button(L10n.dismiss, role: .cancel)
} message: {
Text(L10n.customDeviceNameSaved(temporaryCustomName))
}
.errorMessage($error)
}
}