如何获取页面中指定的全部元素

一般在开发时不会遇见这种需求,起因也是加入到一家趋于原生开发的公司,开发chrome插件时,遇到了这种场景,选择页面中所有的video标签。我们需要考虑Shadow DOM, iframe中等等情况。简单的通过document.querySelectorAll('video'),他处理当前文档的元素,不能处理Shadow DOM, iframe等情况,下面就来介绍一下吧。

我们通过document.querySelectorAll('video')去进行操作时,他不会去选择通过Shadow DOM定义的自定义元素。如果想选中这种元素内部的内容,我们只能通过当前节点的shadowRoot去获取Shadow DOM定义的跟标签再进行获取。具体请看这里

一般情况下,对于网站的video,我们可以直接获取。但是对于通过 Shadow DOM 定义的标签来说,我们不能直接通过document.querySelectorAll('video')获取到该页面内所有的video元素。我们需要找到自定义标签,然后获取shadow dom根标签在内部进行查找。

js 复制代码
// 例如
    if(location.href.includes("douyu.com")) {
      currentVideos = selectAll("demand-video")?.map(item => item.shadowRoot?.querySelector("video"))
    }

但是这种也是硬编码。只能根据不同网站去做不同处理。可以使用query-selector-shadow-dom 来查找页面所有video标签包括shadow dom中的。

js 复制代码
querySelectorAllDeep("video")

但是这个库有个缺点就是不能获取到iframe中对应的元素。所以我们还是需要通过document.querySelectorAll()来获取。然后进行过滤。

js 复制代码
 let currentVideos = [...new Set([...querySelectorAllDeep("video"), ...selectAll<HTMLVideoElement>("video")])];

但是由于我们在manifest中设置了将当前脚本注入到iframe中,所以就不需要去获取iframe中的video了。

js 复制代码
"content_scripts": [
  {
    "css": ["page.css"],
    "js": ["page.js"],
    "matches": ["*://*/*"],
    "run_at": "document_idle",
    "all_frames": true
  }
],

// 查找页面video, querySelectorAllDeep获取不到iframe中的内容。但是我们通过`"all_frames": true`将该脚本注入到iframe中,所以不需要单独获取iframe中的video
const currentVideos = querySelectorAllDeep("video")

其实这个库的核心原理就是根据传入的根元素收集其内部的所有元素 包括嵌套元素 。(这个根元素可能是一个shadow dom 自定义的元素,所以我们需要通过root.shadowRoot去判断)。然后再进行遍历查找每个元素内部是否存在shadow dom,如果有在进行元素的收集。

最后我们通过Element.matches()方法根据传入的选择器查找到对应的目标元素。

js 复制代码
 function collectAllElementsDeep(selector = null, root = document) {
  const allElements = [];

  // 遍历所有元素
  const findAllElements = function(nodes) {
      for (let i = 0; i < nodes.length; i++) {
          const el = nodes[i];
          allElements.push(el);
          // If the element has a shadow root, dig deeper.
          if (el.shadowRoot) {
              findAllElements(el.shadowRoot.querySelectorAll('*'));
          }
      }
  };

  // 查找当前元素是否是自定义元素。若是在继续加入当前自定义元素下的所有元素
  if(root.shadowRoot) {
      findAllElements(root.shadowRoot.querySelectorAll('*'));
  }

  // 加入跟标签下的所有元素
  findAllElements(root.querySelectorAll('*'));

  // 如果元素被指定的选择器字符串选择,Element.matches() 方法返回 true; 否则返回 false。
  return selector ? allElements.filter(el => el.matches(selector)) : allElements;	
}

xdm,已经快一个月没写框架了,写原生还是了解很多内容。现在也在通读一下mdn文档。感觉还有很多东西不了解!!!

往期年度总结

往期文章

专栏文章

相关推荐
Amy_cx3 分钟前
搭建React Native开发环境
javascript·react native·react.js
代码AI弗森5 分钟前
Python × NumPy」 vs 「JavaScript × TensorFlow.js」生态全景图
javascript·python·numpy
疏狂难除19 分钟前
关于spiderdemo第二题的奇思妙想
javascript·爬虫
渣渣盟39 分钟前
探索Word2Vec:从文本向量化到中文语料处理
前端·javascript·python·文本向量化
Pu_Nine_940 分钟前
Vue 3 + TypeScript 项目性能优化全链路实战:从 2.1MB 到 130KB 的蜕变
前端·vue.js·性能优化·typescript·1024程序员节
沐怡旸44 分钟前
【穿越Effective C++】条款7:为多态基类声明virtual析构函数——C++多态资源管理的基石
c++·面试
云枫晖1 小时前
Webpack系列-Loader
前端·webpack
aggression1 小时前
代码敲击乐:让你了解前端的动静结合和移动端的适配性
前端
yinuo1 小时前
深入理解与实战 Git Submodule
前端
骑自行车的码农1 小时前
React 事件收集函数
前端·react.js