仿照微信WebView实现ProgressBar - Swift

两种进度条逻辑

在网页中,一般我们会用顶部进度条来表示当前网页加载的进度。这里最常见的就是像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,属性要添加@objc和dynamic
@objc dynamic var loadTime: Double = 0.0
// 这个是我们的进度条
var progressLayer: CALayer!
// 统计页面加载时间的timer
var timer: Timer?

// 这是用于监听webView.estimatedProgress和loadTime的两个监听对象
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 {
    // other类型,直接从外部赋值url打开页面时,就属于other
    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() {
    // 设置timer为每0.1秒为loadTime赋值,这样可以大约0.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.9之前,进度条由loadTime决定;到0.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 {
            // 前3秒进度条走50%,那么每秒是走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那样的真实进度。

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

参考文档:

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,324评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,303评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,192评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,555评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,569评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,566评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,927评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,583评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,827评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,590评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,669评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,365评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,941评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,928评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,159评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,880评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,399评论 2 342

推荐阅读更多精彩内容