swiftwatchkitapple-watch-complicationclockkitwatchos-3

watchOS - Complication shows previous entry


I'm creating a watchOS 3 complication that shows departure times from a public transit service. I've created a data model with an array that contains Train objects with a stationName (String) and departureTime (NSDate).

I've implemented the getCurrentTimelineEntry() method and the entries are showing on the watch. The problem is that the watch shows only the previous entry. For example, I've the following departure times:

 Train(startStation: "Station name", endStation: "Station name", departureTime: stringToDate(dateString: "2016-06-20 14:00")),
 Train(startStation: "Station name", endStation: "Station name", departureTime: stringToDate(dateString: "2016-06-20 14:30")),
 Train(startStation: "Station name", endStation: "Station name", departureTime: stringToDate(dateString: "2016-06-20 14:45")),

If the current time is 14:10, the first entry (with time 14:00) is still showing on the watch. Until the current time is 14:30, then that entry is showing up. If the current time is 14:10, I would want to see the 14:30 departure time on my watch.

Can anybody help me with this or point me in the right direction?

func getCurrentTimelineEntry(for complication: CLKComplication, withHandler handler: ((CLKComplicationTimelineEntry?) -> Void)) {
    if let train = dataProvider.getTrains().nextTrain() {
        handler(timelineEntryForTrain(train: train))
    } else {
        handler(nil)
    }
}


extension Array where Element : OnRailable {
    func nextTrain() -> Element?{
        let now = NSDate()
        for d in self {
            if now.compare(d.departureTime) == .orderedAscending{
                return d
            }
        }

        return nil
    }
}

Solution

  • You need to set the timeline date for each entry to be one minute after the previous departure's date. For example:

    This will produce the desired effect you want, by making the upcoming departure be the current timeline entry, one minute after the previous departure:

    This approach is explained in the WWDC 2015 Creating Complications with ClockKit session. In terms of specifying the timeline dates for events, the presenter mentions how

    We should put the templates at the end of the previous event so you have adequate time to get to your next event.

    Now, the naive solution which Paul mentioned in the context of a calendar complication would be to use the match start date to be the date of our timeline entry, but that would have the drawback that it has for the calendar as well which is you wouldn't be able to look at your complication to see what game is about to start.

    You would only be able to see what game has already started.

    So we actually want to do the same thing Paul did with the calendar and move all of these entries farther forward.

    We will have each entry start at the time when the previous match ended.

    In your case, each entry would start right after the time when the previous train departed.

    How to implement your complication:

    If your realtime departure schedule changes (e.g., due to some delay), you can reload the timeline to change any of the upcoming departure times.