Tieba Advanced

贴吧增强 - Gerald倾情打造

  1. // ==UserScript==
  2. // @name Tieba Advanced
  3. // @namespace http://gera2ld.blog.163.com/
  4. // @author Gerald <gera2ld@163.com>
  5. // @icon http://cn.gravatar.com/avatar/a0ad718d86d21262ccd6ff271ece08a3?s=80
  6. // @version 2.7.8
  7. // @description 贴吧增强 - Gerald倾情打造
  8. // @homepageURL http://gerald.top/code/TiebaAdvanced
  9. // @include http://tieba.baidu.com/*
  10. // @exclude http://tieba.baidu.com/tb/*
  11. // @require https://greasyfork.org/scripts/144/code.user.js
  12. // @grant none
  13. // ==/UserScript==
  14.  
  15. // Mask layer
  16. var mask=$('<div class=ge_mask style="padding:100px 100px 20px;">').appendTo('body');
  17. DELAY=2000;
  18.  
  19. // 初始化贴子管理面板
  20. function initPostManager() {
  21. if(utils.postManager) return;
  22. utils.addStyle('\
  23. #ge_tm{display:none;border-collapse:separate;width:100%;height:100%;background:white;color:#333;padding:20px;border-radius:20px;shadow:0 1px 5px #333;border-spacing:5px;table-layout:fixed;}\
  24. #ge_tm tr{height:1px;}\
  25. #ge_tm tr.ge_td{height:auto;}\
  26. .ge_td>td{width:50%;height:100%;vertical-align:top;position:relative;}\
  27. .ge_td>td>*{width:100%;height:100%;border:1px solid;overflow:auto;background:transparent;'+(window.opera?'position:absolute;':'')+'}\
  28. ');
  29. var tm=$('<table id=ge_tm>').appendTo(mask);
  30. tm.listItems=function(t,e,x,s){
  31. var d=[];
  32. if(x) d.push('<option>'+x+'</option>');
  33. t.list.forEach(function(i){d.push('<option>'+i.name+'</option>');});
  34. e.html(d.join(''));
  35. if(s) {x=x?1:0;t.load(s-x);e.prop('selectedIndex',t.last+x);}
  36. };
  37. tm.newItem=function(e,d) {
  38. tm.list.load(tm.list.length-1);
  39. if(!tm.list.cur||tm.list.cur.data) {
  40. tm.list.load(tm.list.push(d));
  41. $('<option>').appendTo(ti).text(tm.list.cur.name);
  42. } else {
  43. tm.list.cur.type=d.type;
  44. tm.list.cur.data=d.data;
  45. }
  46. $(ti).prop('selectedIndex',tm.list.last);
  47. editItem();
  48. };
  49. var th=document.createElement('p'),tk='innerText' in th?'innerText':'textContent';
  50. function h2t(h){ // html to text
  51. th.innerHTML=h.replace(/<br>/g,'\n');
  52. return th[tk];
  53. }
  54. function t2h(t){ // text to html
  55. th[tk]=t;
  56. return th.innerHTML.replace(/ /g,'&nbsp; ').replace(/(^| ) /g,'&nbsp;$1').replace(/\r?\n/g,'<br>');
  57. }
  58. function editItem(e) {
  59. if(e) tm.list.load(ti.prop('selectedIndex'),1);
  60. var t=tm.list.cur;
  61. tc.prop('disabled',!t);
  62. if(!t) t={type:'s',data:''};
  63. tt.val(t.type);
  64. if(['j','h','H'].indexOf(t.type)>=0) tc.val(t.data);
  65. else tc.val(h2t(t.data));
  66. liveShow();
  67. }
  68. function saveItem(e) {
  69. var t=tm.list.cur;if(!t) return;
  70. switch(t.type=tt.val()) {
  71. case 'j':try{eval(t.data=tc.val());}catch(e){}break;
  72. case 's':t.data=t2h(tc.val());break;
  73. case 'H':t.data=tc.val();break;
  74. default:tv.find('img').each(function(i,e){
  75. e.setAttribute('pic_type',1);
  76. unsafeWindow.EditorUI.resizeImage(e,560);
  77. });
  78. tc.val(t.data=tv.html());
  79. }
  80. }
  81. function liveShow(e) {
  82. function show() {
  83. if(--count) return;
  84. var t=tt.val(),s;
  85. if(t=='j') try{s=eval(tc.val());}catch(e){s='<font color=red>JS代码有误!</font>';}
  86. else s=tc.val();
  87. if(t=='s') s=t2h(s);
  88. else if(t=='H') s=s.split('\n').shift();
  89. tv.html(s);
  90. }
  91. count++;
  92. setTimeout(show,500);
  93. }
  94. var count=0;
  95. tm.loadPanel=function(t,n,c) {
  96. tm.list=t;tn.text(n);tm.callback=c;
  97. tm.listItems(t,ti);editItem(1);
  98. mask.fadeIn('fast',function() {
  99. tm.css({display:'table'}).animate({top:'0px',left:'0px'},300);
  100. });
  101. };
  102. var c=$('<td colspan=2>').appendTo($('<tr>').appendTo(tm)),
  103. tn=$('<strong class=ge_rsep>').appendTo(c),
  104. ti=$('<select>').appendTo(c).change(editItem);
  105. $('<span class="ge_sbtn ge_rsep">改名</span>').appendTo(c).click(function(e) {
  106. if(!tm.list.cur) return;
  107. var t=prompt('修改名称:',tm.list.cur.name);
  108. if(t) {tm.list.cur.name=t;ti.children('option:eq('+tm.list.last+')').text(t);}
  109. });
  110. var tt=$('<select>').appendTo($('<label class=ge_rsep>类型:'+utils.getLink('advanced',{title:'帮助',html:'(?)'})+'</label>').appendTo(c)).html('<option value="s" checked>普通字串</option><option value="h">HTML代码</option><option value="H">HTML随机</option><option value="j">JS代码</option>').change(liveShow).blur(saveItem);
  111. $('<span class=ge_sbtn>添加</span>').appendTo(c).click(tm.newItem);
  112. $('<span class=ge_sbtn>删除</span>').appendTo(c).click(function() {
  113. var l=tm.list.last;tm.list.pop(l);ti.children().eq(l).remove();editItem(1);
  114. });
  115. c=$('<div style="float:right;"></div>').appendTo(c);
  116. $('<span class=ge_sbtn>关闭</span>').appendTo(c).click(function() {
  117. tm.list.save();if(tm.callback) tm.callback();
  118. tm.animate({top:innerHeight+'px'},300,function() {$(this).hide();mask.fadeOut('fast');});
  119. });
  120. $('<tr><td>编辑框</td><td align=right>预览框</td></tr>').appendTo(tm);
  121. c=$('<tr class=ge_td>').appendTo(tm);
  122. var tc=$('<textarea>').appendTo($('<td>').appendTo(c)).blur(saveItem).keyup(liveShow).mouseup(liveShow);
  123. var tv=$('<div>').appendTo($('<td>').appendTo(c));
  124. utils.postManager=tm;
  125. }
  126. // 灌水
  127. function initAddWater(editor) {
  128. initPostManager();
  129. var tails=utils.list('tails',null,function(){return {type:'s',data:'',name:'新尾巴'};},[
  130. {type:'j',name:'UA',data:'"——我喂自己袋盐<br>&gt;&gt;"+navigator.userAgent'},
  131. {type:'h',name:'求妹纸',data:'<img pic_type="1" class="BDE_Image" src="http://imgsrc.baidu.com/forum/w%3D580/sign=6ca77dcee5dde711e7d243fe97edcef4/b03533fa828ba61e111605e44134970a314e5905.jpg" width="560" height="11"><br><img pic_type="1" src="http://static.tieba.baidu.com/tb/editor/images/tsj/t_0010.gif" class="BDE_Smiley" height="40" width="40">少壮不追妹,老大去相亲'},
  132. ]).load(),water=utils.list('water',null,function(){return {type:'s',data:'',name:'新水贴'};},[
  133. {type:'s',name:'打酱油',data:'我是打酱油的~'},
  134. ]).load();
  135. function initTails(){utils.postManager.listItems(tails,ti,'随机',utils.getObj('tailindex',1));}
  136. function initWater(){utils.postManager.listItems(water,wi,'随机',utils.getObj('waterindex',0));}
  137. function getItem(t,s){
  138. var l=s.prop('selectedIndex'),L=t.length;if(!L) return;
  139. if(!l) l=Math.floor(Math.random()*L); else l--;
  140. t=t.list[l];var d=t.data;
  141. if(t.type=='j') d=eval(d);
  142. else if(t.type=='H') {d=d.split('\n');d=d[Math.floor(Math.random()*d.length)];}
  143. return d;
  144. }
  145. var op=utils.addRPopup(utils.addSButton('灌 水')).panel;
  146. $('<div class=ge_sbtn style="cursor:default">智能灌水</div>').appendTo(op);
  147. var ti=$('<select class=ge_rsep>').appendTo($('<label>尾巴:</label>').appendTo(op)).change(function(e){utils.setObj('tailindex',this.selectedIndex);});
  148. $('<br>').appendTo(op);
  149. var tail=utils.bindProp($('<input type=checkbox>').prependTo($('<label class=ge_rsep>自动附加尾巴</label>').appendTo(op)),'checked','usetail',true);
  150. $('<br>').appendTo(op);
  151. $('<span class=ge_sbtn>存为新尾巴</span>').appendTo(op).click(function(e){
  152. utils.postManager.loadPanel(tails,'尾巴管理',initTails);
  153. utils.postManager.newItem(e,{type:'h',name:'新尾巴',data:editor.$body.html()});
  154. });
  155. $('<span class=ge_sbtn>管理</span>').appendTo(op).click(function(e){utils.postManager.loadPanel(tails,'尾巴管理',initTails);});
  156. $('<hr>').appendTo(op);
  157. var wi=$('<select class=ge_rsep>').appendTo($('<label>水贴:</label>').appendTo(op)).change(function(e){utils.setObj('waterindex',this.selectedIndex);});
  158. $('<br>').appendTo(op);
  159. $('<span class=ge_sbtn>存为新水贴</span>').appendTo(op).click(function(e){
  160. utils.postManager.loadPanel(water,'水贴管理',initWater);
  161. utils.postManager.newItem(e,{type:'h',name:'新水贴',data:editor.$body.html()});
  162. });
  163. $('<span class=ge_sbtn>管理</span>').appendTo(op).click(function(e){utils.postManager.loadPanel(water,'水贴管理',initWater);});
  164. $('<br>').appendTo(op);
  165. $('<span class=ge_sbtn>载入</span>').appendTo(op).click(function(e){
  166. editor.execCommand('inserthtml',getItem(water,wi));
  167. });
  168. $('<span class=ge_sbtn>发表</span>').appendTo(op).click(function(e){
  169. editor.$body.html(getItem(water,wi));
  170. unsafeWindow.test_poster.post();
  171. });
  172. /*$('<span class=ge_sbtn>人工置顶</span>').appendTo(op).click(function(e){
  173. function post(){PostHandler.post(rich_postor._option.url,b,delay,function(){});}
  174. function delay(m){
  175. if(m) {
  176. if(m.no) d+=1000; else {d=DELAY;e.text('停止('+(++c)+')');}
  177. }
  178. if(!m||!m.no) b.content=getItem(water,wi);
  179. setTimeout(post,d);
  180. }
  181. (e=$(this)).unbind('click').text('停止').click(function(){location.reload();});
  182. var c=0,d=0;b=rich_postor._getData();delay();
  183. });*/
  184. var tailed=false;
  185. initTails();initWater();
  186. utils.wait(unsafeWindow,'test_poster',function(){
  187. utils.hook(unsafeWindow.test_poster,'post',{before:function(){
  188. var t=getItem(tails,ti);
  189. if(!tail.prop('checked')||!t||tailed) return;
  190. editor.$body.append('&nbsp;<br><br>'+t);tailed=true; // 加个空格以免破坏@
  191. }});
  192. utils.hook(unsafeWindow.test_poster,'showPostSuccess',{after:function(){
  193. tailed=false;
  194. }});
  195. });
  196. }
  197.  
  198. // 尾页直达功能
  199. function initLastPage() {
  200. utils.addStyle('.threadlist_rep_num{cursor:pointer;}');
  201. $('.threadlist_rep_num').prop('title','直达尾页').click(function(e){
  202. e=$(e.target);e.unbind('click');
  203. var s='',d=JSON.parse(e.parents('.j_thread_list').attr('data-field'));
  204. setInterval(function(){
  205. if(s.length>2) s=''; else s+='.';
  206. e.html(s);
  207. },300);
  208. d='/p/'+d.id;
  209. $.get(d,function(data){
  210. var m=data.match(/共<span class="red">(\d+)/)[1];
  211. if(m=='1') m=''; else m='?pn='+m;
  212. location.href=d+m;
  213. });
  214. });
  215. }
  216.  
  217. // 召唤术增强
  218. var calllist=utils.list('calllist','calllast',function(){return {name:'新列表',data:[]};},true).load();
  219. function initCard() {
  220. function fix(){
  221. var t=this._j_card,c=calllist.cur.data;
  222. setTimeout(function(){
  223. function update(a) {a.html((j<0?'加入':'移出')+'@列表');}
  224. var u=t.getData().un,j=c.indexOf(u),w=t.find('.interaction_wrap'),a;
  225. if(w.length) update(a=$('<a href=# class="btn-small btn-encourage">').appendTo(w).click(function(e){
  226. e.preventDefault();
  227. if(j<0) {j=c.length;c.push(u);}
  228. else {for(;j<c.length-1;j++) c[j]=c[j+1];c.pop();j=-1;}
  229. calllist.save();
  230. update(a);
  231. }));
  232. },0);
  233. }
  234. function hook(b){
  235. utils.hook(b.__proto__,'buildVisitCard',{before:function(){
  236. utils.hook(this._visit_card,'setContent',{after:fix});
  237. }});
  238. }
  239. unsafeWindow._.Module.use("ihome/widget/UserVisitCard",{},hook);
  240. unsafeWindow._.Module.use("puser/widget/UserVisitCard",{},hook);
  241. }
  242. function initCall(editor) {
  243. var pl,sl,be,bs,c=calllist;
  244. utils.addStyle('\
  245. #callList{border:1px solid;height:125px;overflow:auto;background:white;margin:0 auto;width:380px;}\
  246. #callList a{padding:2px;border-radius:2px;margin:2px;display:inline-block;}\
  247. #callList a.selected{background:limegreen;color:white}\
  248. .edui-btn-toolbar .edui-btn .edui-button{width:21px;height:20px;line-height:20px;text-align:center;font-size:16px;font-weight:bold;cursor:pointer;}\
  249. ');
  250. function newList(e) {
  251. c.load(c.push());$('<option>').appendTo(sl).text(c.cur.name);
  252. sl.prop('selectedIndex',c.last);
  253. editList(e);
  254. }
  255. function editList(e) {
  256. if(e) c.load(sl.prop('selectedIndex')); else sl.prop('selectedIndex',c.last);
  257. pl.empty();
  258. c.cur.data.forEach(function(i){$('<a href=#>').html(i).appendTo(pl);});
  259. pl.prop('contenteditable',false);
  260. }
  261. function loadLists(p) {
  262. var op=p.panel;
  263. c.load();op.empty();
  264. $('<div class=ge_sbtn style="cursor:default">超级召唤</div>').appendTo(op);
  265. sl=$('<select>').appendTo($('<label>选择名单:</label>').appendTo(op)).change(editList);
  266. $('<span class="ge_sbtn ge_rsep">改名</span>').appendTo(op).click(function(e) {
  267. e.preventDefault();
  268. var t=prompt('列表名称:',c.cur.name);
  269. if(t) {sl.children(':eq('+c.last+')').text(t);c.cur.name=t;c.save();}
  270. });
  271. $('<span class=ge_sbtn>新建列表</span>').appendTo(op).click(newList);
  272. $('<span class="ge_sbtn ge_rsep">删除列表</span>').appendTo(op).click(function(e){
  273. e.preventDefault();
  274. var l=c.last;c.pop(l);editList();sl.children().eq(l).remove();
  275. });
  276. pl=$('<div id=callList>').appendTo(op).click(function(e){
  277. e.preventDefault();
  278. e=e.target;if(e.tagName=='A') $(e).toggleClass('selected');
  279. }).dblclick(function(e){
  280. e.stopPropagation();
  281. var s=window.getSelection();
  282. if(!s.rangeCount) return;
  283. var r=s.getRangeAt(0),c=r.startContainer,k=r.startOffset;
  284. var i=c.data.substr(0,k).search(/\s\S*$/),j=c.data.substr(k).search(/\s/);
  285. r.setStart(c,i+1);r.setEnd(c,j<0?c.data.length:k+j);
  286. s.removeAllRanges();s.addRange(r); // Compatible with Chrome
  287. });
  288. $('<label>名单管理:</label>').appendTo(op);
  289. be=$('<span>').appendTo(op);
  290. $('<span class=ge_sbtn>编辑</span>').appendTo(be).click(function(e){
  291. bs.show();be.hide();
  292. pl.prop('contenteditable',true);
  293. pl.text(c.cur.data.join(' '));
  294. });
  295. $('<span class=ge_sbtn>全选/不选</span>').appendTo(be).click(function(e){
  296. e.preventDefault();
  297. var a=pl.children('a:not(.selected)');
  298. if(a.length) a.addClass('selected'); else pl.children('a').removeClass('selected');
  299. });
  300. bs=$('<span>').appendTo(op).hide();
  301. $('<span class=ge_sbtn>去重</span>').appendTo(bs).click(function(e){
  302. var d=pl.text().replace(/^\s+|\s+$/,'').split(/\s+/),h={};
  303. d.forEach(function(i){h[i]=0;});
  304. pl.text(Object.getOwnPropertyNames(h).join(' '));
  305. });
  306. $('<span class="ge_sbtn ge_rsep">完成</span>').appendTo(bs).click(function(e){
  307. c.cur.data=pl.text().replace(/^\s+|\s+$/,'').split(/\s+/);
  308. c.save();editList(e);be.show();bs.hide();
  309. });
  310. $('<span>空格隔开,双击选中一个名字</span>').appendTo(bs);
  311. var b=$('<div style="float:right">').appendTo(op);be=be.add(b);
  312. $('<span class=ge_sbtn title="普通召唤,超过十个ID将会失败">召唤</span>').appendTo(b).click(function(e){
  313. var se=p==pM?editor:unsafeWindow.LzlEditor._s_p._se;
  314. pl.children('a.selected').each(function(i,e){se.execCommand('inserthtml','@'+e.innerHTML+'&nbsp;');});
  315. p.onclose();
  316. });
  317. /*$('<span class=ge_sbtn title="插入一个占位符,将自动替换成召唤名单">自动召唤</span>').appendTo(b).click(function(e){
  318. e=[];pl.children('a.selected').each(function(){e.push(this.innerHTML);});
  319. p.onclose();
  320. if(e.length) {
  321. p.holder.names=e;
  322. e=p.holder==E?editor:unsafeWindow.LzlEditor._s_p._se;
  323. e.execCommand('inserthtml','<img class=BDE_Smiley title="将在此自动插入召唤名单" alt="召唤列表" height=18>');
  324. }
  325. });*/
  326. c.list.forEach(function(i){$('<option>').text(i.name).appendTo(sl);});
  327. editList();
  328. }
  329. var l=/<img [^>]*?alt="召唤列表"[^>]*>/;
  330. function addNames(e,n){
  331. if(n.splice) n='@'+n.splice(0,10).join(' @')+' ';
  332. return e.replace(l,n);
  333. }
  334. // 主编辑框
  335. var o=utils.addTButton($('<div unselectable="on" class="edui-button" style="color:red" title="召唤">@</div>')),
  336. pM=utils.addTPopup(o,loadLists);
  337. /*utils.hook(unsafeWindow.PostHandler,'post',{before:function(f,a){
  338. function post(){f.hook_func(a[0],a[1],E.names?delay:a[2],a[3]);}
  339. function delay(m){
  340. if(m){if(m.no) d+=1000; else d=DELAY;}
  341. if((!m||!m.no)&&E.names) {
  342. a[1].content=addNames(e,E.names);
  343. if(!E.names.length) delete E.names;
  344. }
  345. setTimeout(post,d);
  346. }
  347. var e=a[1].content,d=0;f.hookStop();
  348. if(E.names&&e.search(l)<0) delete E.names;
  349. delay();
  350. }});*/
  351. // 楼中楼
  352. /*lzl_init.push(function(){
  353. utils.hook(unsafeWindow.SimplePostor.prototype,'_submitData',{before:function(f){
  354. function post(){
  355. $.tb.post(FORUM_POST_URL.postAdd,b,delay);
  356. }
  357. function delay(m){
  358. if(o.names) {
  359. if(m){if(m.no) d+=1000; else d=DELAY;}
  360. if(!m||!m.no) {
  361. b.content=addNames(a,o.names);
  362. if(!o.names.length) delete o.names;
  363. }
  364. setTimeout(post,d);
  365. } else location.reload();
  366. }
  367. if(o.names&&this._se.editArea.innerHTML.search(l)>=0) {
  368. f.hookStop();var d=0,b=this._getData(),a=b.content;delay();
  369. }
  370. }});
  371. });*/
  372. var o=$('<span title="召唤" style="color:red">@</span>'),p=utils.addLPopup(o,loadLists);
  373. lzl_init.push(p.getfunc('reinit'));
  374. addPButton(o.click(p.ontoggle));
  375. }
  376.  
  377. // 楼中楼初始化
  378. var lzl_init=[],lzl_buttons=[];
  379. function addPButton(o) { // 新增楼中楼按钮
  380. lzl_buttons.push(o.addClass('lzl-button'));
  381. }
  382. function initLzL(editor) {
  383. // 楼中楼初始化
  384. function fixLzl() {
  385. var p=editor.cur_sec.find('.lzl_panel_btn');
  386. p.parent().css('width','50%').prev().css('width','auto');
  387. p.next().css({left:'auto',right:24});
  388. lzl_buttons.forEach(function(i){utils.insertButton(p,i);});
  389. lzl_init.forEach(function(i){i();});
  390. }
  391. if(lzl_buttons.length)
  392. utils.addStyle('.lzl-button{width:22px;height:20px;margin:2px 6px 0;float:right;cursor:pointer;line-height:20px;font-size:16px;font-weight:bold;text-align:center;}');
  393. if(editor.cur_sec) fixLzl();
  394. utils.hook(editor,'_buildEditor',{after:fixLzl});
  395. }
  396.  
  397. function initEditor(editor){
  398. initAddWater(editor); // 灌水+尾巴
  399. initCall(editor); // 召唤增强,召唤列表
  400. }
  401.  
  402. // 以下为模块调用,可将不需要的模块注释,不要改变顺序
  403. if($&&PageData&&PageData.user) {
  404. initCard(); // 用户卡片上添加召唤按钮
  405. // 以下模块无需登录
  406. if(PageData.thread) { // 以下模块仅在帖子浏览页面加载
  407. } else {
  408. initLastPage(); // 尾页直达功能
  409. }
  410. //以下模块仅在登录时加载
  411. if(PageData.user.is_login&&unsafeWindow.PosterContext&&unsafeWindow.PosterContext.isPostAllowed()) {
  412. utils.wait(unsafeWindow,'test_editor',initEditor);
  413. utils.wait(unsafeWindow,'LzlEditor',initLzL);
  414. }
  415. }