JXSegmentedIndicatorLineView.swift 5.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  1. //
  2. // JXSegmentedIndicatorLineView.swift
  3. // JXSegmentedView
  4. //
  5. // Created by jiaxin on 2018/12/26.
  6. // Copyright © 2018 jiaxin. All rights reserved.
  7. //
  8. import UIKit
  9. public enum JXSegmentedIndicatorLineStyle {
  10. case normal
  11. case lengthen
  12. case lengthenOffset
  13. }
  14. open class JXSegmentedIndicatorLineView: JXSegmentedIndicatorBaseView {
  15. open var lineStyle: JXSegmentedIndicatorLineStyle = .normal
  16. /// lineStyle为lengthenOffset时使用,滚动时x的偏移量
  17. open var lineScrollOffsetX: CGFloat = 10
  18. open override func commonInit() {
  19. super.commonInit()
  20. indicatorHeight = 3
  21. }
  22. open override func refreshIndicatorState(model: JXSegmentedIndicatorSelectedParams) {
  23. super.refreshIndicatorState(model: model)
  24. backgroundColor = indicatorColor
  25. layer.cornerRadius = getIndicatorCornerRadius(itemFrame: model.currentSelectedItemFrame)
  26. let width = getIndicatorWidth(itemFrame: model.currentSelectedItemFrame, itemContentWidth: model.currentItemContentWidth)
  27. let height = getIndicatorHeight(itemFrame: model.currentSelectedItemFrame)
  28. let x = model.currentSelectedItemFrame.origin.x + (model.currentSelectedItemFrame.size.width - width)/2
  29. var y: CGFloat = 0
  30. switch indicatorPosition {
  31. case .top:
  32. y = verticalOffset
  33. case .bottom:
  34. y = model.currentSelectedItemFrame.size.height - height - verticalOffset
  35. case .center:
  36. y = (model.currentSelectedItemFrame.size.height - height)/2 + verticalOffset
  37. }
  38. frame = CGRect(x: x, y: y, width: width, height: height)
  39. }
  40. open override func contentScrollViewDidScroll(model: JXSegmentedIndicatorTransitionParams) {
  41. super.contentScrollViewDidScroll(model: model)
  42. guard canHandleTransition(model: model) else {
  43. return
  44. }
  45. let rightItemFrame = model.rightItemFrame
  46. let leftItemFrame = model.leftItemFrame
  47. let percent = model.percent
  48. var targetX: CGFloat = leftItemFrame.origin.x
  49. var targetWidth = getIndicatorWidth(itemFrame: leftItemFrame, itemContentWidth: model.leftItemContentWidth)
  50. let leftWidth = targetWidth
  51. let rightWidth = getIndicatorWidth(itemFrame: rightItemFrame, itemContentWidth: model.rightItemContentWidth)
  52. let leftX = leftItemFrame.origin.x + (leftItemFrame.size.width - leftWidth)/2
  53. let rightX = rightItemFrame.origin.x + (rightItemFrame.size.width - rightWidth)/2
  54. switch lineStyle {
  55. case .normal:
  56. targetX = JXSegmentedViewTool.interpolate(from: leftX, to: rightX, percent: CGFloat(percent))
  57. if indicatorWidth == JXSegmentedViewAutomaticDimension {
  58. targetWidth = JXSegmentedViewTool.interpolate(from: leftWidth, to: rightWidth, percent: CGFloat(percent))
  59. }
  60. case .lengthen:
  61. //前50%,只增加width;后50%,移动x并减小width
  62. let maxWidth = rightX - leftX + rightWidth
  63. if percent <= 0.5 {
  64. targetX = leftX
  65. targetWidth = JXSegmentedViewTool.interpolate(from: leftWidth, to: maxWidth, percent: CGFloat(percent*2))
  66. }else {
  67. targetX = JXSegmentedViewTool.interpolate(from: leftX, to: rightX, percent: CGFloat((percent - 0.5)*2))
  68. targetWidth = JXSegmentedViewTool.interpolate(from: maxWidth, to: rightWidth, percent: CGFloat((percent - 0.5)*2))
  69. }
  70. case .lengthenOffset:
  71. //前50%,增加width,并少量移动x;后50%,少量移动x并减小width
  72. let maxWidth = rightX - leftX + rightWidth - lineScrollOffsetX*2
  73. if percent <= 0.5 {
  74. targetX = JXSegmentedViewTool.interpolate(from: leftX, to: leftX + lineScrollOffsetX, percent: CGFloat(percent*2))
  75. targetWidth = JXSegmentedViewTool.interpolate(from: leftWidth, to: maxWidth, percent: CGFloat(percent*2))
  76. }else {
  77. targetX = JXSegmentedViewTool.interpolate(from:leftX + lineScrollOffsetX, to: rightX, percent: CGFloat((percent - 0.5)*2))
  78. targetWidth = JXSegmentedViewTool.interpolate(from: maxWidth, to: rightWidth, percent: CGFloat((percent - 0.5)*2))
  79. }
  80. }
  81. self.frame.origin.x = targetX
  82. self.frame.size.width = targetWidth
  83. }
  84. open override func selectItem(model: JXSegmentedIndicatorSelectedParams) {
  85. super.selectItem(model: model)
  86. let targetWidth = getIndicatorWidth(itemFrame: model.currentSelectedItemFrame, itemContentWidth: model.currentItemContentWidth)
  87. var toFrame = self.frame
  88. toFrame.origin.x = model.currentSelectedItemFrame.origin.x + (model.currentSelectedItemFrame.size.width - targetWidth)/2
  89. toFrame.size.width = targetWidth
  90. if canSelectedWithAnimation(model: model) {
  91. UIView.animate(withDuration: scrollAnimationDuration, delay: 0, options: .curveEaseOut, animations: {
  92. self.frame = toFrame
  93. }) { (_) in
  94. }
  95. }else {
  96. frame = toFrame
  97. }
  98. }
  99. }