Swift開發(fā) 自定義Segment

引言

自定義封裝了一個(gè)Segment,使用簡單,下面是效果圖。


自定義Segment效果圖.gif

代碼部分

  • 首先定義一個(gè)SegmentStyle棚唆,把我們需要的一些屬性都放進(jìn)去来惧,如是否顯示下劃線冗栗,是否顯示遮罩等等,這樣在我們封裝好后可以更加方便我們的使用供搀。
public struct SegmentStyle{
    /// 是否顯示遮蓋
    public var showCover = false
    /// 是否顯示下劃線
    public var showLine = false
    /// 是否縮放文字
    public var scaleTitle = false
    /// 是否可以滾動(dòng)標(biāo)題
    public var scrollTitle = true
    /// 下面的滾動(dòng)條的高度 默認(rèn)2
    public var scrollLineHeight: CGFloat = 2
    /// 下面的滾動(dòng)條的顏色
    public var scrollLineColor = UIColor.brown
    /// 遮蓋的背景顏色
    public var coverBackgroundColor = UIColor.lightGray
    /// 遮蓋圓角
    public var coverCornerRadius: CGFloat = 10.0
    /// cover的高度 默認(rèn)28
    public var coverHeight: CGFloat = 28.0
    /// 文字間的間隔 默認(rèn)15
    public var titleMargin: CGFloat = 15
    /// 文字 字體 默認(rèn)14.0
    public var titleFont = UIFont.systemFont(ofSize: 14.0)
    /// 放大倍數(shù) 默認(rèn)1.3
    public var titleBigScale: CGFloat = 1.1
    /// 默認(rèn)倍數(shù) 不可修改
    let titleOriginalScale: CGFloat = 1.0
    
    /// 文字正常狀態(tài)顏色 請使用RGB空間的顏色值!! 如果提供的不是RGB空間的顏色值就可能crash
    public var normalTitleColor = UIColor(red: 51.0/255.0, green: 53.0/255.0, blue: 75/255.0, alpha: 1.0)
    /// 文字選中狀態(tài)顏色 請使用RGB空間的顏色值!! 如果提供的不是RGB空間的顏色值就可能crash
    public var selectedTitleColor = UIColor(red: 255.0/255.0, green: 0.0/255.0, blue: 121/255.0, alpha: 1.0)
    public init() {
        
    }
  • 自定義SegmentView隅居,重新創(chuàng)建一個(gè)Swift文件,初始化
import UIKit

class ce: UIView {
   public init(frame: CGRect, segmentStyle: SegmentStyle, titles: [String]) {
        self.segmentStyle = segmentStyle
        self.titles = titles
        super.init(frame: frame)
        
        addSubview(scrollView)
        // 根據(jù)Titles添加相應(yīng)的控件
        setupTitles()
        // 設(shè)置Frame
        setupUI()
    }
    
    required init?(coder aDecoder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }
}

把需要的屬性加進(jìn)去

    open var segmentStyle: SegmentStyle
    /// 點(diǎn)擊響應(yīng)
    open var titleBtnOnClick:((_ label: UILabel, _ index: Int)->Void)?
    /// 所有標(biāo)題的寬度
    fileprivate var titleWidthArry: [CGFloat] = []
    /// 所有的標(biāo)題
    fileprivate var titles: [String]
    /// 緩存標(biāo)題
    fileprivate var labelsArray: [UILabel]  = []
    /// self.bounds.size.width
    fileprivate var currentWidth: CGFloat = 0
    /// 記錄當(dāng)前選中的下標(biāo)
    fileprivate var currentIndex = 0
    /// 記錄上一個(gè)下標(biāo)
    fileprivate var oldIndex = 0
    /// 所以文字的總寬度
    fileprivate var labelWithMax: CGFloat = 0
    /// 遮罩x和文字x的間隙
    fileprivate var xGap = 5
    /// 遮罩寬度比文字寬度多的部分
    fileprivate var wGap: Int {
        return 2 * xGap
    }
  • 懶加載一個(gè)ScrollView作為容器
/// 管理標(biāo)題的滾動(dòng)
    fileprivate lazy var scrollView: UIScrollView = {
        let scrollV = UIScrollView()
        scrollV.frame = CGRect(x: 0, y: 0, width: self.frame.width, height: self.frame.height)
        scrollV.showsHorizontalScrollIndicator = false
        scrollV.bounces = true
        scrollV.isPagingEnabled = false
        scrollV.scrollsToTop = false
        return scrollV
    }()
  • 是否顯示滾動(dòng)條或者遮罩
/// 是否顯示滾動(dòng)條
    fileprivate lazy var scrollLine: UIView? = {[unowned self] in
        let line = UIView()
        return self.segmentStyle.showLine ? line : nil
    }()
  
    /// 是否顯示遮罩
    fileprivate lazy var coverView: UIView? = {[unowned self] in
        let cover = UIView()
        cover.layer.cornerRadius = CGFloat(self.segmentStyle.coverCornerRadius)
        cover.layer.masksToBounds = true
        return self.segmentStyle.showCover ? cover : nil
    }()
  • 基本的東西都設(shè)置好了葛虐,現(xiàn)在我們要添加相應(yīng)的item
// 根據(jù)Titles添加相應(yīng)的控件
    fileprivate func setupTitles() {
        for (index, title) in titles.enumerated(){
            let label = CustomLabel(frame: CGRect.zero)
            label.tag = index
            label.text = title
            label.font = segmentStyle.titleFont
            label.textColor = UIColor.black
            label.textAlignment = .center
            label.isUserInteractionEnabled = true
            // 添加點(diǎn)擊手勢
            let tapGes = UITapGestureRecognizer(target: self, action: #selector(self.titleLabelOnClick(_:)))
            label.addGestureRecognizer(tapGes)
            // 計(jì)算文本寬高
            let size = (title as NSString).boundingRect(with: CGSize(width: CGFloat(MAXFLOAT), height: 0.0), options: .usesLineFragmentOrigin, attributes: [NSAttributedStringKey.font: label.font], context: nil)
            // 緩存文字寬度
            titleWidthArry.append(size.width)
            // 緩存label
            labelsArray.append(label)
            // 添加label
            scrollView.addSubview(label)
        }
    }
// 設(shè)置Frame
    fileprivate func setupUI() {
        // 設(shè)置Label位置
        currentWidth = bounds.size.width
        setUpLabelsPosition()
        // 設(shè)置滾動(dòng)條和遮罩
        setupScrollLineAndCover()
        
        if segmentStyle.scrollTitle{
            if let lastLabel = labelsArray.last {
                scrollView.contentSize = CGSize(width: lastLabel.frame.maxX + segmentStyle.titleMargin, height: 0)
            }
        }
    }
/// 設(shè)置label的位置
    fileprivate func setUpLabelsPosition() {
        var titleX: CGFloat = 0.0
        let titleY: CGFloat = 0.0
        var titleW: CGFloat = 0.0
        let titleH = bounds.size.height - segmentStyle.scrollLineHeight
        if !segmentStyle.scrollTitle{
            titleW = currentWidth/CGFloat(titles.count)
            for(index, label) in labelsArray.enumerated(){
                titleX = titleW * CGFloat(index)
                
                label.frame = CGRect(x: titleX, y: titleY, width: titleW, height: titleH)
            }
        }else{
            // 計(jì)算標(biāo)題長度總和
            for (index, labelWith) in titleWidthArry.enumerated(){
                labelWithMax += labelWith + 2 * segmentStyle.titleMargin
            }
            // 當(dāng)標(biāo)題的長度總和沒有屏幕寬度長時(shí)胎源,平分屏幕寬度
            if labelWithMax <= currentWidth{
                for(index, label) in labelsArray.enumerated(){
                    let currWidth = currentWidth - 2 * segmentStyle.titleMargin
                    titleW = currWidth/CGFloat(labelsArray.count)
                    titleX = segmentStyle.titleMargin
                    if index != 0{
                        let lastLabel = labelsArray[index - 1]
                        titleX = lastLabel.frame.maxX 
                    }
                    label.frame = CGRect(x: titleX, y: titleY, width: titleW, height: titleH)
                }
            }
            // 當(dāng)標(biāo)題的長度總和比屏幕寬度短時(shí)
            else{
                for(index, label) in labelsArray.enumerated(){
                    titleW = titleWidthArry[index]
                
                    titleX = segmentStyle.titleMargin
                    if index != 0{
                        let lastLabel = labelsArray[index - 1]
                        titleX = lastLabel.frame.maxX + segmentStyle.titleMargin * 2
                    }
                    label.frame = CGRect(x: titleX, y: titleY, width: titleW, height: titleH)
                }
            }
        }
        if let firstLabel = labelsArray[0] as? CustomLabel {
            
            // 縮放, 設(shè)置初始的label的transform
            if segmentStyle.scaleTitle {
                firstLabel.currentTransformSx = segmentStyle.titleBigScale
            }
            // 設(shè)置初始狀態(tài)文字的顏色
            firstLabel.textColor = segmentStyle.selectedTitleColor
        }
    }
    /// 設(shè)置滾動(dòng)條和遮罩
    fileprivate func setupScrollLineAndCover(){
        if let line = scrollLine {
            line.backgroundColor = segmentStyle.scrollLineColor
            scrollView.addSubview(line)
        }
        if let cover = coverView {
            cover.backgroundColor = segmentStyle.coverBackgroundColor
            scrollView.insertSubview(cover, at: 0)
        }
        let coverX = labelsArray[0].frame.origin.x
        let coverW = labelsArray[0].frame.size.width
        let coverH: CGFloat = segmentStyle.coverHeight
        let coverY = (bounds.size.height - coverH) / 2
        
        // 設(shè)置遮罩位置
        if segmentStyle.scrollTitle {
            // 這里x-xGap width+wGap 是為了讓遮蓋的左右邊緣和文字有一定的距離
            coverView?.frame = CGRect(x: coverX - CGFloat(xGap), y: coverY, width: coverW + CGFloat(wGap), height: coverH)
        } else {
            coverView?.frame = CGRect(x: coverX, y: coverY, width: coverW, height: coverH)
        }
        // 設(shè)置滾動(dòng)條位置
        scrollLine?.frame = CGRect(x: coverX, y: bounds.size.height - segmentStyle.scrollLineHeight, width: coverW, height: segmentStyle.scrollLineHeight)
    }
  • 當(dāng)有多個(gè)標(biāo)題時(shí),可以讓選中的標(biāo)題居中屿脐,這樣可以方便使用涕蚤。
/// 讓選中標(biāo)簽居中顯示
    public func adjustTitleOffSetToCurrentIndex(_ currentIndex: Int){
        let currentLabel = labelsArray[currentIndex]
        
        for index in labelsArray.enumerated(){
            if index.offset != currentIndex{
                index.element.textColor = self.segmentStyle.normalTitleColor
            }
        }
        /// scrollView需要移動(dòng)的偏移量
        var offSetX = currentLabel.center.x - currentWidth/2
        if offSetX < 0 {
            offSetX = 0
        }
        /// scrollView最大偏移量
        var maxOffSetX = scrollView.contentSize.width - currentWidth
        // 可以滾動(dòng)的區(qū)域小余屏幕寬度
        if maxOffSetX < 0 {
            maxOffSetX = 0
        }
        // 當(dāng)offSetX偏移量大于最大偏移量時(shí),就直接等于最大偏移量的诵,否則會(huì)出現(xiàn)最后一個(gè)標(biāo)簽也居中顯示
        if offSetX > maxOffSetX {
            offSetX = maxOffSetX
        }
        // 設(shè)置scrollView的偏移量
        scrollView.setContentOffset(CGPoint(x:offSetX, y: 0), animated: true)
    }
  • 實(shí)現(xiàn)Label手勢點(diǎn)擊方法
    @objc func titleLabelOnClick(_ tapGes: UITapGestureRecognizer){
        guard let currentLabel = tapGes.view as? CustomLabel else { return }
        currentIndex = currentLabel.tag
        print(currentLabel.tag)
        adjustUIWhenBtnOnClickWithAnimate(true)
    }

使用部分

        var style = SegmentStyle()
        style.scrollTitle = true
        style.showLine = true
        style.scrollLineColor = UIColor.blue
        let scrollview = ScrollSegmentView(frame: CGRect(x: 0, y: 64, width: self.view.frame.width, height: 40), segmentStyle: style, titles: ["絕地求生","絕地大逃殺"])
        self.view.addSubview(scrollview)

好了万栅,最后附上Demo的GItHub地址。希望能夠幫助到一些需要的人西疤。自己也可以嘗試多種組合烦粒。

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市代赁,隨后出現(xiàn)的幾起案子扰她,更是在濱河造成了極大的恐慌,老刑警劉巖芭碍,帶你破解...
    沈念sama閱讀 218,284評論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件徒役,死亡現(xiàn)場離奇詭異,居然都是意外死亡豁跑,警方通過查閱死者的電腦和手機(jī)廉涕,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,115評論 3 395
  • 文/潘曉璐 我一進(jìn)店門泻云,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人狐蜕,你說我怎么就攤上這事宠纯。” “怎么了层释?”我有些...
    開封第一講書人閱讀 164,614評論 0 354
  • 文/不壞的土叔 我叫張陵婆瓜,是天一觀的道長。 經(jīng)常有香客問我贡羔,道長廉白,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,671評論 1 293
  • 正文 為了忘掉前任乖寒,我火速辦了婚禮猴蹂,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘楣嘁。我一直安慰自己磅轻,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,699評論 6 392
  • 文/花漫 我一把揭開白布逐虚。 她就那樣靜靜地躺著聋溜,像睡著了一般。 火紅的嫁衣襯著肌膚如雪叭爱。 梳的紋絲不亂的頭發(fā)上撮躁,一...
    開封第一講書人閱讀 51,562評論 1 305
  • 那天,我揣著相機(jī)與錄音买雾,去河邊找鬼把曼。 笑死,一個(gè)胖子當(dāng)著我的面吹牛漓穿,可吹牛的內(nèi)容都是我干的祝迂。 我是一名探鬼主播,決...
    沈念sama閱讀 40,309評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼器净,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了当凡?” 一聲冷哼從身側(cè)響起山害,我...
    開封第一講書人閱讀 39,223評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎沿量,沒想到半個(gè)月后浪慌,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,668評論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡朴则,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,859評論 3 336
  • 正文 我和宋清朗相戀三年权纤,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 39,981評論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡汹想,死狀恐怖外邓,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情古掏,我是刑警寧澤损话,帶...
    沈念sama閱讀 35,705評論 5 347
  • 正文 年R本政府宣布,位于F島的核電站槽唾,受9級特大地震影響丧枪,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜庞萍,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,310評論 3 330
  • 文/蒙蒙 一拧烦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧钝计,春花似錦恋博、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,904評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至践付,卻和暖如春秦士,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背永高。 一陣腳步聲響...
    開封第一講書人閱讀 33,023評論 1 270
  • 我被黑心中介騙來泰國打工隧土, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人命爬。 一個(gè)月前我還...
    沈念sama閱讀 48,146評論 3 370
  • 正文 我出身青樓曹傀,卻偏偏與公主長得像,于是被迫代替她去往敵國和親饲宛。 傳聞我的和親對象是個(gè)殘疾皇子皆愉,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,933評論 2 355

推薦閱讀更多精彩內(nèi)容