仿照微信WebView实现ProgressBar

两种进度条逻辑

在网页中,一般我们会用顶部进度条来表示当前网页加载的进度。这里最常见的就是像Safari或Chrome浏览器那样的,用真实的进度百分比来更新进度条。当网速较慢时,进度条几乎完全不动;当网速较快时,进度条则会从大约20%位置嗖一下快速变为100%。

还有一种,就是微信App里的网页加载进度条。这里的进度条反映的不是真实的加载进度,其设计初衷应该就是让网页加载『看起来』更快。经过观察,大约是这样的一个逻辑:

打开网页,进度条就进到10%;

再用3秒钟,进度条从10%走到60%;

再用4秒钟,进度条从60%走到80%;

再用8秒钟,进度条从80%走到90%;

从90%位置开始,进度条开始反应真实的加载进度。若此时网络连接极差,那么将会在90%卡住很久。

在以上的15秒内,若真实进度超过90%,则直接切换到真实进度,所以2秒打开的网页,也只会用2秒,不会固定加载15秒。

从用户提交角度,可以对比不同网速下打开网页时进度条的表现:

  • 网速快,那么微信用3秒就进到60%,然后第4秒刷一下到100%;而Safari则是慢慢地移动到30%左右,然后刷的进到100%。
  • 网速慢,那么微信用15秒加载了90%,只差最后10%加载不出;而Safari则一直处于不足10%的加载状态。

对于小白用户而言,微信的加载条让人『感觉』更快。

除了这一点,WKWebViewestimatedProgress并不会均匀地返回结果。很可能第一次返回结果就是0.5,然后就是0.1。这样Safari加载时,会看到进度条忽快忽慢。

总结起来:

  • estimatedProgress返回值不均匀,这样进度条进度并不平滑;
  • 虚假进度给人『更好』的用户体验。

仿微信网页进度条实现方式Swift4

该实现依赖于对KVO有一定的了解,若不了解,可以参考另一篇:理解KVO - 用Swift在WKWebView中添加进度条

首先,声明必要的变量。

// 使KVO@objc
@objc var webView = WKWebView()
// 使KVO@objcdynamic
@objc dynamic var loadTime: Double = 0.0
// 
var progressLayer: CALayer!
// timer
var timer: Timer?

// webView.estimatedProgressloadTime
var progressObservation: NSKeyValueObservation?
var loadTimeObservation: NSKeyValueObservation?

接着,创建进度条。

func setUpWebView() {
    webView.frame = view.bounds
    webView.navigationDelegate = self
    webView.autoresizingMask = [.flexibleWidth, .flexibleHeight]
    guard let url = URL(string: urlString!) else {
        print("url is nil")
        return
    }
    webView.load(URLRequest(url: url))

    let progress = UIView(frame: CGRect(x: 0, y: 0, width: webView.frame.width, height: 3))
    webView.addSubview(progress)
    progressLayer = CALayer()
    progressLayer.backgroundColor = APPColor.orange.cgColor
    progress.layer.addSublayer(progressLayer!)

    view.addSubview(webView)
    // 10%
    progressLayer!.frame = CGRect(x: 0, y: 0, width: webView.frame.width * 0.1, height: 3)
}

声明遵循WKNavigationDelegate 协议后,在协议方法中添加设置监听对象和包含对应处理方法的闭包。

func webView(_ webView: WKWebView, decidePolicyFor navigationAction: WKNavigationAction, decisionHandler: @escaping (WKNavigationActionPolicy) -> Void) {
    switch navigationAction.navigationType {
    // otherurlother
    case .other:
        print("its an other situation")
    case .reload:
        print("it's a reload situation")
    case .backForward:
        print("its going back")
    case .formResubmitted:
        print("resubmited")
    case .formSubmitted:
        print("from submitted")
    // 
    case .linkActivated:
        print("link activited")
    }

    startProgress()	// progressBar
    destroyTimer()	// timer
    startTimer()	// timer
	
    // 访
    decisionHandler(.allow)
}

func startTimer() {
    // timer0.1loadTime0.1
    timer = Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true, block: { (timer) in
        weak var weakself = self
        weakself?.loadTime += 0.1
    })
}

func destroyTimer() {
    timer?.invalidate()
    loadTime = 0.0
}

func startProgress() {
    progressLayer.opacity = 1
    progressLayer!.frame = CGRect(x: 0, y: 0, width: webView.frame.width * 0.1, height: 3)
    setupObservations()	// 
}

// 
func setupObservations() {
    setupProgressObservation()
    setupLoadTimeObservation()
}

// 
func stopObservations() {
    progressObservation?.invalidate()
    loadTimeObservation?.invalidate()
}

下面是设置监听的具体方法,也是重头戏:

// webView.estimatedProgress
func setupProgressObservation() {
    progressObservation = webView.observe(\.estimatedProgress, options: [.old, .new], changeHandler: { (webView, change) in
        let newValue = change.newValue  ?? 0
        let oldValue = change.oldValue  ?? 0

        weak var weakself = self
        //  0.9loadTime0.9
        if newValue > oldValue && newValue > 0.9 {
            weakself?.progressLayer.frame = CGRect(x: 0, y: 0, width: (weakself?.webView.frame.width)! * CGFloat(newValue), height: 3)
        }

        if newValue == 1.0 {
            // timer
            weakself?.stopObservations()
            weakself?.destroyTimer()

            // progress bar
            let time1 = DispatchTime.now() + 0.4
            let time2 = time1 + 0.1
            DispatchQueue.main.asyncAfter(deadline: time1) {
                weakself?.progressLayer.opacity = 0
            }

            DispatchQueue.main.asyncAfter(deadline: time2) {
                weakself?.progressLayer.frame = CGRect(x: 0, y: 0, width: 0, height: 3)
            }
        }
    })
}

// loadTime
func setupLoadTimeObservation() {
    loadTimeObservation = observe(\.loadTime, changeHandler: { (self, changes) in
        weak var weakself = self
        // 90%loadTime
        if weakself!.progressLayer.frame.width >= weakself!.webView.frame.width * 0.9 { return }

        var ratio = 0.0 // 
        guard let time = weakself?.loadTime else { return }
        if time <= 3 {
            // 350%0.5 / 3
            // 0.1
            ratio = time * 0.5 / 3 + 0.1
        } else if time > 3 && time <= 7 {
            ratio = (time - 3) * 0.2 / 4 + 0.6
        } else if time > 7 && time <= 15 {
            ratio = (time - 7) * 0.1 / 8 + 0.8
        } else if time > 15 && time <  25 {
            ratio = 0.9
        }

        weakself?.progressLayer.frame = CGRect(x: 0, y: 0, width: weakself!.webView.frame.width * CGFloat(ratio), height: 3)
    })
}

这样,进度条的全部实现已经完成。

如果对KVO有不理解,可以参考我的另一篇使用KVO的例子:理解KVO - 用Swift在WKWebView中添加进度条。在这篇文章中,我用于实现进度条的逻辑正是像Safari那样的真实进度。

本人初学,有错误或疏漏之处,欢迎斧正!

参考文档:


comments powered by Disqus