【问题标题】:Format UITextField MM/YY as expiration将 UITextField MM/YY 格式化为过期
【发布时间】:2019-01-08 22:06:51
【问题描述】:

尝试为信用卡到期日期设置textField,以下代码正常工作只是为了改变行为,目前当您从第三个数字输入时,/ 字符将添加为过期格式。如果我想在用户输入第二个数字后添加 / 字符,例如如果用户输入 01 直接插入分隔符

open func reformatAsExpiration(_ textField: UITextField) {
        guard let string = textField.text else { return }
        let expirationString = String(ccrow.expirationSeparator)
        let cleanString = string.replacingOccurrences(of: expirationString, with: "", options: .literal, range: nil)
        if cleanString.length >= 3 {
            let monthString = cleanString[Range(0...1)]
            var yearString: String
            if cleanString.length == 3 {
                yearString = cleanString[2]
            } else {
                yearString = cleanString[Range(2...3)]
            }
            textField.text = monthString + expirationString + yearString
        } else {
            textField.text = cleanString
        }
    }

【问题讨论】:

标签: ios iphone swift


【解决方案1】:

斯威夫特 5:

这是验证输入月份和年份的解决方案

使用TextField委托方法shouldChangeCharactersIn

func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
    guard let oldText = textField.text, let r = Range(range, in: oldText) else {
        return true
    }
    let updatedText = oldText.replacingCharacters(in: r, with: string)

    if string == "" {
        if updatedText.count == 2 {
            textField.text = "\(updatedText.prefix(1))"
            return false
        }
    } else if updatedText.count == 1 {
        if updatedText > "1" {
            return false
        }
    } else if updatedText.count == 2 {
        if updatedText <= "12" { //Prevent user to not enter month more than 12
            textField.text = "\(updatedText)/" //This will add "/" when user enters 2nd digit of month
        }
        return false
    } else if updatedText.count == 5 {
        self.expDateValidation(dateStr: updatedText)
    } else if updatedText.count > 5 {
        return false
    }

    return true
}

以下函数中的验证逻辑

func expDateValidation(dateStr:String) {

    let currentYear = Calendar.current.component(.year, from: Date()) % 100   // This will give you current year (i.e. if 2019 then it will be 19)
    let currentMonth = Calendar.current.component(.month, from: Date()) // This will give you current month (i.e if June then it will be 6)

    let enteredYear = Int(dateStr.suffix(2)) ?? 0 // get last two digit from entered string as year
    let enteredMonth = Int(dateStr.prefix(2)) ?? 0 // get first two digit from entered string as month
    print(dateStr) // This is MM/YY Entered by user

    if enteredYear > currentYear {
        if (1 ... 12).contains(enteredMonth) {
            print("Entered Date Is Right")
        } else {
            print("Entered Date Is Wrong")
        }
    } else if currentYear == enteredYear {
        if enteredMonth >= currentMonth {
            if (1 ... 12).contains(enteredMonth) {
               print("Entered Date Is Right")
            } else {
               print("Entered Date Is Wrong")
            }
        } else {
            print("Entered Date Is Wrong")
        }
    } else {
       print("Entered Date Is Wrong")
    }

}

【讨论】:

  • 不要在 Swift (5) 中使用NSString。而是将NSRange 转换为Range&lt;String.Index&gt;
  • 工作完美,感谢赛凡
  • 按预期工作得很好。
【解决方案2】:

你的 textfield 委托方法 shouldChangeCharactersIn 来完成这个

 func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {

        if string == "" {
            return true
        }

        let currentText = textField.text! as NSString
        let updatedText = currentText.replacingCharacters(in: range, with: string)

        textField.text = updatedText
        let numberOfCharacters = updatedText.count
        if numberOfCharacters == 2 {
            textField.text?.append("/")
        }
        return false
    }

【讨论】:

  • 我会试试的,提前谢谢
【解决方案3】:

在您的文本字段中使用此代码确实开始为该文本字段进行编辑。

  let datePickerView = UIDatePicker()
                datePickerView.datePickerMode = .date                
                datePickerView.addTarget(self, action: #selector(handleChange(sender:)), for: .valueChanged)
                yourTextField.inputView = datePickerView

现在在选择器视图中实现此方法以处理更改

@objc func handleChange(sender: UIDatePicker){
            let dateFormatter = DateFormatter()
            dateFormatter.dateFormat = "MM/yyyy"
            yourTextfield.text = dateFormatter.string(from: sender.date)
      }

希望对你有帮助:)

【讨论】:

  • OP 只需要月份和年份进行验证
  • @Junaid 这种方法将无法正常工作,因为UIDatePicker 给予 mm/dd/yyyy
  • @A.K.M 他们无法在 datepicker 中更改日期格式?
  • @junaid 您只能更改所选值的日期格式
  • @A.K.M 是的,我已经搜索到无法更改日期选择器日期格式,它只是模式日期、日期时间等...感谢您的纠正:)
【解决方案4】:

这是从格式为 MM/YY 的文本字段计算到期日期的完整答案

var cleanString = string.replacingOccurrences(of: "/", with: "")

    if cleanString.rangeOfCharacter(from: unsupportedCharacterSet) != nil {
        return ""
    }
    
    let dateString: String
    
    if cleanString.count == 0 {
        return string
    } else if cleanString.count > 4 {
        // trim the string down to 4
        
        let reqIndex = cleanString.index(cleanString.startIndex, offsetBy: 4)
        cleanString = String(cleanString[..<reqIndex])
        
    }
    
    if cleanString.hasPrefix("0") == false && cleanString.hasPrefix("1") == false {
        dateString = "0" + cleanString
    } else {
        dateString = cleanString
    }
    let currentYear = Calendar.current.component(.year, from: Date()) % 100   // This will give you current year (i.e. if 2019 then it will be 19)
      let currentMonth = Calendar.current.component(.month, from: Date()) // This will give you current month (i.e if June then it will be 6)
    var newText = ""
    for (index, character) in dateString.enumerated() {
        print("index: \(index)")
        if index == 1 {
            let enterdMonth = Int(dateString.prefix(2)) ?? 0  // get first two digit from entered string as month
            print("enterdMonth at 1:\(enterdMonth)")
            if (1 ... 12).contains(enterdMonth){
                if enterdMonth < 10 {
                    newText = "0" + "\(enterdMonth)" + "/"
                }else {
                    newText = "\(enterdMonth)" + "/"
                }
            }else{
                
            }
        }else if index == 2 {
            if (2 ... 99).contains(Int(String(character))!) { // Entered year should be greater than 2019.
                newText.append(character)
            }else{
                
            }
        }else if index == 3 {
            print("index---: \(index)")
            let enterdYr = Int(dateString.suffix(2)) ?? 0   // get last two digit from entered string as year
            let enterdMonth = Int(dateString.prefix(2)) ?? 0  // get first two digit from entered string as month
            print("enterdYr: \(enterdYr)")
            print("currentYear: \(currentYear)")
            if (2 ... 99).contains(enterdYr) { // Entered year should be greater than 2019
                if enterdYr >= currentYear {
                    if (1 ... 12).contains(enterdMonth) {
                        if enterdMonth < 10 {
                            newText = "0" + "\(enterdMonth)" + "/" + "\(enterdYr)"
                        }else {
                            newText = "\(enterdMonth)" + "/" + "\(enterdYr)"
                        }
                        return newText
                    }
                }
            }
        }
        else {
            newText.append(character)
        }
    }
    return newText
}

【讨论】:

    【解决方案5】:

    Swift 5 格式化日期 mm/yy 的解决方案:

    首先我添加了一个观察者来检查文本的变化:

    dateTextField.addTarget(self, action: #selector(expirationDateDidChange), for: .editingChanged)
    

    这是实时格式化文本的功能:

    @objc func expirationDateDidChange() {
        let currentText = dateTextField.text ?? ""
    
        // To remove " / " when user deletes the space after '/'
        if currentText.count == 4 {
            dateTextField.text = String(currentText.prefix(2))
            return
        }
    
        var dateText = dateTextField.text?.replacingOccurrences(of: "/", with: "") ?? ""
        dateText = dateText.replacingOccurrences(of: " ", with: "")
    
        var newText = ""
        for (index, character) in dateText.enumerated() {
            if index == 1 {
                newText = "\(newText)\(character) / "
            } else {
                newText.append(character)
            }
        }
        dateTextField.text = newText
    }
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2017-09-30
      • 1970-01-01
      • 2014-01-03
      • 1970-01-01
      • 1970-01-01
      • 1970-01-01
      • 2021-07-20
      • 1970-01-01
      相关资源
      最近更新 更多