swift - 具有渐变文本颜色的自定义 UILabel 始终变为黑色

标签 swift text colors uilabel gradient

我创建了一个自定义的 UILabel 来显示 grediant 文本,但它总是将文本显示为黑色......当我在 View Controller 上使用相同的代码时它可以工作!

import UIKit

class GradientLabel: UILabel {

    override func awakeFromNib() {
        super.awakeFromNib()
        let gredient = GradientView.init(frame: CGRect(x: 0, y: 0, width: frame.size.width, height: frame.size.height))
        gredient.bottomColor = #colorLiteral(red: 0.1764705926, green: 0.4980392158, blue: 0.7568627596, alpha: 1)
        gredient.topColor = #colorLiteral(red: 0.1764705926, green: 0.01176470611, blue: 0.5607843399, alpha: 1)
        setTextColorToGradient(image: imageWithView(view: gredient)!)
    }


    func imageWithView(view: UIView) -> UIImage? {//bet7awel uiview to image
        UIGraphicsBeginImageContextWithOptions(view.bounds.size, view.isOpaque, 0.0)
        view.drawHierarchy(in: view.bounds, afterScreenUpdates: true)
        let img = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
        return img
    }

    func setTextColorToGradient(image: UIImage) {//beta7'od image we tekteb beha el text fe el label
        UIGraphicsBeginImageContext(frame.size)
        image.draw(in: bounds)
        let myGradient = UIGraphicsGetImageFromCurrentImageContext()
        UIGraphicsEndImageContext()
        textColor = UIColor(patternImage: myGradient!)

    }

}

最佳答案

您不想依赖 awakeFromNib。此外,您真的不想在 init 中执行此操作,因为您希望能够响应大小更改(例如,如果您有约束和 frame首次创建标签后更改)。

相反,从 layoutSubviews 更新渐变,每当 View 的 frame 发生变化时调用它:

@IBDesignable
class GradientLabel: UILabel {

    @IBInspectable var topColor: UIColor = #colorLiteral(red: 0.1764705926, green: 0.4980392158, blue: 0.7568627596, alpha: 1) {
        didSet { setNeedsLayout() }
    }

    @IBInspectable var bottomColor: UIColor = #colorLiteral(red: 0.1764705926, green: 0.01176470611, blue: 0.5607843399, alpha: 1) {
        didSet { setNeedsLayout() }
    }

    override func layoutSubviews() {
        super.layoutSubviews()

        updateTextColor()
    }

    private func updateTextColor() {
        let image = UIGraphicsImageRenderer(bounds: bounds).image { _ in
            let gradient = GradientView(frame: bounds)
            gradient.topColor = topColor
            gradient.bottomColor = bottomColor
            gradient.drawHierarchy(in: bounds, afterScreenUpdates: true)
        }

        textColor = UIColor(patternImage: image)
    }

}

结果是:

example label

注意,

  • 我使用了新的 UIGraphicsImageRenderer 而不是 UIGraphicsBeginImageContextUIGraphicsGetImageFromCurrentImageContextUIGraphicsEndImageContext

  • 我没有手动构建 CGRect,而是使用 bounds

  • 我已将它设为@IBDesignable,以便我可以直接在 Interface Builder 中使用它。我还制作了颜色 @IBInspectable,这样您就可以直接在 IB 中调整颜色,而无需编写代码。很明显,如果你想看到在 IB 中渲染的渐变效果,你只需要这样做。

  • 我已经让它在需要重新布局标签时更新渐变 (a); (b) 每当您更改其中一种颜色时。


就其值(value)而言,这是我在本示例中使用的 GradientView:

@IBDesignable
class GradientView: UIView {

    override class var layerClass: AnyClass { return CAGradientLayer.self }

    private var gradientLayer: CAGradientLayer { return layer as! CAGradientLayer }

    @IBInspectable var topColor: UIColor = .white { didSet { updateColors() } }
    @IBInspectable var bottomColor: UIColor = .blue { didSet { updateColors() } }

    override init(frame: CGRect = .zero) {
        super.init(frame: frame)
        updateColors()
    }

    required init?(coder aDecoder: NSCoder) {
        super.init(coder: aDecoder)
        updateColors()
    }

    private func updateColors() {
        gradientLayer.colors = [topColor.cgColor, bottomColor.cgColor]
    }

}

在这种情况下,因为我们将 layerClass 设置为渐变,所以我们需要做的就是在 init 期间配置它,而基础 layerClass 将负责响应尺寸变化。


或者,您可以使用 CoreGraphics 绘制渐变:

@IBDesignable
class GradientLabel: UILabel {

    @IBInspectable var topColor: UIColor = #colorLiteral(red: 0.1764705926, green: 0.4980392158, blue: 0.7568627596, alpha: 1) {
        didSet { setNeedsLayout() }
    }

    @IBInspectable var bottomColor: UIColor = #colorLiteral(red: 0.1764705926, green: 0.01176470611, blue: 0.5607843399, alpha: 1) {
        didSet { setNeedsLayout() }
    }

    override func layoutSubviews() {
        super.layoutSubviews()

        updateTextColor()
    }

    private func updateTextColor() {
        let image = UIGraphicsImageRenderer(bounds: bounds).image { context in
            let colors = [topColor.cgColor, bottomColor.cgColor]
            guard let gradient = CGGradient(colorsSpace: nil, colors: colors as CFArray, locations: nil) else { return }
            context.cgContext.drawLinearGradient(gradient,
                                                 start: CGPoint(x: bounds.midX, y: bounds.minY),
                                                 end: CGPoint(x: bounds.midX, y: bounds.maxY),
                                                 options: [])
        }

        textColor = UIColor(patternImage: image)
    }

}

这实现了与第一个示例相同的效果,但可能更高效。

关于swift - 具有渐变文本颜色的自定义 UILabel 始终变为黑色,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55923582/

相关文章:

text - 如何在冒号后对齐文本

ios - 如何正确配置添加到现有 TableView Controller 的搜索栏?

ios - 尝试下载文件时 Swift Alamofire 出错

javascript - 将 HTML 文本设置为 JS 变量

Python 在图像数组中查找颜色并用值填充二维数组

html - 为什么 BG 颜色不显示?

javascript - 如何使用 javascript 更改网页上的字体颜色?

ios - 如何在 swift 5 中使用 for 循环创建数组的数组

swift - 没有特征信息(运行时警告)

java - 如何将 InputStream() 转换为 Java Scanner()