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
    }()
相关推荐
Zender Han19 小时前
Flutter Bloc / Cubit 最新详解与实战指南(2026版)
android·flutter·ios
tangweiguo030519871 天前
iOS vs Android 开发对照手册
android·ios
weixin_443478511 天前
flutter组件学习之Cupertino 组件(iOS风格)
学习·flutter·ios
Andy_GF2 天前
iOS26 系统适配-直接隐藏 UIBarButtonItem 的 Liquid glass 效果
ios
wjm0410062 天前
ios学习路线-- swift基础2
学习·ios·swift
开心就好20252 天前
全面解析WhatsApp Web抓包:原理、工具与安全
后端·ios
程序喵大人2 天前
OpenClaw的成功是AI Agent的“iPhone时刻”吗?
人工智能·ios·iphone·openclaw
marteker2 天前
苹果的冰球营销:用iPad设计面具,讲述科技之外的情感故事
科技·ios·ipad
H_老邪2 天前
贪心算法的应用
算法·ios·贪心算法