iOS 支持毛玻璃效果修改模糊度

有时候,设计要求的毛玻璃模糊程度,系统的视图无法满足我们的需求,就需要使用UIViewPropertyAnimator 来实现了,代码如下

复制代码
//
// LBVisualEffectView.swift
//  
//
//

import SnapKit
import Foundation

class LBViewPropertyAnimator: UIViewPropertyAnimator {}

public class LBVisualEffectView: UIView {
    private var animator: LBViewPropertyAnimator?
    private var effectView = UIVisualEffectView()

    private var visualEffect: UIVisualEffect? {
        didSet {
            animator?.stopAnimation(true)
            effectView.effect = visualEffect
            buildBlurFraction(effectFraction)
        }
    }

    private var effectFraction: CGFloat?

    private var visualEffectToken: LBThemeToken.VisualEffect? {
        didSet {
            visualEffect = visualEffectToken?.literaryContent
        }
    }

    deinit {
        animator?.stopAnimation(true)
    }

    override public init(frame: CGRect) {
        super.init(frame: frame)
        render()
        makeObservation()
    }

    @available(*, unavailable)
    required init?(coder: NSCoder) {
        fatalError("init(coder:) has not been implemented")
    }

    private func render() {
        addSubview(effectView)
        effectView.snp.remakeConstraints { make in
            make.edges.equalToSuperview()
        }
    }

    private func makeObservation() {
        NotificationCenter
            .default
            .addObserver(
                forName: UIApplication.willEnterForegroundNotification,
                object: nil,
                queue: .main
            ) { [weak self] _ in
                guard let self else {
                    return
                }
                animator?.stopAnimation(true)
                buildBlurFraction(effectFraction)
            }

        LBTheme.addThemeObserver(self)
    }

    public override func didMoveToWindow() {
        super.didMoveToWindow()
        buildBlurFraction(effectFraction)
    }

    @discardableResult
    public func buildVisualEffect(visualEffect: LBThemeToken.VisualEffect) -> Self {
        visualEffectToken = visualEffect
        return self
    }

    @discardableResult
    public func buildVisualEffect(style: UIBlurEffect.Style) -> Self {
        visualEffect = UIBlurEffect(style: style)
        return self
    }

    @discardableResult
    public func buildVisualEffect(blurStyle: UIBlurEffect.Style, vibrancyStyle: UIVibrancyEffectStyle) -> Self {
        visualEffect = UIVibrancyEffect(blurEffect: UIBlurEffect(style: blurStyle), style: vibrancyStyle)
        return self
    }

    @discardableResult
    public func buildBlurFraction(_ fraction: CGFloat?) -> Self {
        effectFraction = fraction
        guard let fraction, fraction >= 0, fraction <= 1 else {
            animator?.stopAnimation(true)
            return self
        }
        effectView.effect = nil
        animator?.stopAnimation(true)
        animator = nil
        animator = LBViewPropertyAnimator(duration: 0.5, curve: .linear)
        animator?.addAnimations { [weak self] in
            guard let self else {
                return
            }
            //[UIBlurEffect effectWithStyle:UIBlurEffectStyleExtraLight]
            effectView.effect = UIBlurEffect(style: .light)
        }
//        animator?.addCompletion { [weak self] _ in
//            guard let self else {
//                return
//            }
//            if let window {
//                buildBlurFraction(fraction)
//            }
//        }
//        animator?.startAnimation()
//        animator?.pauseAnimation()
        animator?.fractionComplete = fraction
        return self
    }
}

extension LBVisualEffectView: LBThemeableCustomItem {
    public func themeDidChange() {
        visualEffect = visualEffectToken?.literaryContent
    }
}

使用方法

复制代码
 private let blurView = {
        let view = LBVisualEffectView()
        view.buildVisualEffect(
            visualEffect: .createVisualEffect(light: .createBlur(style: .light), dark: .createBlur(style: .light))).buildBlurFraction(0.4)
        return view
    }()
相关推荐
鹤卿1236 小时前
(OC)UI学习——网易云仿写
ui·ios·objective-c
不自律的笨鸟6 小时前
最新屏蔽 iOS 系统更新描述文件保姆级教程
ios
开心猴爷7 小时前
Flutter 如何自动上传 可以 IPA 把构建和上传分开处理
后端·ios
秋雨梧桐叶落莳11 小时前
iOS——QQ音乐仿写项目总结
学习·macos·ui·ios·mvc·objective-c·xcode
iUNPo13 小时前
WWDC26 技术解读:Apple Intelligence、Siri AI 与苹果生态的下一步
macos·ios·wwdc
代码的小搬运工13 小时前
【iOS】谓词与正则表达式
ios
恋猫de小郭14 小时前
解析华为 DevEco Code 和小米 MiMo Code,都基于 OpenCode ,有什么区别?
android·前端·ios
wjm0410061 天前
ios内存管理
ios·objective-c·swift·客户端开发
黑科技iOS上架1 天前
ios应用被封号后再次上架很难么?
经验分享·ios
柚鸥ASO优化1 天前
一篇讲透安卓ASO!开发者千万别只盯着iOS了
android·ios·aso优化