GitHub Sortable Filelist

appends sorting function to github directories

当前为 2016-01-27 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name GitHub Sortable Filelist
  3. // @namespace trespassersW
  4. // @description appends sorting function to github directories
  5. // @include https://github.com/*
  6. // @version 16.01.27
  7. // 16.01.27 * bugfix
  8. // 15.08.12 ++ octicons for file extensions
  9. // 15.08.07 + case-insensitive sorting
  10. // 15.05.07 sorting is now faster
  11. // .12 new age format; fix for chrome
  12. // .10 datetime auto-updating fix; right-aligned datetime column; proper local time; .ext sorting fix;
  13. // .8 sorting by file extention
  14. // .7 date/time display mode switching
  15. // @created 2014-11-10
  16. //
  17. // @author trespassersW
  18. // @license MIT
  19. // @icon https://i.imgur.com/8buFLcs.png
  20. // (C) Icon: Aaron Nichols CC Attribution 3.0 Unported
  21. // @run-at document-end
  22. // @grant unsafeWindow
  23. // ==/UserScript==
  24.  
  25. if(document.body && document.querySelector('#js-repo-pjax-container')){
  26.  
  27. var llii=0, _l= function(){/* * /
  28. for (var s=++llii +':', li=arguments.length, i = 0; i<li; i++)
  29. s+=' ' + arguments[i];
  30. console.log(s)
  31. /* */
  32. }
  33. //_l=console.log.bind(console);
  34. var fakejs = // avoid compiler warning
  35. (function(){ "use strict";
  36.  
  37. var ii=0,tt;
  38. var d0=[0,0,1];
  39. var C=[{c:1, d: 0, s: 0},{c:2, d: 0, s: 0},{c:3, d: 1, s: 0}];
  40. var ASC;
  41. var oa=[],ca=[],clock,ext,dtStyle,upc;
  42. var D=document, TB;
  43. var catcher,locStor;
  44. var prefs={dtStyle:0, ext: 0, upc: 1};
  45. var W= unsafeWindow || window;
  46.  
  47. // see: https://octicons.github.com/
  48. var extIcon=[
  49. //0...........1..............2..............3..............4.......
  50. "octoface" ,"zap" ,"list-unordered","paintcan" ,"eye"
  51. //5...........6..............7..............8..............9.......
  52. ,"globe" ,"file-binary" ,"file-zip" ,"file-pdf" ,"megaphone"
  53. //10..........11.............12.............13.............14......
  54. ,"gear" ,"triangle-right","ruby" ,"info" ,"device-camera"
  55. //15..........16.............17.............18.............19......
  56. ,"pencil" ,"terminal" ,"book"
  57. ]
  58. var extList={
  59. md:0,
  60. js:1,jsm:1,
  61. json:2,xml:2,xul:2,rdf:2,yml:2,
  62. css:3,scss:3,less:3,
  63. png:4,bmp:4,gif:4,cur:4,ico:4,svg:4,
  64. htm:5,html:5,php:5,
  65. bin:6,exe:6,dll:6,
  66. zip:7,rar:7,arj:7,
  67. pdf:8,
  68. wav:9,mp3:9,ogg:9,mp4:9,aac:9,
  69. cfg:10,ini:10,
  70. c:11,cpp:11,cc:11,h:11,hpp:11,asm:11,m:11,
  71. rb:12,py:12,
  72. EmptyExt:13,
  73. jpg:14,jpeg:14,
  74. pl:15,java:15,jar:15,cs:15,
  75. sh:16,mak:16,cmd:16,bat:16,
  76. doc:17,rtf:17,djvu:17
  77. }
  78.  
  79. function stickStyle(css){
  80. var s=document.createElement("style"); s.type="text/css";
  81. s.appendChild(document.createTextNode(css));
  82. return (document.head||document.documentElement).appendChild(s);
  83. }
  84. function insBefore(n,e){
  85. return e.parentNode.insertBefore(n,e);
  86. }
  87. function insAfter(n,e){
  88. if(e.nextElementSibling)
  89. return e.parentNode.insertBefore(n,e.nextElementSibling);
  90. return e.parentNode.appendChild(n);
  91. }
  92. function outerNode(target, node) {
  93. if (target.nodeName==node) return target;
  94. if (target.parentNode)
  95. while (target = target.parentNode) try{
  96. if (target.nodeName==node)
  97. return target;
  98. }catch(e){};
  99. return null;
  100. }
  101. function savePrefs(){
  102. if(locStor) locStor.setItem('GHSFL',JSON.stringify(prefs));
  103. }
  104.  
  105. function css(){
  106. stickStyle('\
  107. .fsort-butt,\n\
  108. .tables.file td.content, .tables.file td.message, .tables.file td.age\n\
  109. {position: relative; }\n\
  110. \n\
  111. .fsort-butt:before{\n\
  112. position: absolute; display: inline-block;\n\
  113. cursor: pointer;\n\
  114. text-align:center; vertical-align: top;\n\
  115. width: 18px; height: 14px;\n\
  116. line-height: 14px;\n\
  117. padding:0; margin:0;\n\
  118. border-color: transparent;\n\
  119. border-width: 0;\n\
  120. content: "";\n\
  121. opacity: .2;\n\
  122. z-index: 99;\
  123. }\n\
  124. .fsort-butt.fsort-asc:before,.fsort-butt.fsort-desc:before{\n\
  125. left:1.5em; top: -1em;\n\
  126. }\n\
  127. td.age .fsort-butt.fsort-asc:before, td.age .fsort-butt.fsort-desc:before{\n\
  128. left: 4.5em; \n\
  129. }\n\
  130. .fsort-asc:before,.fsort-desc:before{\n\
  131. background-color: #48C;\n\
  132. }\n\
  133. .fsort-asc:before{\n\
  134. border-radius: 24px 24px 8px 8px;\n\
  135. }\n\
  136. .fsort-desc:before{\n\
  137. border-radius: 8px 8px 24px 24px;\n\
  138. }\n\
  139. .fsort-asc:before,\n\
  140. .fsort-desc.fsort-sel:hover:before\n\
  141. {\n\
  142. content: url();\n\
  143. }\n\
  144. .fsort-desc:before,\n\
  145. .fsort-asc.fsort-sel:hover:before{content: url();\n\
  146. }\n\
  147. \n\
  148. .fsort-butt.fsort-sel:before\n\
  149. {\n\
  150. background-color: #4183C4 !important;\n\
  151. opacity:.6 !important;\n\
  152. }\n\
  153. \n\
  154. span.fsort-butt:hover:before\n\
  155. ,span.fsort-butt:hover span:before\n\
  156. { opacity: 1 !important;}\n\
  157. \n\
  158. #fsort-clock:before{\n\
  159. left:6.5em; top: -15px; \n\
  160. text-align:center; vertical-align: top; top:-15px;\n\
  161. width: 16px; height: 16px;\
  162. border-radius: 16px;\n\
  163. content: url(\n\
  164. );}\n\
  165. .fsort-on:before{ background-color: #4183C4 !important; } \n\
  166. #fsort-ext:before{\n\
  167. left:4em; top:-14px;\n\
  168. width:28px; height: 14px;\n\
  169. border-radius: 6px;\n\
  170. content:url();\n\
  171. }\n\
  172. #fsort-ext:before{ background-color: #BBB}\n\
  173. /* 150806 uppercase */\n\
  174. #fsort-upc:before{\n\
  175. left:7em; top:-14px;\n\
  176. width:16px; height: 16px;\n\
  177. border-radius: 0 4px 0 4px;\n\
  178. content:url();\n\
  179. }\n\
  180. #fsort-upc:before{ background-color: #BBB}\n\
  181. \n\
  182. table.files td.age .css-truncate.css-truncate-target{\n\
  183. width: 99% !important; \n\
  184. max-width: none !important;\n\
  185. }\n\
  186. /*table.files td.age span.css-truncate time{\n\
  187. position: relative !important;\n\
  188. }*/\n\
  189. .fsort-time {\n\
  190. visibility: hidden;\n\
  191. display: none;\n\
  192. padding-right: 0px;\n\
  193. }\n\
  194. .fsort-time i {\n\
  195. display:inline-block;\
  196. color: #BBB;\
  197. font-style: normal !important;\n\
  198. transform: scale(0.9);\n\
  199. margin-left: 0px;\n\
  200. /* font-size: 12px;*/\n\
  201. }\n\
  202. \n\
  203. /* patches (--min-width:12em!important;) */\n\
  204. table.files td.age {text-align: right !important; padding-right: 4px !important;\n\
  205. width:12em!important;\n\
  206. \n\
  207. max-width:none!important;\n\
  208. overflow:visible!important;\n\
  209. }\n\
  210. table.files td.message {overflow: visible !important;}\n\
  211. /*.file-wrap .include-fragment-error { display: table-row !important;}*/\n\
  212. /* 150315 wide filelist *150426 better not touch this* /\n\
  213. div.wrapper div.container{\n\
  214. min-width: 980px!important;\n\
  215. width:90%!important;}\n\
  216. div.wrapper div#js-repo-pjax-container{\n\
  217. min-width: 790px!important;\n\
  218. width: calc(100% - 200px)!important;\n\
  219. }/* */\n\
  220. \
  221. ');
  222.  
  223. dtStyle=stickStyle('\
  224. td.age span.css-truncate time{\
  225. visibility: hidden !important;\
  226. display: none !important;\
  227. }\
  228. td.age span.css-truncate .fsort-time {\
  229. visibility: visible !important;\
  230. display: inline !important;\
  231. }\
  232. ')
  233. }
  234.  
  235. function setC(n){
  236. for(var i=0,il=C.length; i<il; i++ ){
  237. if(i!=n) C[i].s= 0, C[i].d=d0[i];
  238. else C[i].s=1;
  239. oa[i].className='fsort-butt fsort-'+(C[i].d?'desc':'asc')+(C[i].s?' fsort-sel':'') ;
  240. //oa[i].title=C[i].d? '\u21ca' : '\u21c8';
  241. }
  242. }
  243.  
  244. function dd(s)
  245. { s=s.toString(); if(s.length<2)return'0'+s; return s}
  246. function d2s(n){
  247. var hs=dd(n.getHours())+':'+dd(n.getMinutes());
  248. return {
  249. d: n.getFullYear()+'-'+dd(n.getMonth()+1)+'-'+dd(n.getDate())+'<i>'+ hs+'</i>',
  250. t: hs+':'+dd(n.getSeconds())
  251. }
  252. }
  253.  
  254. var xmatch=/(.*)\.(.*)$/;
  255. function filext(x){
  256. var m= x.match(xmatch);
  257. if(!m || !m[2]) return "EmptyExt";
  258. return m[2].toLowerCase();
  259. }
  260. function setIcon(tr){
  261. var xt,tc,ti=tr.querySelector('td.icon > span.octicon-file-text');
  262. if(!ti) return;
  263. tc=tr.querySelector('td.content > span.css-truncate');
  264. if(!tc) return;
  265. tc=tc.textContent;
  266. if(!tc) return;
  267. xt=filext(tc);
  268. if(!xt) return;
  269. xt=extList[xt];
  270. if(typeof xt === "undefined") return;
  271. ti.className='octicon octicon-'+ extIcon[xt];
  272. //_l('setIcon '+xt);
  273. }
  274.  
  275. function setDateTime(x){
  276. var dt,dtm,dta,dtd,tc,m,now,t;
  277. var DT=D.querySelectorAll('td.age span.css-truncate time');
  278. _l('sDT',x?'refresh':'create');
  279. try{
  280. now = new Date();
  281. for(var dl=DT.length, i=0; i<dl; i++){
  282. dta=DT[i].getAttribute('datetime');
  283. dtd=new Date(dta);
  284. dt= d2s(dtd); // 2014-07-24T17:06:11Z
  285. dtm=null;
  286. if(x){
  287. dtm=DT[i].parentNode.querySelector('.fsort-time');
  288. }
  289. if(!dtm){
  290. dtm=D.createElement('span');
  291. dtm.className='fsort-time';
  292. x=0;
  293. }
  294. if(!x || !dtm.title || dtm.title != DT[i].title)
  295. { dtm.title= DT[i].title;
  296. t= dt.d;
  297. if( (now.getTime() - dtd.getTime() < 12*3600*1000) ||
  298. ((now.getTime() - dtd.getTime() < 24*3600*1000) &&
  299. (now.getDate() == dtd.getDate()) )
  300. ) t=dt.t;
  301. dtm.innerHTML=t;
  302. }
  303. if(!x) insAfter(dtm,DT[i]);
  304. if(!x)
  305. setIcon(outerNode(DT[i],'TR'));
  306. }
  307. /* 150810 */
  308. }catch(e){(console.log(e+'\n*GHSFL* wrong datetime'+x))}
  309. }
  310.  
  311. function isDir(x){
  312. var c= TB.rows[x].cells[0].querySelector("span");
  313. if(c){
  314. if(c.className.indexOf("-directory")>0) return 0;
  315. if(c.className.indexOf("octicon-")>0) return -1;
  316. }
  317. return 1;
  318. }
  319. function getCell(r,c,s,p){
  320. var rc=TB.rows[r].cells[c],q=null;
  321. if(typeof rc == "undefined") {
  322. _l('r:',r,'c:',c,'- ???' );
  323. }else
  324. q=rc.querySelector(s);
  325. if(q) q= p? q.getAttribute(p): q.textContent;
  326. if(q) return q;
  327. return "";
  328. }
  329. var sDir,sCells,sExts;
  330. var fa=[
  331. function(a){
  332. var r=getCell(a,1,'span.css-truncate-target a');
  333. return prefs.upc? r.toUpperCase(): r;
  334. },
  335. function(a){
  336. var r= getCell(a,2,'span.css-truncate');
  337. r=r.replace(/\s+/,' ').replace(/^\s|\s$/,'');
  338. return prefs.upc? r.toUpperCase(): r;
  339. },
  340. function(a){
  341. var c = getCell(a,3,'span.css-truncate>time','datetime');
  342. if(c) return c;
  343. return "2099-12-31T23:59:59Z"
  344. }
  345. ]
  346.  
  347. var b9='\x20\x20\x20'; b9+=b9+b9;
  348. function pad9(s){
  349. if(s.length<9) return (s+b9).substr(0,9);
  350. return s;
  351. }
  352. function sort_p(n){// prepare data for sorting
  353. sDir=[],sCells=[];
  354. for(var tl=TB.rows.length, a=0; a<tl; a++)
  355. sDir.push(isDir(a));
  356. if( n === 0 && prefs.ext ){
  357. for( a=0; a<tl; a++){ // f.x -> x.f
  358. var x=fa[n](a),
  359. m= x.match(/(.*)(\..*)$/);
  360. if(!m || !m[2]) m=['',x,''];
  361. x=pad9(m[2])+' '+m[1];
  362. sCells.push(x);
  363. }
  364. }else{
  365. for( a=0; a<tl; a++) sCells.push(fa[n](a));
  366. }
  367. }
  368.  
  369. function sort_fn(a,b){
  370. var x=sDir[a], y=sDir[b];
  371. if(x!=y) return ((x<y)? 1: -1);
  372. x= sCells[a], y= sCells[b];
  373. return x==y? 0: (((x>y)^ASC)<<1)-1;
  374. }
  375.  
  376. var CNn={content: 0, message: 1, age: 2}
  377.  
  378. function oClr(){
  379. var o= catcher.querySelectorAll('.fsort-butt,.fsort-time')
  380. for(var ol=o.length,i=0;i<ol;i++)
  381. o[i] && o[i].parentNode.removeChild(o[i]);
  382. }
  383. //
  384. function extclassName(){
  385. ext.className='fsort-butt'+ (prefs.ext? ' fsort-on': '' );
  386. }
  387. function clockclassName(){
  388. clock.className='fsort-butt'+ (prefs.dtStyle? '': ' fsort-on');
  389. }
  390. function upcclassName(){
  391. upc.className='fsort-butt'+ (prefs.upc? ' fsort-on': '' );
  392. }
  393. //
  394. function doSort(t){
  395. TB=outerNode(t,'TBODY');
  396. if(!TB){ _l( "*GHSFL* TBODY not found"); return; }
  397. var n = CNn[t.parentNode.className];
  398. if(typeof n=="undefined") n= CNn[t.parentNode.parentNode.className];
  399. if(typeof n=="undefined"){ _l( "*GHSFL* undefined col"); return; }
  400. if(t.id=='fsort-clock'){
  401. dtStyle.disabled = (prefs.dtStyle ^= 1);
  402. savePrefs();
  403. clockclassName();
  404. return;
  405. }
  406. if (t.id=='fsort-ext'){
  407. if(C[n].s) prefs.ext ^= 1;
  408. else prefs.ext= 1;
  409. savePrefs();
  410. extclassName();
  411. C[n].d^=C[n].s; // don't toggle dir on ext.click
  412. }else
  413. if (t.id=='fsort-upc'){
  414. if(C[n].s) prefs.upc ^= 1;
  415. else prefs.upc= 1;
  416. savePrefs();
  417. upcclassName();
  418. C[n].d^=C[n].s; // don't toggle case on upc.click
  419. }
  420. var tb=[],ix=[], i, tl,ti,tx;
  421. _l('n:'+n);
  422. tl=TB.rows.length;
  423. ASC=C[n].d^=C[n].s;
  424. for( i=0; i<tl; i++)
  425. ix.push(i);
  426. oClr();
  427. var ms=new Date();
  428. sort_p(n);
  429. ix.sort(sort_fn);
  430. for( i=0; i<tl; i++)
  431. tb.push(TB.rows[ix[i]]);
  432. for( i=tl-1; i>=0; i--)
  433. TB.removeChild(TB.rows[i]);
  434. for( i=0; i<tl; i++)
  435. TB.appendChild(tb[i]);
  436. ms=(new Date())-ms;
  437. //console.info('sorted by '+ms+'ms');
  438. setC(n);
  439. gitDir(0);
  440. }
  441.  
  442. function onClik(e){doSort(e.target)}
  443.  
  444. function gitDir(x){
  445. if(x && document.querySelector('.fsort-butt')) {
  446. _l('gitDir'+x+ '- already'); return;
  447. }
  448. _l('gitDir',x?'create':'refresh')
  449. var c,o;
  450. ca=[];
  451. c= D.querySelector('.file-wrap table.files td.content >span');
  452. if(!c){ _l( '*GHSFL* no content') ; return; }
  453. ca.push(c);
  454. c=D.querySelector('.file-wrap table.files td.message >span');
  455. if(!c){ _l( '*GHSFL* no messages'); return; }
  456. ca.push(c);
  457. c=D.querySelector('.file-wrap table.files td.age >span');
  458. if(!c){_l( '*GHSFL* no ages'); return; }
  459. ca.push(c);
  460. if(x){ oClr(); oa=[];
  461. o=D.createElement('span');
  462. o.textContent='';
  463. oa.push(o);
  464. o=o.cloneNode(true);
  465. oa.push(o);
  466. o=o.cloneNode(true);
  467. oa.push(o);
  468. clock=D.createElement('span');
  469. clock.id='fsort-clock'; clockclassName();
  470. ext=D.createElement('span');
  471. ext.id='fsort-ext'; extclassName();
  472. upc=D.createElement('span');
  473. upc.id='fsort-upc'; upcclassName();
  474. setDateTime();
  475. setC(-1);
  476. }
  477. o=insBefore(oa[0],ca[0]);
  478. o.appendChild(upc);
  479. o.appendChild(ext);
  480. insBefore(oa[1],ca[1]);
  481. o=insBefore(oa[2],ca[2]);
  482. o.appendChild(clock);
  483. }
  484.  
  485.  
  486. catcher= D.querySelector('#js-repo-pjax-container');
  487. if(!catcher){ _l( "*GHSFL* err0r"); return; }
  488.  
  489. catcher.addEventListener('mousedown',function(e){
  490. if(e.target.nodeName && e.target.nodeName=='SPAN' &&
  491. e.target.className.indexOf('fsort-butt')>-1)
  492. { onClik(e); }
  493. }
  494. ,false);
  495.  
  496. _l('startup()');
  497.  
  498. try {
  499. locStor = W.localStorage;
  500. tt=locStor.getItem("GHSFL");
  501. } catch(e){ locStor =null}
  502.  
  503. if(locStor && tt) try{
  504. var pa =JSON.parse(tt);
  505. for (var a in pa) prefs[a]=pa[a];
  506. _l('prefs:'+JSON.stringify(prefs));
  507. }catch(e){ console.log(e+"\n*GHSFL* bad prefs") }
  508.  
  509. css();
  510. dtStyle.disabled=(prefs.dtStyle===1);
  511.  
  512. gitDir(1);
  513. var target = catcher; //document.body; //D.querSelector('.file-wrap');
  514. var MO = window.MutationObserver;
  515. if(!MO) MO= window.WebKitMutationObserver;
  516. if(!MO) return;
  517. var __started=0;
  518. var mII=0,mI=0;;
  519. var observer = new MO(function(mutations) {
  520. var t=mutations[0].target;
  521. //mI++; _l('mut'+mI+'.'+mutations[0].target.nodeName);
  522. var fc=document.getElementById('fsort-clock')
  523. if(!fc){
  524. gitDir(1);
  525. __started=1;
  526. return; }
  527. if(!fc.parentNode.parentNode.querySelector('.fsort-time')){
  528. setDateTime(1);
  529. return;}
  530. });
  531. // D.body
  532. observer.observe(catcher, { attributes: true, childList: true, subtree: true } );
  533. /* attributes: true , childList: true, subtree: true,
  534. characterData: true, attributeOldValue:true, characterDataOldValue:true
  535. */
  536.  
  537. })()};