fast search

该脚实现选中文本快速复制,快速搜索的功能

  1. // ==UserScript==
  2. // @name fast search
  3. // @namespace https://github.com/peihaojie/fastSearch
  4. // @author HaoJie
  5. // @description 该脚实现选中文本快速复制,快速搜索的功能
  6. // @description 注意: 需要增加适配网站,请手动修改 @include
  7. // @supportURL https://github.com/peihaojie/fastSearch/issues
  8. // @run-at document-body
  9. // @resource https://at.alicdn.com/t/font_3148281_vf5cr0hy39.css?spm=a313x.7781069.1998910419.83&file=font_3148281_vf5cr0hy39.css
  10. // @include *
  11. // @exclude *://www.iconfont.cn/*
  12. // @version 1.2.1
  13. // @icon 
  14. // ==/UserScript==
  15. class InitSearch {
  16. constructor() {
  17. this.initStatic();
  18. this.initStyle();
  19. this.mouseupListener();
  20. }
  21.  
  22. initStatic() {
  23. this.href =
  24. "https://at.alicdn.com/t/font_3148281_vf5cr0hy39.css?spm=a313x.7781069.1998910419.83&file=font_3148281_vf5cr0hy39.css";
  25.  
  26. this.style = `
  27. *::selection {
  28. background: #CCCCCC;
  29. color: #3399CC;
  30. }
  31.  
  32. .search--btn__wrap {
  33. height: 30px;
  34. border-radius: 5px;
  35. background: #fff;
  36. position: absolute;
  37. cursor: pointer;
  38. box-shadow: 0px 0px 5px 2px #9e9e9e;
  39. overflow: hidden;
  40. display: flex;
  41. z-index: 9999;
  42. }
  43.  
  44. .search--btn__wrap .search--btn {
  45. cursor: pointer;
  46. width: 30px;
  47. height: 30px;
  48. font-size: 20px;
  49. display: inline-flex;
  50. align-items: center;
  51. justify-content: center;
  52. position: relative;
  53. }
  54.  
  55. .search--btn__wrap .search--btn:hover {
  56. background-color: #9e9e9e;
  57. }
  58.  
  59. .iconfont::before {
  60. z-index: 1;
  61. }
  62.  
  63. .iconfont::after {
  64. content: attr(after);
  65. // position: absolute;
  66. // top: 0;
  67. // right: 0;
  68. // bottom: 0;
  69. // left: 0;
  70. // display: flex;
  71. // align-items: center;
  72. // justify-content: center;
  73. font-size: 12px;
  74. }
  75. `;
  76. }
  77.  
  78. initStyle() {
  79. const stylesheet = document.createElement("link");
  80. stylesheet.rel = "stylesheet";
  81. stylesheet.href = this.href;
  82. document.head.appendChild(stylesheet);
  83. const style = document.createElement("style");
  84. const styleText = document.createTextNode(this.style);
  85. style.appendChild(styleText);
  86. document.head.appendChild(style);
  87. }
  88.  
  89. mouseupListener() {
  90. document.body.addEventListener(
  91. "mouseup",
  92. (e) => {
  93. setTimeout(() => {
  94. const selection = document.getSelection();
  95. const selectStr = selection.toString();
  96.  
  97. const delBtn = document.getElementById("btn");
  98. delBtn && document.body.removeChild(delBtn);
  99.  
  100. if (selectStr === "") {
  101. return false;
  102. }
  103.  
  104. const { anchorOffset, focusOffset, isCollapsed } = selection;
  105.  
  106. if (isCollapsed) {
  107. return false;
  108. }
  109.  
  110. const position = this.getPosition(e, anchorOffset, focusOffset);
  111. const btnWrap = this.createBtnWrap(position, selectStr);
  112. document.body.appendChild(btnWrap);
  113. }, 100);
  114. },
  115. false
  116. );
  117. }
  118.  
  119. getPosition(e, anchorOffset, focusOffset) {
  120. let left = e.clientX + 10;
  121. let top = e.pageY;
  122. if (anchorOffset > focusOffset) {
  123. left -= 200;
  124. top -= 45;
  125. }
  126. return { left: `${left}px`, top: `${top}px` };
  127. }
  128.  
  129. createBtnWrap({ left, top }, selectStr) {
  130. const btn = document.createElement("div");
  131. btn.classList.add("search--btn__wrap");
  132. btn.style.left = left;
  133. btn.style.top = top;
  134. btn.setAttribute("id", "btn");
  135. btn.setAttribute("data-text", selectStr);
  136. this.createBtnList(btn);
  137. return btn;
  138. }
  139.  
  140. createBtnList(btn) {
  141. const btnList = [
  142. {
  143. url: "https://www.google.com/search?q=",
  144. title: "google搜索",
  145. after: "谷歌",
  146. icon: "icon-google",
  147. },
  148. {
  149. url: "https://cn.bing.com/search?q=",
  150. title: "必应搜索",
  151. after: "必应",
  152. icon: "icon-Bing",
  153. },
  154. {
  155. url: "https://www.baidu.com/s?wd=",
  156. title: "百度一下",
  157. after: "百度",
  158. icon: "icon-baidu",
  159. },
  160. {
  161. url: "https://www.so.com/s?q=",
  162. title: "360搜索",
  163. after: "360",
  164. icon: "icon-icon-test",
  165. },
  166. {
  167. url: "https://www.sogou.com/web?query=",
  168. title: "搜狗搜索",
  169. after: "搜狗",
  170. icon: "icon-sougoushuru",
  171. },
  172. {
  173. url: "https://www.zhihu.com/search?q=",
  174. title: "知乎搜索",
  175. after: "知乎",
  176. icon: "icon-shejiaotubiao-10",
  177. },
  178. {
  179. url: false,
  180. title: "点击复制",
  181. after: "复制",
  182. icon: "icon-fuzhi",
  183. },
  184. ];
  185. btnList.forEach((btnItem) => this.createBtnItem(btn, btnItem));
  186. }
  187.  
  188. createBtnItem(btn, { url, title, icon, after }) {
  189. const search = document.createElement("div");
  190. search.classList.add("search--btn", "iconfont", icon);
  191. search.setAttribute("title", title);
  192. search.addEventListener("mouseup", (e) => {
  193. const text = btn.getAttribute("data-text");
  194. switch (url) {
  195. case false:
  196. this.copyText(text);
  197. break;
  198. default:
  199. window.open(`${url}${text}`);
  200. break;
  201. }
  202. e.stopPropagation();
  203. document.body.removeChild(btn);
  204. });
  205. btn.appendChild(search);
  206. setTimeout(() => {
  207. const content = window
  208. .getComputedStyle(search, "before")
  209. .getPropertyValue("content");
  210. if (content === "none") {
  211. search.setAttribute("after", after);
  212. }
  213. }, 100);
  214. }
  215.  
  216. copyText(text) {
  217. const input = document.createElement("input");
  218. input.value = text;
  219. document.body.appendChild(input);
  220. input.select();
  221. document.execCommand("copy");
  222. document.body.removeChild(input);
  223. }
  224. }
  225.  
  226. new InitSearch();

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址