iOS UISearchController样式全面设置

image

Q: 如果想要做到上图所示的搜索框需要怎么设置?

搜索框的样式设置都是基于UISearchControllersearchBar也就是UISearchBar设置的。
但是UISearchBar并没有提供多少自定义设置。 图标相关的设置也主要是系统提供的几种简单样式。

而iOS11和iOS11以下的搜索框由于样式不一样,很多属性的设置不尽相同。

所以这篇文章就是对怎么修改UISearchController样式的梳理

首先UISearchController主要是对UISearchBar的封装,而UISearchBar又包含了UITextFieldUINavigationButton(取消按钮)

一 UITextField设置

获取TextField

有两种方法可以获取,KVC或者遍历子视图。
我们这里采用KVC,效率比较高。

let searchField = searchBar.value(forKey: "searchField")

func findTextView(in view: UIView)->UITextField {
    for view in searchBar.subviews[0].subviews {
        if view is UITextField {
            return view as! UITextField
        }
    }
    return nil
}
    

这里的遍历都是对seachBar.subviews[0]行遍历,如果以后在某个版本比如iOS12中如果UISearchBar的结构又发生了改变,那可能就找不到了。 所以如果要使用遍历的话,还是递归seachBar的所有子视图去找比较保险。
改变属性名称的概率比较小,所以用KVO相对安全一点。

1. 图标

这里我们讨论三种图标的设置

  • UISearchBarIcon.search:这个是搜索框左侧的那个放大镜图标
  • UISearchBarIcon.bookmark:这个是搜索框右侧显示的图标,默认是不现实的。效果就是微信语音输入的那个话筒一样。
  • UISearchBarIcon.clear: 这个是在输入过程中显示的清除按钮。

如果设置了bookmark,在没有输入的时候显示bookmark,输入的时候显示清除按钮

代码

这三种图标都可以通过以下方法来设置。

searchBar.setImage(icon, for: .searchBar, state: .normal)

除此之外也可以获取具体的UI组件来设置
如果是想用默认的图片但只是想改变其颜色的话,

1)可以通过下面这个方法来重绘颜色。把返回的图片再通过setImage来设置即可。

extension UIImage {
    func reRender(with color: UIColor) -> UIImage? {
        UIGraphicsBeginImageContextWithOptions(size, false, scale)
        if let context = UIGraphicsGetCurrentContext(), let cg = cgImage {
            context.translateBy(x: 0, y: size.height)
            context.scaleBy(x: 1.0, y: -1.0)
            context.setBlendMode(.normal)
            color.setFill()
            let rect = CGRect(x: 0, y: 0, width: size.width, height: size.height)
            context.clip(to: rect, mask: cg) // 乘以alpha
            context.fill(rect)
            let newImage: UIImage? = UIGraphicsGetImageFromCurrentImageContext()
            UIGraphicsEndImageContext()
            return newImage }
        else {
            print("Failed to get current context or cgImage not exist")
            return nil
        }
    }
}
let icon = image.reRender(with: .white)
searchBar.setImage(icon, for: .searchBar, state: .normal)

2)获取UI组件,更改其渲染模式

以修改放大镜颜色为例。
这里我们定义一个属性,然后通过属性监测器来应用到图片上。这里我们取的是在文章开头定义的searchField。然后取它的左视图UIImageView来修改。

public var leftIconColor: UIColor? {
        didSet {
            if let color = leftIconColor,
               let iconView = searchField?.leftView as? UIImageView,
               let icon = iconView.image {
                    iconView.image = icon.withRenderingMode(.alwaysTemplate)
                    iconView.tintColor = color
            }

        }
    }

可以通过获取原图片然后修改图片的。
其它图标的修改请参考文末附的源码。

2. 文字相关

这部分的设置都是跟UISearchField的一样

searchField?.textColor // 文字颜色
searchField?.font // 字体
searchField?.placeholder // 占位符
searchField?.attributedPlaceholder // 富文本占位符

3. 光标颜色

通过设置searchFieldtintColor即可修改光标颜色
修改searchBartintColor也可以达到效果,因为子视图的tintColor会自动继承父级视图。
但是设置searchBartintColor还会修改取消按钮的颜色。如果你想要他们同一个颜色话,直接设置seachBar更好。

4. 背景颜色

iOS11以下的直接设置searchField?.backgroundColor即可。
而iOS11的要设置

let searchFieldBackgroudView = searchField?.subviews.first
searchFieldBackgroudView.backgroundColor

二 取消按钮

取消按钮就是在输入的时候显示的cancel按钮。只要设置showsCancelButton即可。

searchBar.showsCancelButton = true

因为模拟器默认语言都是英语的,所以你的取消按钮默认也是显示cancel。如果切换成中文的话显示就是取消

而如果要自定义这个按钮的话,我们采用的也是遍历子视图获取到这个按钮的方法。

func setupCancelButton() {
    for view in searchBar.subviews[0].subviews {
        if view is UIButton { // 如果是UIButton
            let button = view as! UIButton
            //接下来就像普通button一样设置就好了
            button.setAttributedTitle(attributedTitle, for: .normal) // 通过设置副文本标题可以更改标题
            break
        }
    }
}

然后在searchBar的代理事件的searchBarShouldBeginEditing或者searchBarDidBeginEditing中调用即可。

其它还有一些细节的设置可以参考下面的源码
源码已经提交到Github上了。
并且简化了所有的属性和代理方法。
欢迎大家来提交问题和分支。
如果觉得有帮助的话记得给颗星:P

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343

推荐阅读更多精彩内容