SwiftUI изменяет формат вывода текста, используя как .timer

Есть ли способ изменить выходной формат Text с помощью init(_ date: Date, style: Text.DateStyle)?

Используя .timer, вывод такой: 0:42, но я хочу что-то вроде 00:00:42.


Задний план

Я хочу создать виджет (iOS 14), в котором работает таймер, и, как я думаю, не рекомендуется запускать обновление виджета каждую секунду, и это может даже не работать надежно, по крайней мере, это не так, как виджет имеет отступ быть использованным. Поэтому я подумал об использовании этой предопределенной функции таймераText.

Я новичок в SwiftUI и еще не знаю всех возможностей. Так, может быть, я мог бы сам создать подобное настраиваемое текстовое представление? Есть идеи, как этого добиться?

Или спросить по-другому: есть ли способ самостоятельно создавать такие самообновляющиеся компоненты, которые также работают в виджете iOS 14? (Похоже на использованиеTimer.publish для обновления View не работает в виджете iOS 14)

0 ответов

Нет решения => Обходной путь

На сегодняшний день, похоже, для этого нет подходящего решения. Но поскольку интерес к "решению" или обходному пути, похоже, есть, я просто опубликую то, что у меня получилось, пока.

Код

В основном я просто вручную обновляю Text один раз в секунду и вычислите разницу от исходной даты до "сейчас".

struct SomeView: View {
    let referenceDate = Date() // <- Put your start date here

    @State var duration: Int = 0
    let timer = Timer.publish(every: 1, on: .current, in: .common).autoconnect()
    
    var body: some View {
        Text(self.duration.formatted(allowedUnits: [.hour, .minute, .second]) ?? "")
            .onReceive(self.timer) { _ in
                self.duration = referenceDate.durationToNow ?? 0
            }

    }
}

extension Date {
    var durationToNow: Int? {
        return Calendar.current
            .dateComponents([.second], from: self, to: Date())
            .second
    }
}

extension Int {
    func formatted(allowedUnits: NSCalendar.Unit = [.hour, .minute]) -> String? {
        let formatter = DateComponentsFormatter()
        formatter.allowedUnits = allowedUnits
        formatter.zeroFormattingBehavior = .pad
        return formatter.string(from: DateComponents(second: self))
    }
}

Ограничения WidgetKit

К сожалению, это не работает для WidgetKit, поскольку таймер не запускается или, по крайней мере, пользовательский интерфейс не обновляется. Итак, я закончил, отображая только минуты в виджете (что вроде нормально для моей цели) и просто установил обновление на "временной шкале" для каждой минуты.

Другие вопросы по тегам