繁体   English   中英

如何设置 UITextField 的字母间距

[英]How to set letter spacing of UITextField

我有一个应用程序,用户必须在其中键入一个四位数的密码。 所有数字必须彼此保持一定距离。

如果PinTextFieldUIView的子类,有没有办法做到这一点? 我知道在ViewController您可以使用UITextFieldTextDidChangeNotification并为每个更改设置属性文本。 通知似乎在UIView不起作用。

另外,我想知道如果要设置UITextField文本的字母间距,是否没有比为每次更新制作属性字符串更简单的方法?

正确间距:

具有正确间距的 TextField

错误的间距:

间距错误的文本字段

不需要去attributedText ,老实说,用修改后的间距实现是一团糟。 我一合上键盘,间距就消失了,这促使我进一步挖掘。

每个 UITextField 都有一个名为defaultTextAttributes的属性,根据 Apple 的说法,它“返回具有默认值的文本属性字典”。 . Apple 文档还说“此属性将指定的属性应用于文本字段的整个文本”

只需在您的代码中找到合适的位置,通常是初始化文本字段的位置,然后复制并粘贴以下内容。

在 Swift 3.0 中回答

textfield.defaultTextAttributes.updateValue(spacing, forKey: NSKernAttributeName)

其中间距是 CGFloat 类型。 例如 2.0

这也适用于不同的字体。

干杯!!


最新的语法似乎是:

 yourField.defaultTextAttributes.updateValue(36.0, 
     forKey: NSAttributedString.Key.kern)

这就是最终为每次更改设置字距的方法

    textField.addTarget(self, action: "textFieldDidChange", forControlEvents: .EditingChanged)

    func textFieldDidChange () {    
        let attributedString = NSMutableAttributedString(string: textField.text)
        attributedString.addAttribute(NSKernAttributeName, value: 5, range: NSMakeRange(0, count(textField.text)))
        attributedString.addAttribute(NSFontAttributeName, value: font, range: NSMakeRange(0, count(textField.text)))
        attributedString.addAttribute(NSForegroundColorAttributeName, value: UIColor.blackColor(), range: NSMakeRange(0, count(textField.text)))

        textField.attributedText = attributedString
    }

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

        if count(textField.text) < 4 {
            return true
            // Else if 4 and delete is allowed
        }else if count(string) == 0 {
            return true
            // Else limit reached
        }else{
            return false
        }
    }

然而问题仍然存在,因为不同的数字具有不同的宽度,我将回到为每个数字制作一个UITextField

使用UITextFielddefaultTextAttributes属性。 它将为您处理到NSAttributedString的转换并应用您设置的属性。 例如:

    NSMutableDictionary *attrs = [self.textField.defaultTextAttributes mutableCopy];
[attrs addEntriesFromDictionary:@{
    NSKernAttributeName: @18,
    NSUnderlineColorAttributeName: [UIColor grayColor],
    NSUnderlineStyleAttributeName: @(NSUnderlineStyleSingle | NSUnderlinePatternDash)
}];
self.textField.defaultTextAttributes = attrs;

不太确定任何其他解决方案,而不是使用属性字符串。

但是对于通知部分,您可以将textFields委托设置为UIView并在视图中定义以下方法。

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string; 

每次在文本字段中输入的文本更改时都会调用上述方法。

在将委托设置为文本字段后试试这个代码。希望它会起作用。

- (BOOL)textField:(UITextField *)textField shouldChangeCharactersInRange:(NSRange)range replacementString:(NSString *)string
{
    NSMutableAttributedString *attributedString = [[NSMutableAttributedString alloc] initWithString:textField.text];
    [attributedString addAttribute:NSKernAttributeName
                             value:@(5.4)
                             range:NSMakeRange(0, textField.text.length)];
    textField.attributedText = attributedString;
    return YES;
}

这在 Swift 2.2 中运行良好。 希望这对您在文本字段中的字母间距有所帮助

override func viewDidLoad() {
 // Do any additional setup after loading the view.
        NSNotificationCenter.defaultCenter().addObserver(self, selector: #selector(SignupVC.limitTextField(_:)), name: "UITextFieldTextDidChangeNotification", object: txtContactNumber)
}
 func limitTextField(Notif:NSNotification) {

    let limit=10;

    let attributedString = NSMutableAttributedString(string: txtContactNumber.text!)
    attributedString.addAttribute(NSKernAttributeName, value: 7, range: NSMakeRange(0, (txtContactNumber.text?.characters.count)!))
   // attributedString.addAttribute(NSFontAttributeName, value: font, range: NSMakeRange(0, count(textField.text)))
    attributedString.addAttribute(NSForegroundColorAttributeName, value: UIColor.blackColor(), range: NSMakeRange(0,(txtContactNumber.text?.characters.count)!))

    txtContactNumber.attributedText = attributedString
    if(txtContactNumber.text?.characters.count>limit)
    {
        txtContactNumber.text=txtContactNumber.text?.substringToIndex(limit)
    }
}

需要计算每个字符的字距并将其删除为最后一个字符。 Swift 5.3 上有示例

func textField(_ textField: UITextField, shouldChangeCharactersIn range: NSRange, replacementString string: String) -> Bool {
    
    let maxLength = 6
    let symbolWidth = CGFloat(43)
    let font = UIFont.systemFont(ofSize: 30)
    
    if string == "" { // when user remove text
        return true
    }
        
    if textField.text!.count + string.count - range.length > maxLength { // when user type extra text
        return false
    }
    
    let currentText = NSMutableAttributedString(attributedString: textField.attributedText ?? NSMutableAttributedString())
    currentText.deleteCharacters(in: range) // delete selected text
    var newStringLength = 0 
    
    for char in string{
        let newSymbol = NSMutableAttributedString(string: String(char))
        newSymbol.addAttribute(.font, value: font, range: NSMakeRange(0, 1))
        let currentSymbolWidth = newSymbol.size().width
        let kern = symbolWidth - currentSymbolWidth
        newSymbol.addAttribute(.kern, value: kern, range: NSMakeRange(0,1))
        
        currentText.insert(newSymbol, at: range.location + newStringLength)
        newStringLength += 1
    }
    
    if currentText.length == maxLength{
        currentText.addAttribute(.kern, value: 0, range: NSMakeRange(maxLength - 1, 1))
    }
    
    textField.attributedText = currentText
    
    return false

}

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM