能不能好好说话?

首字母缩写划词翻译工具

  1. // ==UserScript==
  2. // @name 能不能好好说话?
  3. // @namespace https://lab.magiconch.com/nbnhhsh
  4. // @version 0.15
  5. // @description 首字母缩写划词翻译工具
  6. // @author itorr
  7. // @license MIT
  8. // @icon https://lab.magiconch.com/favicon.ico
  9. // @match *://weibo.com/*
  10. // @match *://*.weibo.com/*
  11. // @match *://*.weibo.cn/*
  12. // @match *://tieba.baidu.com/*
  13. // @match *://*.bilibili.com/
  14. // @match *://*.bilibili.com/*
  15. // @match *://*.douban.com/group/*
  16. // @require https://cdnjs.cloudflare.com/ajax/libs/vue/2.6.11/vue.min.js
  17. // @inject-into content
  18. // @grant none
  19. // ==/UserScript==
  20.  
  21. let Nbnhhsh = ((htmlText,cssText)=>{
  22.  
  23. const API_URL = 'https://lab.magiconch.com/api/nbnhhsh/';
  24.  
  25. const request = (method,url,data,onOver)=>{
  26. let x = new XMLHttpRequest();
  27. x.open(method,url);
  28. x.setRequestHeader('content-type', 'application/json');
  29. x.withCredentials = true;
  30. x.onload = ()=> onOver(x.responseText ? JSON.parse(x.responseText) : null);
  31. x.send(JSON.stringify(data));
  32. return x;
  33. };
  34.  
  35. const Guess = {};
  36. const guess = (text,onOver)=>{
  37. text = text.match(/[a-z0-9]{2,}/ig).join(',');
  38.  
  39. if(Guess[text]){
  40. return onOver(Guess[text]);
  41. }
  42.  
  43. if(guess._request){
  44. guess._request.abort();
  45. }
  46.  
  47. app.loading = true;
  48. guess._request = request('POST',API_URL+'guess',{text},data=>{
  49. Guess[text] = data;
  50. onOver(data);
  51. app.loading = false;
  52. });
  53. };
  54.  
  55. const submitTran = name=>{
  56. let text = prompt('输入缩写对应文字 末尾可通过括号包裹(简略注明来源)','');
  57.  
  58. if(!text || !text.trim || !text.trim()){
  59. return;
  60. }
  61.  
  62. request('POST',API_URL+'translation/'+name,{text},()=>{
  63. alert('感谢对好好说话项目的支持!审核通过后这条对应将会生效');
  64. });
  65. };
  66.  
  67. const transArrange = trans=>{
  68. return trans.map(tran=>{
  69. const match = tran.match(/^(.+?)([(\(](.+?)[)\)])?$/);
  70.  
  71. if(match.length === 4){
  72. return {
  73. text:match[1],
  74. sub:match[3]
  75. }
  76. }else{
  77. return {
  78. text:tran
  79. }
  80. }
  81. })
  82. };
  83.  
  84. const getSelectionText = _=>{
  85. let text = getSelection().toString().trim();
  86.  
  87. if(!!text && /[a-z0-9]/i.test(text)){
  88. return text;
  89. }else{
  90. return null;
  91. }
  92. };
  93.  
  94. const fixPosition = _=>{
  95. let rect = getSelection().getRangeAt(0).getBoundingClientRect();
  96.  
  97. const activeEl = document.activeElement;
  98. if(['TEXTAREA','INPUT'].includes(activeEl.tagName)) rect = activeEl.getBoundingClientRect();
  99.  
  100. let scrollTop = document.documentElement.scrollTop || document.body.scrollTop;
  101.  
  102. let top = Math.floor( scrollTop + rect.top +rect.height );
  103. let left = Math.floor( rect.left );
  104.  
  105. if(top === 0 && left === 0){
  106. app.show = false;
  107. }
  108. app.top = top;
  109. app.left = left;
  110.  
  111. };
  112.  
  113. const timer = _=>{
  114. if(getSelectionText()){
  115. setTimeout(timer,300);
  116. }else{
  117. app.show = false;
  118. }
  119. };
  120.  
  121. const nbnhhsh = _=>{
  122. let text = getSelectionText();
  123.  
  124. app.show = !!text && /[a-z0-9]/i.test(text);
  125.  
  126. if(!app.show){
  127. return;
  128. }
  129.  
  130. fixPosition();
  131.  
  132. guess(text,data=>{
  133. if(!data.length){
  134. app.show = false;
  135. }else{
  136. app.tags = data;
  137. }
  138. });
  139.  
  140. setTimeout(timer,300);
  141. };
  142.  
  143. const _nbnhhsh = _=>{
  144. setTimeout(nbnhhsh,1);
  145. };
  146.  
  147. document.body.addEventListener('mouseup',_nbnhhsh);
  148. document.body.addEventListener('keyup',_nbnhhsh);
  149.  
  150. const createEl = html=>{
  151. createEl._el.innerHTML = html;
  152. let el = createEl._el.children[0];
  153. document.body.appendChild(el);
  154. return el;
  155. };
  156. createEl._el = document.createElement('div');
  157.  
  158.  
  159. createEl(`<style>${cssText}</style>`);
  160.  
  161. const el = createEl(htmlText);
  162.  
  163. const app = new Vue({
  164. el,
  165. data: {
  166. tags:[],
  167. show:false,
  168. loading:false,
  169. top:0,
  170. left:0,
  171. },
  172. methods:{
  173. submitTran,
  174. transArrange,
  175. }
  176. });
  177.  
  178. return {
  179. guess,
  180. submitTran,
  181. transArrange,
  182. }
  183. })(`
  184. <div class="nbnhhsh-box nbnhhsh-box-pop" v-if="show" :style="{top:top+'px',left:left+'px'}" @mousedown.prevent>
  185. <div class="nbnhhsh-loading" v-if="loading">
  186. 加载中…
  187. </div>
  188. <div class="nbnhhsh-tag-list" v-else-if="tags.length">
  189. <div class="nbnhhsh-tag-item" v-for="tag in tags">
  190. <h4>{{tag.name}}</h4>
  191. <div class="nbnhhsh-tran-list" v-if="tag.trans">
  192. <span class="nbnhhsh-tran-item" v-for="tran in transArrange(tag.trans)">
  193. {{tran.text}}<sub v-if="tran.sub">{{tran.sub}}</sub>
  194. </span>
  195. </div>
  196. <div class="nbnhhsh-notran-box" v-else-if="tag.trans===null">
  197. 无对应文字
  198. </div>
  199. <div v-else-if="tag.inputting && tag.inputting.length !==0">
  200. <div class="nbnhhsh-inputting-list">
  201. <h5>有可能是</h5>
  202. <span class="nbnhhsh-inputting-item" v-for="input in tag.inputting">{{input}}</span>
  203. </div>
  204. </div>
  205. <div class="nbnhhsh-notran-box" v-else @click.prevent="submitTran(tag.name)">
  206. 尚未录入,我来提交对应文字
  207. </div>
  208. <a v-if="tag.trans!==null" @click.prevent="submitTran(tag.name)" class="nbnhhsh-add-btn" title="我来提交对应文字"></a>
  209. </div>
  210. </div>
  211. </div>
  212. `, `
  213. .nbnhhsh-box{
  214. font:400 14px/1.4 sans-serif;
  215. color:#333;
  216. }
  217. .nbnhhsh-box-pop{
  218. position: absolute;
  219. z-index:99999999999;
  220. width: 340px;
  221. background:#FFF;
  222. box-shadow: 0 3px 30px -4px rgba(0,0,0,.3);
  223. margin: 10px 0 100px 0;
  224. }
  225. .nbnhhsh-box-pop::before{
  226. content: '';
  227. position: absolute;
  228. top:-7px;
  229. left:8px;
  230. width: 0;
  231. height: 0;
  232. border:7px solid transparent;
  233.  
  234. border-top:1px;
  235. border-bottom-color:#FFF;
  236. }
  237. .nbnhhsh-box sub{
  238. vertical-align: middle;
  239. background: rgba(0,0,0,.07);
  240. color: #777;
  241. font-size: 12px;
  242. line-height:16px;
  243. display: inline-block;
  244. padding: 0 3px;
  245. margin:-2px 0 0 2px;
  246. border-radius: 2px;
  247. letter-spacing: -0.6px;
  248. bottom:0;
  249. }
  250. .nbnhhsh-tag-list{
  251. /*padding:4px 0;*/
  252. }
  253. .nbnhhsh-tag-item{
  254. padding:4px 14px;
  255. position: relative;
  256. }
  257. .nbnhhsh-tag-item:nth-child(even){
  258. background: rgba(0, 99, 255, 0.06);
  259. }
  260. .nbnhhsh-tag-item h4{
  261. font-weight:bold;
  262. font-size:20px;
  263. line-height:28px;
  264. letter-spacing: 1.5px;
  265. margin:0;
  266. }
  267. .nbnhhsh-tran-list{
  268. color:#444;
  269. padding: 0 0 4px 0;
  270. line-height:18px;
  271. }
  272. .nbnhhsh-tran-item{
  273. display: inline-block;
  274. padding: 2px 15px 2px 0;
  275. }
  276.  
  277. .nbnhhsh-inputting-list{
  278. color:#222;
  279. padding: 0 0 4px 0;
  280. }
  281. .nbnhhsh-inputting-list h5{
  282. font-size:12px;
  283. line-height:24px;
  284. color:#999;
  285. margin:0;
  286. }
  287. .nbnhhsh-inputting-item{
  288. margin-right:14px;
  289. display:inline-block;
  290. }
  291. .nbnhhsh-notran-box{
  292. padding:4px 0;
  293. color:#999;
  294. cursor: pointer;
  295. }
  296. .nbnhhsh-add-btn{
  297. position: absolute;
  298. top:0;
  299. right:0;
  300. width: 30px;
  301. line-height: 30px;
  302. text-align: center;
  303. color: #0059ff;
  304. font-size: 16px;
  305. font-weight: bold;
  306. cursor: pointer;
  307. }
  308. .nbnhhsh-add-btn:after{
  309. content: '+';
  310. }
  311. .nbnhhsh-loading{
  312. text-align: center;
  313. color:#999;
  314. padding:20px 0;
  315. }
  316. `);

QingJ © 2025

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