使用Runtime+KVC改变TextField光标颜色

首先,是有简单方法可以改变TextField和TextView的光标颜色的

textField.tintColor = someOtherColor;
// 或
[[UITextField appearance] setTintColor:someOtherColor]

由于面试时被问到以上方法之外的方法,当时没想出来。
下面讨论的是使用Runtime + KVC方法改变光标的颜色

1.首先,由于Apple的命名基本是见名知意的,所以先看一下UITextField这个类内成员变量的情况,看看能否找到与光标颜色对应的成员变量:

- (void)getIvarsOfUITextField
{
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([UITextField class], &count);
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
        NSLog(@"%s---%s",ivar_getName(ivar),ivar_getTypeEncoding(ivar));
    }
}
这里写图片描述

上面的代码输出了UITextField内的50个成员变量的变量名和TypeEncoding,但是经过一番查找,并没有直接找到字面上和光标有关的变量。
对于一些这个类中已经有暴露出来的成员以及一些用于呈现文本或背景的视图,如_backgroundView_floatingContentView_displayLabel 基本可以猜到与我们想要的实现无关。

2.挑选一些私有类进行尝试,这里选了 _systemInputViewController---@"UISystemInputViewController"_traits---@"UITextInputTraits" 这样的可能选项。
在前面的方法中增加代码:

- (void)getIvars
{
    unsigned int count = 0;
    Ivar *ivars = class_copyIvarList([UITextField class], &count);
    for (int i = 0; i < count; i++) {
        Ivar ivar = ivars[i];
//        NSLog(@"%s---%s",ivar_getName(ivar),ivar_getTypeEncoding(ivar));
        if ( strcmp(ivar_getName(ivar), "_systemInputViewController") == 0) {
            NSLog(@"%s",ivar_getTypeEncoding(ivar));
            NSLog(@"=========_systemInputViewController start==========");
            unsigned int subcount = 0;
            NSString *fullClassString = [NSString stringWithCString:ivar_getTypeEncoding(ivar) encoding:NSUTF8StringEncoding];
            fullClassString = [fullClassString substringFromIndex:2];
            fullClassString = [fullClassString substringToIndex:fullClassString.length-1];
            Class someClass = NSClassFromString(fullClassString);
            Ivar *subVars = class_copyIvarList(someClass, &subcount);
            for (int j= 0; j < subcount; j++) {
                Ivar var = subVars[j];
                NSLog(@"%s---%s",ivar_getName(var),ivar_getTypeEncoding(var));
            }
            NSLog(@"=========_systemInputViewController end==========");
        }
        
        if ( strcmp(ivar_getName(ivar), "_traits") == 0) {
            NSLog(@"%s",ivar_getTypeEncoding(ivar));
            NSLog(@"=========_traits start==========");
            unsigned int subcount = 0;
            NSString *fullClassString = [NSString stringWithCString:ivar_getTypeEncoding(ivar) encoding:NSUTF8StringEncoding];
            fullClassString = [fullClassString substringFromIndex:2];
            fullClassString = [fullClassString substringToIndex:fullClassString.length-1];
            Class someClass = NSClassFromString(fullClassString);
            Ivar *subVars = class_copyIvarList(someClass, &subcount);
            for (int j= 0; j < subcount; j++) {
                Ivar var = subVars[j];
                NSLog(@"%s---%s",ivar_getName(var),ivar_getTypeEncoding(var));
            }
            NSLog(@"=========_traits end==========");
        }
    }
}

其中: _systemInputViewController---@"UISystemInputViewController" 这个成员变量的类中输入:

这里写图片描述

可以看出 UISystemInputViewController 这个类中存放着一些和accessoryView、keyboard等相关的Controller和Constraints,并没有我们想要的内容,至少从成员变量名上看起来是这样的。

接下来 _traits---@"UITextInputTraits" 中输出:

这里写图片描述

可以看到有insertionPointColor---@"UIColor"insertionPointWidth---Q 这两个成员变量,从命名上看很可能是我们想要的,加上insertionPointColor 的类型又是UIColor

  1. 验证
    在代码中使用KVC对这个keyPath进行赋值验证:
[self.field setValue:[UIColor redColor] forKeyPath:@"_traits.insertionPointColor"];

结果textfield的光标颜色果然改变了~
效果:


这里写图片描述

但TextField中还有一个_nonAtomTraits 成员为UITextInputTraits 类,下面用它来试试:

[self.field setValue:[UIColor redColor] forKeyPath:@"_nonAtomTraits.insertionPointColor"];

结果光标颜色并没有发生改变,暂时它与_traits

你可能感兴趣的:(使用Runtime+KVC改变TextField光标颜色)