仅显示来自 Swift/SwiftUI 的 Date 对象的时间 Text.DateStyle

问题描述

我一直在使用结束时间显示音频持续时间。一位优秀的 Stack Overflow 开发人员提出了这个解决方案。

因此,我已将其添加到我的视图中:

struct ContentView: View {
    @Observedobject var viewmodel = Myviewmodel() //<- here
    var body: some View {
        vstack {
            
            Text(viewmodel.endTime) //<- here
            
        }
    }
}

这是我的 viewmodel 中的代码

class Myviewmodel: ObservableObject {
    @Published var endDate: Date? 
    var endTime: String{
        endDate == nil ? "":endDate!.description 
    }

    func play(){
        let path = Bundle.main.path(forResource: "song",ofType:"mp3")!
        let url = URL(fileURLWithPath: path)
        do {
            player = try AVAudioPlayer(contentsOf: url)
            
            endDate = Date() + player.duration
            if player.isPlaying{
                
                player.pause()
                
            }
            else{
                player.play()
                
            }
            isPlaying = player.isPlaying
        }catch{print("error")}

但是我得到了错误的时间(不是系统时间)但是日期正确。我只想显示正确的时间而不显示日期。

解决方法

从 iOS 14+ 开始,Text 有一个名为 style 的属性,用于显示日期。

Text.DateStyle

用于显示日期的预定义样式。

init(Date,style: Text.DateStyle)

创建一个使用特定样式显示本地化日期和时间的实例。 https://developer.apple.com

您不需要使用 DateFormatter。使用 Text()time 作为样式可能就足够了。

Text(Date(),style: .time) 

就你而言:

Text(viewModel.endTime,style: .time)
,

要显示格式化的日期,您可能需要使用 DateFormatter

let dateFormatter = DateFormatter()
dateFormatter.dateStyle = .medium
dateFormatter.timeStyle = .none
 
let date = Date(timeIntervalSinceReferenceDate: 118800)
 
// US English Locale (en_US)
dateFormatter.locale = Locale(identifier: "en_US")
print(dateFormatter.string(from: date)) // Jan 2,2001

来源:https://developer.apple.com/documentation/foundation/dateformatter

(请注意,众所周知,创建 DateFormatter 是一项代价高昂的操作,因此请注意不要过于频繁地创建它们)

您的视图模型中也有一些逻辑错误。以下是如何处理此问题的示例:

struct ContentView : View {
    @ObservedObject private var vm = MyViewModel()
    
    var body: some View {
        Button(action:{
            vm.play()
        } ) {
            Text(vm.isPlaying ? "Stop" : "Play")
        }
        Text(vm.endTime)
    }
}

class MyViewModel: ObservableObject {
    @Published var endDate: Date?
    @Published var isPlaying = false
    
    var player = AVAudioPlayer()
    var dateFormatter = DateFormatter()
    
    var endTime: String{
        guard let endDate = endDate else {
            return ""
        }
        dateFormatter.dateStyle = .none
        dateFormatter.timeStyle = .medium
        return dateFormatter.string(from: endDate)
    }

    func play(){
        do{
            if player.isPlaying {
                player.pause()
            } else {
                let path = Bundle.main.path(forResource: "All Of Me Final Mix",ofType:"mp3")!
                let url = URL(fileURLWithPath: path)
                player = try AVAudioPlayer(contentsOf: url)
                endDate = Date() + player.duration - player.currentTime
                player.play()
            }
            isPlaying = player.isPlaying
        } catch {
            print("error: \(error)")
        }
    }
}