博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
对于单页应用中如何监听 URL 变化的思考
阅读量:6931 次
发布时间:2019-06-27

本文共 2525 字,大约阅读时间需要 8 分钟。

周末开发了一个在 GitHub 中给 repo 增加自定义备注的 chrome 扩展。

开发这个扩展的原因是我在 GitHub 中所 star 的项目实在太多了(截止目前 671 个),有的项目过个几天回来看就忘了为什么 star 了,有的轮子想找的时候发现忘记叫什么了,这么多一个个找实在浪费时间。于是我一直在想有这么个工具,可以自定义对 GitHub 中的项目进行备注,然后可以根据备注进行搜索,于是便有了这个扩展。

如需安装体验请点击 进行安装,源码移步

当然本文并不是讲这个扩展的制作过程,而是在这过程中碰到的一个问题。

以我的 GitHub 账户举例,在 页面,我需要插入一些 dom,使得页面如下:

675542-20180527222010195-306801098.jpg

乍一想,似乎很简单,在这个页面插入 content.js 就行,在 manifest.json 中增加配置:

{  "matches": ["*://github.com/.*tabs=stars"],  "js": ["content-scripts/repoDetail.js"],  "css": [],  "run_at": "document_end" }

但是问题来了,直接打开页面 并没有问题,但是如果从 点击跳转到 ,因为 GitHub 用了 pjax 技术,content.js 其实应该加载在页面 的,但是那个页面并没有参照的 dom 结构(好把我们需要的 dom 插入)。所以问题似乎就转为了,在 pjax 页面,如何能够监听到页面 url 的变化

首先一个很简单的方案是,弄个定时器循环监听,但是太耗费性能了,pass

因为页面是 pjax 跳转,所以 hashchange 这样的事件自然也用不上;而 pushstate 事件是监听浏览器前进后退的,所以也并不符合场景

有个想法很好,重写 pushState 事件(代码来自 ):

(function(history){    var pushState = history.pushState;    history.pushState = function(state) {        if (typeof history.onpushstate == "function") {            history.onpushstate({state: state});        }        // ... whatever else you want to do        // maybe call onhashchange e.handler        return pushState.apply(history, arguments);    }})(window.history);

但是因为 chrome 扩展和源代码之间并不共享作用域,所以重写的 history.pushState 方法在原来页面其实并没有改变,所以也没什么卵用

然后从 chrome API 入手,发现有个 chrome API 可以检测当前 url 改变的 tab(参考 ):

chrome.tabs.onUpdated.addListener(function(tabId, changeInfo, tab) {  if (changeInfo.url) {      console.log('Tab %d got new URL: %s', tabId, changeInfo.url)  }})

这段代码是在 background.js 中,然后一旦侦测到 URL 变化,background.js 和 content.js 通信告知即可。

但是,但是,也有问题,url 一改变确实能监听到,但是 chrome 扩展需要在相应 dom ready 后才能作用(才能在之前 dom 的基础上插入新的 dom),你无法检测到什么时候 dom 已经准备就绪了,所以在这里加个定时器延迟作用是可行的,时间需要自己估算下

这个方法不是很优雅,最后想到,pjax 的过程中,dom 肯定有变化,所以监听 dom 变化是否可行?答案是可以的:

MutationObserver = window.MutationObserver || window.WebKitMutationObserver;var observer = new MutationObserver(function(mutations, observer) {  let url = location.href   let p = /.*\/\/github.com\/.*\?tab=stars.*/  if (p.test(url)) {    initStarsPage()  }})observer.observe(document.getElementById('js-pjax-container'), {  childList: true})$(document).ready(() => {  let url = location.href   let p = /.*\/\/github.com\/.*\?tab=stars.*/  if (p.test(url)) {    initStarsPage()  }})

写到最后,其实并不是监听了 URL,而是监听了 dom 的变化。

最后的最后,意外在 so 上找到 似乎就是对应我的问题,大概看了下,除了我举例的几个方案外,其实我遗漏了最简单的一个方案,既然页面使用了 pjax,那么直接监听 pjax:complete 事件即可:

$(document).on('pjax:complete', function() {  let url = location.href   let p = /.*\/\/github.com\/.*\?tab=stars.*/  if (p.test(url)) {    initStarsPage()  }})

突然让人感觉,有时候 "真相" 往往总是那么简单。

转载地址:http://wbmjl.baihongyu.com/

你可能感兴趣的文章
HBase是什么?为什么要使用HBase?
查看>>
C#高级编程(第8版)
查看>>
《iOS应用逆向工程:分析与实战》迷你书
查看>>
常用PHP框架功能对比表
查看>>
Unity Shader中各种空间及变换方法
查看>>
Android Export aborted because fatal error were fo
查看>>
简略加密字符串
查看>>
工作笔记——Nginx配置HTTPS访问
查看>>
Linux命令大全
查看>>
码农创业:6年80万,有梦想才可能有机会
查看>>
Bitmap算法
查看>>
据说是iOS开发一年总结的笔记
查看>>
pdo数据库操作类
查看>>
在window平台下模拟Liunx使用GCC环境进行编译C的SO库。
查看>>
从0到1使用Kubernetes系列(四):搭建第一个应用程序
查看>>
原来一直纠结MQ的用法,今天看到了一个最经典的例子。
查看>>
向量空间
查看>>
[日推荐]『共享记账』你的私人会计师!
查看>>
Cocoa 用户界面组件使用指导(教程翻译)
查看>>
apache 日志不记录图片 css js 文件访问
查看>>