Swift: Как я могу сделать uipickerview, который показывает дату и время

Я могу показать время и изменить временные интервалы, но я не знаю, как получить временной интервал с датами, подобными фотографии ниже, если я не напишу это жестко. Стараюсь избегать использования кокапод, если мне не нужно. Прямо сейчас я жестко кодирую это, но только время, я не могу получить дату без другого UIPicker,

Код, относящийся к UIPickerView обсуждаемый:

private var deliveryTimesPicker = UIPickerView()
private var deliveryTimes = ["10 AM - 11 AM", "11 AM - 12 PM", "12 PM - 1 PM", "1 PM - 2 PM", "2 PM - 3 PM", "3 PM - 4 PM", "4 PM - 5 PM", "5 PM - 6 PM", "6 PM - 7 PM", "7 PM - 8 PM", "8 PM - 9 PM", "9 PM - 10 PM", "10 PM - 11 PM"]
private var selectedDeliveryTime = 0

override func viewDidLoad() {
    ...
    self.deliveryTimeTF.text = self.deliveryTimes.first!

    self.doneToolbar.barStyle = .default
    self.doneToolbar.isTranslucent = true
    self.doneToolbar.isUserInteractionEnabled = true
    self.doneToolbar.setItems([
        UIBarButtonItem(title: "Cancel", style: .plain, target: self, action: #selector(self.dismissKeyboard)),
        UIBarButtonItem(barButtonSystemItem: .flexibleSpace, target: nil, action: nil),
        UIBarButtonItem(title: "Done", style: .done, target: self, action: #selector(self.pickerViewDone))
        ], animated: true)
    self.doneToolbar.sizeToFit()

    self.deliveryTimesPicker.delegate = self
    self.deliveryTimesPicker.dataSource = self
    self.deliveryTimesPicker.tag = 0
    self.deliveryTimesPicker.selectRow(self.selectedDeliveryTime, inComponent: 0, animated: true)
    self.deliveryTimeTF.inputView = self.deliveryTimesPicker
    self.deliveryTimeTF.inputAccessoryView = self.doneToolbar
}


@objc func pickerViewDone() {
    if self.deliveryTimeTF.isEditing {
        self.deliveryTimeTF.text = self.deliveryTimes[self.selectedDeliveryTime]
    }
    dismissKeyboard()
}

extension RequestViewController: UIPickerViewDelegate {
    func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
        if pickerView.tag == 0 {
            self.selectedDeliveryTime = row
            return self.deliveryTimes[self.selectedDeliveryTime]
        }
    }

    func pickerView(_ pickerView: UIPickerView, didSelectRow row: Int, inComponent component: Int) {
        if pickerView.tag == 0 {
             self.selectedDeliveryTime = row
        }
    }

}

Насколько моя:

введите описание изображения здесь

Как я хочу, чтобы это было (цветовая тема важна не только для ее функции): Из приложения skipthedishes

2 ответа

Я не знаю, как получить временные рамки с датами, как на фото ниже, если я не кодирую их жестко.

Тогда вы должны прочитать Date, DateInterval, DateComponentsи т. д. Выберите начальную дату и используйте ее для создания списка пар начальных и конечных дат. Вот некоторый псевдокод:

let start = Date()
var timeframes = (Date,Date)[]
for i in (0..9) {
    let t1 = Date(timeInterval: (i * secondsBetweenStarts) since:start)
    let t2 = Date(timeInterval: timeFrameDuration since:t1)
    let timeframe = (t1, t2)
    timeframes.append(timeframe)
}

Вам нужно будет заполнить значения для secondsBetweenStarts а также timeFrameDuration конечно. И есть множество других способов построения дат - например, использование компонентов даты позволит легко создавать периоды, которые начинаются и заканчиваются на часовых границах. В любом случае, идея состоит в том, чтобы просто создать массив времени начала и окончания. Оттуда должно быть легко сгенерировать список строк, которые описывают эти временные рамки, и вы можете использовать этот список строк в средстве выбора.

Всегда избегайте использования тегов для идентификации вещей, сборщик будет правильным.

Этот подход возвращает два элемента в сборщике, из которых пользователь может выбрать, вам нужно только настроить первую дату со временем, с которого вы хотите, чтобы она начиналась. Пользователь сможет выбрать дату доставки и время доставки, когда значение средства выбора изменится, код предоставит вам выбранную дату с диапазоном времени доставки, вы также можете использовать логику didSelectRow в своей функции pickerViewDone с помощью средства выбора var (deliveryTimesPicker) в качестве источника.

Единственное, что я не сделал, - это проверял значения опций, и вам также необходимо убедиться, что средство выбора имеет как минимум два компонента (picker.numberOfComponents) и правильно ли сгенерированы даты.

let deliveryDateDayFormatter = DateFormatter()
let deliveryDateTimeFormatter = DateFormatter()

let deliveryDateDays: [String] = {
    deliveryDateDayFormatter.dateFormat = "E, d MMM y"
    deliveryDateTimeFormatter.dateFormat = "h:mm a"

    var deliveryDate = Date()
    var deliveryDateDaysArray: [String] = []
    deliveryDateDaysArray.append("Today")

    for dateIndex in 0 ... 8 {
        deliveryDate = Calendar.current.date(byAdding: .day, value: dateIndex, to: deliveryDate)!

        let deliveryDateDay = deliveryDateDayFormatter.string(from: deliveryDate)

        deliveryDateDaysArray.append("\(deliveryDateDay)")
    }
    return deliveryDateDaysArray
}()


let deliveryDateTimes: [String] = {
    deliveryDateDayFormatter.dateFormat = "E, d MMM y"
    deliveryDateTimeFormatter.dateFormat = "h:mm a"

    var deliveryDate = Date()
    var deliveryDateTimesArray: [String] = []
    deliveryDateTimesArray.append("ASAP")

    for _ in 0 ... 8 {
        let deliveryDateTimeFrom = deliveryDateTimeFormatter.string(from: deliveryDate)

        deliveryDate = Calendar.current.date(byAdding: .minute, value: 30, to: deliveryDate)!

        let deliveryDateTimeTo = deliveryDateTimeFormatter.string(from: deliveryDate)

        deliveryDateTimesArray.append("\(deliveryDateTimeFrom) - \(deliveryDateTimeTo)")
    }
    return deliveryDateTimesArray
}()

func numberOfComponents(in pickerView: UIPickerView) -> Int {
    return 2
}

func pickerView(_ pickerView: UIPickerView, numberOfRowsInComponent component: Int) -> Int {
    var pickerComponents: [[String]] = []

    pickerComponents.append(deliveryDateDays)
    pickerComponents.append(deliveryDateTimes)

    return pickerComponents[component].count
}

func pickerView(_ pickerView: UIPickerView, didSelectRow row: Int, inComponent component: Int) {
    let dateFormatter = DateFormatter()
    dateFormatter.dateFormat = "E, d MMM y h:mm a"

    let selectedDeliveryDayIndex = pickerView.selectedRow(inComponent: 0)
    let selectedDeliveryTimeIndex = pickerView.selectedRow(inComponent: 1)

    let selectedDeliveryDay = deliveryDateDays[selectedDeliveryDayIndex]
    let selectedDeliveryTime = deliveryDateTimes[selectedDeliveryTimeIndex]

    let fullTimeRange = selectedDeliveryTime.components(separatedBy: " - ")

    let timeFrom   = fullTimeRange[0]
    let timeTo = fullTimeRange[1]

    let selectedDateStringFrom = "\(selectedDeliveryDay) \(timeFrom)"
    let selectedDateStringTo = "\(selectedDeliveryDay) \(timeTo)"

    //Delivery date from
    let selectedDateTimeFrom = dateFormatter.date(from: selectedDateStringFrom)!

    //Delivery date to
    let selectedDateTimeTo = dateFormatter.date(from: selectedDateStringTo)!

    //Delivery range
    var deliveryDate = (selectedDateTimeFrom,selectedDateTimeTo)

    print(selectedDateTimeFrom)
    print(selectedDateTimeTo)

    print(deliveryDateDays)
    print(deliveryDateTimes)
}

func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
    var pickerComponents: [[String]] = []

    pickerComponents.append(deliveryDateDays)
    pickerComponents.append(deliveryDateTimes)

    return pickerComponents[component][row]
}

Основываясь на вашем коде, вы можете использовать вид клавиатуры следующим образом:

protocol DeliveryTimeKeyboardDelegate  {
    func closeKeyboard()
    func validate(deliveryTime: (Date, Int)?)
}

class DeliveryTimeKeyboard: UIStackView {
    let datePickerView = UIDatePicker()
    let hourPickerView = UIPickerView()
    var delegate: DeliveryTimeKeyboardDelegate?
    private var deliveryTimes = ["10 AM - 11 AM", "11 AM - 12 PM", "12 PM - 1 PM", "1 PM - 2 PM", "2 PM - 3 PM", "3 PM - 4 PM", "4 PM - 5 PM", "5 PM - 6 PM", "6 PM - 7 PM", "7 PM - 8 PM", "8 PM - 9 PM", "9 PM - 10 PM", "10 PM - 11 PM"]

    var cancelButton: UIBarButtonItem?
    var clearButton: UIBarButtonItem?
    var doneButton: UIBarButtonItem?

    var deliveryTime: (Date, Int) {
        let day = datePickerView.date
        let period = hourPickerView.selectedRow(inComponent: 0)
        return (day, period)
    }

    var toolbar: UIToolbar {
        let toolBar = UIToolbar()

        toolBar.barStyle = UIBarStyle.default
        toolBar.isTranslucent = true
        toolBar.tintColor = UIColor.black
        toolBar.sizeToFit()

        let cancelButton = UIBarButtonItem(title: "Cancel", style: UIBarButtonItemStyle.plain, target: self, action: #selector(self.cancelAction))
        let clearButton = UIBarButtonItem(barButtonSystemItem: .trash, target: self, action: #selector(self.clearAction))
        let doneButton = UIBarButtonItem(title: "Done", style: UIBarButtonItemStyle.plain, target: self, action: #selector(self.doneAction))
        let spaceButton = UIBarButtonItem(barButtonSystemItem: UIBarButtonSystemItem.flexibleSpace, target: nil, action: nil)
        self.cancelButton = cancelButton
        self.clearButton = clearButton
        self.doneButton = doneButton

        toolBar.setItems([cancelButton, spaceButton, clearButton, spaceButton, doneButton], animated: false)
        toolBar.isUserInteractionEnabled = true
        return toolBar
    }

    convenience init() {
        self.init(frame:  CGRect(x: 0, y: 0, width: 1, height: 216))
        addArrangedSubview(datePickerView)
        addArrangedSubview(hourPickerView)
        datePickerView.datePickerMode = .date
        hourPickerView.delegate = self
        hourPickerView.dataSource = self

        axis = .horizontal
    }


    @objc func cancelAction() {
        delegate?.closeKeyboard()
    }

    @objc func clearAction() {
        delegate?.validate(deliveryTime: nil)
    }

    @objc func doneAction() {
        delegate?.validate(deliveryTime: deliveryTime)
    }
}

extension DeliveryTimeKeyboard: UIPickerViewDelegate {
    func pickerView(_ pickerView: UIPickerView, titleForRow row: Int, forComponent component: Int) -> String? {
        guard row < deliveryTimes.count else { return nil }
        return deliveryTimes[row]
    }
}
extension DeliveryTimeKeyboard: UIPickerViewDataSource {
    func numberOfComponents(in pickerView: UIPickerView) -> Int {
        return 1
    }

    func pickerView(_ pickerView: UIPickerView, numberOfRowsInComponent component: Int) -> Int {
        return deliveryTimes.count
    }
}

class ViewController: UIViewController {

    @IBOutlet weak var inputDeliveryTime: UITextField!

    override func viewDidLoad() {
        super.viewDidLoad()
        let deliveryTimeKeyboard = DeliveryTimeKeyboard()
        inputDeliveryTime.inputView = deliveryTimeKeyboard
        inputDeliveryTime.inputAccessoryView = deliveryTimeKeyboard.toolbar
    }
}
Другие вопросы по тегам