分段控件中的渐变色
Gradient tint color in segmented control
我用这个方法得到了渐变图像
func gradient(size:CGSize,color:[UIColor]) -> UIImage?{
//turn color into cgcolor
let colors = color.map{[=10=].cgColor}
//begin graphics context
UIGraphicsBeginImageContextWithOptions(size, true, 0.0)
guard let context = UIGraphicsGetCurrentContext() else {
return nil
}
// From now on, the context gets ended if any return happens
defer {UIGraphicsEndImageContext()}
//create core graphics context
let locations:[CGFloat] = [0.0,1.0]
guard let gredient = CGGradient.init(colorsSpace: CGColorSpaceCreateDeviceRGB(), colors: colors as NSArray as CFArray, locations: locations) else {
return nil
}
//draw the gradient
context.drawLinearGradient(gredient, start: CGPoint(x:0.0,y:size.height), end: CGPoint(x:size.width,y:size.height), options: [])
// Generate the image (the defer takes care of closing the context)
return UIGraphicsGetImageFromCurrentImageContext()
}
然后我将分段控件的tintColor设置为渐变:
let gradientImage = gradient(size: listSegmentedControl.frame.size, color: [UIColor.black, UIColor.red])!
listSegmentedControl.tintColor = UIColor(patternImage: gradientImage)
那是行不通的。但是,相同的代码适用于设置 backgroundColor:
let gradientImage = gradient(size: listSegmentedControl.frame.size, color: [UIColor.black, UIColor.red])!
listSegmentedControl.backgroundColor = UIColor(patternImage: gradientImage)
有人知道为什么吗?我真的需要设置 gradient tintColor。非常感谢任何帮助。
编辑:
理想情况下,我希望我的分段控件看起来像这样:
这是一个已知的更改 UISegmentedControl
色调的技巧
let sortedViews = listSegmentedControl.subviews.sorted( by: { [=10=].frame.origin.x < .frame.origin.x } )
for (index, view) in sortedViews.enumerated() {
if index == listSegmentedControl.selectedSegmentIndex {
view.tintColor = UIColor(patternImage: gradientImage)
} else {
view.tintColor = UIColor.gray //Whatever the color of non selected segment controller tab
}
}
虽然看起来像一个丑陋的 hack,但我已经使用了很长一段时间并且看起来相当简单。希望对你有帮助。
编辑:
这是你需要的伙计吗?
如果是,让我知道,我会post相同的代码。
编辑 2:
正如 OP 在他的评论中提到的那样,他期望的输出与我在上图中显示的输出相同,并提供了相同的代码。
免责声明:
正如 rmaddy 在他下面的评论中所提到的,这是一个 hack 并利用未记录的(完整 public API 但是一个非常众所周知的 hack 来改变UISegemntedControl 的色调从 iOS 5 开始就存在(这就是我记得的方式,让我知道我是否错了)
因此请注意,在 iOS 的未来版本中,Apple 可能会 更改 UISegemntedControl
中子视图的结构并可能影响你的O/P。我看不到的任何东西都不会导致崩溃,但可能会影响 O/P 在屏幕上的呈现方式。
我已经声明了一个变量,这样 GradientImage 只能生成一次,但是如何使用它取决于您的实现方式
var gradientImage : UIImage! = nil
在ViewDidLoad
中我将gradientImage
和UISegmentedControl
初始化为
override func viewDidLoad() {
super.viewDidLoad()
gradientImage = gradient(size: segmentControl.frame.size, color: [UIColor.black, UIColor.red])!
//I have specified custom font need not necessarily be used
//Font color attribute is important though, usually `UISegementedControl` title takes color from tint color, because we might need a different color for text to highlight above gradient color am using custom font colors
let font = UIFont(name: "HelveticaNeue-Medium", size: 20)
segmentControl.setTitleTextAttributes([NSFontAttributeName : font!, NSForegroundColorAttributeName : UIColor.blue], for: .normal)
segmentControl.setTitleTextAttributes([NSForegroundColorAttributeName : UIColor.white], for: .selected)
//Set the border color and border to `UISegmentedControl` and also make it round corner
segmentControl.layer.borderColor = UIColor(patternImage: gradientImage).cgColor
segmentControl.layer.borderWidth = 2
segmentControl.layer.masksToBounds = true
segmentControl.layer.cornerRadius = 10
//In order to update the selected Segment tint and background color we need to call multiple statements every time selection changes hence I have moved it to the function and called it in viewDidLoad
updateGradientBackground()
}
最后 updateGradientBackground 函数定义与我 post 在我的原始答案中编辑的相同
fileprivate func updateGradientBackground() {
let sortedViews = segmentControl.subviews.sorted( by: { [=13=].frame.origin.x < .frame.origin.x } )
for (index, view) in sortedViews.enumerated() {
if index == segmentControl.selectedSegmentIndex {
//very important thing to notice here is because tint color was not honoring the `UIColor(patternImage` I rather used `backgroundColor` to create the effect and set clear color as clear color
view.backgroundColor = UIColor(patternImage: self.gradientImage)
view.tintColor = UIColor.clear
} else {
//very important thing to notice here is because tint color was not honoring the `UIColor(patternImage` I rather used `backgroundColor` to create the effect and set clear color as clear color
view.backgroundColor = UIColor.white //Whatever the color of non selected segment controller tab
view.tintColor = UIColor.clear
}
}
}
最后,在UISegmentedControl的IBAction中,直接调用
@IBAction func segmentControllerTapped(_ sender: UISegmentedControl) {
self.updateGradientBackground()
}
希望对您有所帮助
为 iOS13+ UISegmentControl 上的旋钮实现了渐变颜色。添加了带渐变的子视图,然后为其设置蒙版并复制动画以匹配原始旋钮。
// 1 - Make subclass
class GradientSegment: UISegmentedControl {
// this could be UIImageView
lazy var gradientView: GradientView = {
let view = GradientView(frame: self.bounds)
view.startColor = ...
view.endColor = ...
view.horizontalMode = true
view.autoresizingMask = [.flexibleHeight, .flexibleWidth]
return view
}()
let gradientMask = UIImageView()
// 2 - override init
override init(frame: CGRect) {
super.init(frame: frame)
setup()
}
required init?(coder: NSCoder) {
super.init(coder: coder)
setup()
}
private func setup() {
addSubview(gradientView)
gradientView.mask = gradientMask
}
// 3 - Copy knob position and animations
override func insertSubview(_ view: UIView, at index: Int) {
super.insertSubview(view, at: index)
if index == 3, let view = view as? UIImageView {
gradientMask.image = view.image
gradientMask.frame = view.frame
if let keys = view.layer.animationKeys() {
for key in keys {
guard let animation = view.layer.animation(forKey: key) else {
continue
}
gradientMask.layer.add(animation, forKey: key)
}
}
}
}
}
这不是安全的方法,因为当 apple 更改控件布局时,外观将变得无效,但目前它正在工作。
我用这个方法得到了渐变图像
func gradient(size:CGSize,color:[UIColor]) -> UIImage?{
//turn color into cgcolor
let colors = color.map{[=10=].cgColor}
//begin graphics context
UIGraphicsBeginImageContextWithOptions(size, true, 0.0)
guard let context = UIGraphicsGetCurrentContext() else {
return nil
}
// From now on, the context gets ended if any return happens
defer {UIGraphicsEndImageContext()}
//create core graphics context
let locations:[CGFloat] = [0.0,1.0]
guard let gredient = CGGradient.init(colorsSpace: CGColorSpaceCreateDeviceRGB(), colors: colors as NSArray as CFArray, locations: locations) else {
return nil
}
//draw the gradient
context.drawLinearGradient(gredient, start: CGPoint(x:0.0,y:size.height), end: CGPoint(x:size.width,y:size.height), options: [])
// Generate the image (the defer takes care of closing the context)
return UIGraphicsGetImageFromCurrentImageContext()
}
然后我将分段控件的tintColor设置为渐变:
let gradientImage = gradient(size: listSegmentedControl.frame.size, color: [UIColor.black, UIColor.red])!
listSegmentedControl.tintColor = UIColor(patternImage: gradientImage)
那是行不通的。但是,相同的代码适用于设置 backgroundColor:
let gradientImage = gradient(size: listSegmentedControl.frame.size, color: [UIColor.black, UIColor.red])!
listSegmentedControl.backgroundColor = UIColor(patternImage: gradientImage)
有人知道为什么吗?我真的需要设置 gradient tintColor。非常感谢任何帮助。
编辑:
理想情况下,我希望我的分段控件看起来像这样:
这是一个已知的更改 UISegmentedControl
let sortedViews = listSegmentedControl.subviews.sorted( by: { [=10=].frame.origin.x < .frame.origin.x } )
for (index, view) in sortedViews.enumerated() {
if index == listSegmentedControl.selectedSegmentIndex {
view.tintColor = UIColor(patternImage: gradientImage)
} else {
view.tintColor = UIColor.gray //Whatever the color of non selected segment controller tab
}
}
虽然看起来像一个丑陋的 hack,但我已经使用了很长一段时间并且看起来相当简单。希望对你有帮助。
编辑:
这是你需要的伙计吗?
如果是,让我知道,我会post相同的代码。
编辑 2:
正如 OP 在他的评论中提到的那样,他期望的输出与我在上图中显示的输出相同,并提供了相同的代码。
免责声明:
正如 rmaddy 在他下面的评论中所提到的,这是一个 hack 并利用未记录的(完整 public API 但是一个非常众所周知的 hack 来改变UISegemntedControl 的色调从 iOS 5 开始就存在(这就是我记得的方式,让我知道我是否错了)
因此请注意,在 iOS 的未来版本中,Apple 可能会 更改 UISegemntedControl
中子视图的结构并可能影响你的O/P。我看不到的任何东西都不会导致崩溃,但可能会影响 O/P 在屏幕上的呈现方式。
我已经声明了一个变量,这样 GradientImage 只能生成一次,但是如何使用它取决于您的实现方式
var gradientImage : UIImage! = nil
在ViewDidLoad
中我将gradientImage
和UISegmentedControl
初始化为
override func viewDidLoad() {
super.viewDidLoad()
gradientImage = gradient(size: segmentControl.frame.size, color: [UIColor.black, UIColor.red])!
//I have specified custom font need not necessarily be used
//Font color attribute is important though, usually `UISegementedControl` title takes color from tint color, because we might need a different color for text to highlight above gradient color am using custom font colors
let font = UIFont(name: "HelveticaNeue-Medium", size: 20)
segmentControl.setTitleTextAttributes([NSFontAttributeName : font!, NSForegroundColorAttributeName : UIColor.blue], for: .normal)
segmentControl.setTitleTextAttributes([NSForegroundColorAttributeName : UIColor.white], for: .selected)
//Set the border color and border to `UISegmentedControl` and also make it round corner
segmentControl.layer.borderColor = UIColor(patternImage: gradientImage).cgColor
segmentControl.layer.borderWidth = 2
segmentControl.layer.masksToBounds = true
segmentControl.layer.cornerRadius = 10
//In order to update the selected Segment tint and background color we need to call multiple statements every time selection changes hence I have moved it to the function and called it in viewDidLoad
updateGradientBackground()
}
最后 updateGradientBackground 函数定义与我 post 在我的原始答案中编辑的相同
fileprivate func updateGradientBackground() {
let sortedViews = segmentControl.subviews.sorted( by: { [=13=].frame.origin.x < .frame.origin.x } )
for (index, view) in sortedViews.enumerated() {
if index == segmentControl.selectedSegmentIndex {
//very important thing to notice here is because tint color was not honoring the `UIColor(patternImage` I rather used `backgroundColor` to create the effect and set clear color as clear color
view.backgroundColor = UIColor(patternImage: self.gradientImage)
view.tintColor = UIColor.clear
} else {
//very important thing to notice here is because tint color was not honoring the `UIColor(patternImage` I rather used `backgroundColor` to create the effect and set clear color as clear color
view.backgroundColor = UIColor.white //Whatever the color of non selected segment controller tab
view.tintColor = UIColor.clear
}
}
}
最后,在UISegmentedControl的IBAction中,直接调用
@IBAction func segmentControllerTapped(_ sender: UISegmentedControl) {
self.updateGradientBackground()
}
希望对您有所帮助
为 iOS13+ UISegmentControl 上的旋钮实现了渐变颜色。添加了带渐变的子视图,然后为其设置蒙版并复制动画以匹配原始旋钮。
// 1 - Make subclass
class GradientSegment: UISegmentedControl {
// this could be UIImageView
lazy var gradientView: GradientView = {
let view = GradientView(frame: self.bounds)
view.startColor = ...
view.endColor = ...
view.horizontalMode = true
view.autoresizingMask = [.flexibleHeight, .flexibleWidth]
return view
}()
let gradientMask = UIImageView()
// 2 - override init
override init(frame: CGRect) {
super.init(frame: frame)
setup()
}
required init?(coder: NSCoder) {
super.init(coder: coder)
setup()
}
private func setup() {
addSubview(gradientView)
gradientView.mask = gradientMask
}
// 3 - Copy knob position and animations
override func insertSubview(_ view: UIView, at index: Int) {
super.insertSubview(view, at: index)
if index == 3, let view = view as? UIImageView {
gradientMask.image = view.image
gradientMask.frame = view.frame
if let keys = view.layer.animationKeys() {
for key in keys {
guard let animation = view.layer.animation(forKey: key) else {
continue
}
gradientMask.layer.add(animation, forKey: key)
}
}
}
}
}
这不是安全的方法,因为当 apple 更改控件布局时,外观将变得无效,但目前它正在工作。