GitHub Sortable Filelist

appends sorting function to github directories

目前為 2015-08-07 提交的版本,檢視 最新版本

  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 15.08.07.1
  7. // 15.08.07 + case-insensitive sorting button
  8. // 15.05.07 sorting is now faster
  9. // 14.11.19.13 fixes for latest github changes
  10. // .12 new age format; fix for chrome
  11. // .11 patch for the very first page;
  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. // @updated 2015-03-25
  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. function stickStyle(css){
  47. var s=document.createElement("style"); s.type="text/css";
  48. s.appendChild(document.createTextNode(css));
  49. return (document.head||document.documentElement).appendChild(s);
  50. }
  51. function insBefore(n,e){
  52. return e.parentNode.insertBefore(n,e);
  53. }
  54. function insAfter(n,e){
  55. if(e.nextElementSibling)
  56. return e.parentNode.insertBefore(n,e.nextElementSibling);
  57. return e.parentNode.appendChild(n);
  58. }
  59. function outerNode(target, node) {
  60. if (target.nodeName==node) return target;
  61. if (target.parentNode)
  62. while (target = target.parentNode) try{
  63. if (target.nodeName==node)
  64. return target;
  65. }catch(e){};
  66. return null;
  67. }
  68. function savePrefs(){
  69. if(locStor) locStor.setItem('GHSFL',JSON.stringify(prefs));
  70. }
  71.  
  72. function css(){
  73. stickStyle('\
  74. .fsort-butt,\n\
  75. .tables.file td.content, .tables.file td.message, .tables.file td.age\n\
  76. {position: relative; }\n\
  77. \n\
  78. .fsort-butt:before{\n\
  79. position: absolute; display: inline-block;\n\
  80. cursor: pointer;\n\
  81. text-align:center; vertical-align: top;\n\
  82. width: 18px; height: 14px;\n\
  83. line-height: 14px;\n\
  84. padding:0; margin:0;\n\
  85. border-color: transparent;\n\
  86. border-width: 0;\n\
  87. content: "";\n\
  88. opacity: .2\n\
  89. }\n\
  90. .fsort-butt.fsort-asc:before,.fsort-butt.fsort-desc:before{\n\
  91. left:1.5em; top: -1em;\n\
  92. }\n\
  93. td.age.fsort-butt.fsort-asc:before,td.age.fsort-butt.fsort-desc:before{\n\
  94. right:3em;\n\
  95. }\n\
  96. .fsort-asc:before,.fsort-desc:before{\n\
  97. background-color: #48C;\n\
  98. }\n\
  99. .fsort-asc:before{\n\
  100. border-radius: 24px 24px 8px 8px;\n\
  101. }\n\
  102. .fsort-desc:before{\n\
  103. border-radius: 8px 8px 24px 24px;\n\
  104. }\n\
  105. .fsort-asc:before,\n\
  106. .fsort-desc.fsort-sel:hover:before\n\
  107. {\n\
  108. content: url();\n\
  109. }\n\
  110. .fsort-desc:before,\n\
  111. .fsort-asc.fsort-sel:hover:before{content: url();\n\
  112. }\n\
  113. \n\
  114. .fsort-butt.fsort-sel:before\n\
  115. {\n\
  116. background-color: #4183C4 !important;\n\
  117. opacity:.6 !important;\n\
  118. }\n\
  119. \n\
  120. span.fsort-butt:hover:before\n\
  121. ,span.fsort-butt:hover span:before\n\
  122. { opacity: 1 !important;}\n\
  123. \n\
  124. #fsort-clock:before{\n\
  125. left:5em; top: -15px; \n\
  126. text-align:center; vertical-align: top; top:-15px;\n\
  127. width: 16px; height: 16px;\
  128. border-radius: 16px;\n\
  129. content: url(\n\
  130. );}\n\
  131. .fsort-on:before{ background-color: #4183C4 !important; } \n\\n\
  132. td.age .fsort-butt.fsort-asc:before,td.age .fsort-butt.fsort-desc:before{\n\
  133. left:3em !important;\n\
  134. }\
  135. #fsort-ext:before{\n\
  136. left:4em; top:-14px;\n\
  137. width:28px; height: 14px;\n\
  138. border-radius: 6px;\n\
  139. content:url();\n\
  140. }\n\
  141. #fsort-ext:before{ background-color: #BBB}\n\
  142. /* 150806 uppercase */\n\
  143. #fsort-upc:before{\n\
  144. left:7em; top:-14px;\n\
  145. width:16px; height: 16px;\n\
  146. border-radius: 0 4px 0 4px;\n\
  147. content:url();\n\
  148. }\n\
  149. #fsort-upc:before{ background-color: #BBB}\n\
  150. \n\
  151. table.files td.age .css-truncate.css-truncate-target{\n\
  152. width: 99% !important; \n\
  153. max-width: none !important;\n\
  154. }\n\
  155. /*table.files td.age span.css-truncate time{\n\
  156. position: relative !important;\n\
  157. }*/\n\
  158. .fsort-time {\n\
  159. visibility: hidden;\n\
  160. display: none;\n\
  161. padding-right: 14px;\n\
  162. }\n\
  163. .fsort-time i {\n\
  164. display:inline-block;\
  165. color: #BBB;\
  166. font-style: normal !important;\n\
  167. transform: scale(0.9);\n\
  168. /* font-size: 12px;*/\n\
  169. }\n\
  170. \n\
  171. /* patches (--min-width:12em!important;) */\n\
  172. table.files td.age {text-align: right !important; padding-right: 10px !important;\n\
  173. width:12em!important;\n\
  174. \n\
  175. max-width:none!important;\n\
  176. overflow:visible!important;\n\
  177. }\n\
  178. table.files td.message {overflow: visible !important;}\n\
  179. /*.file-wrap .include-fragment-error { display: table-row !important;}*/\n\
  180. /* 150315 wide filelist *150426 better not touch this* /\n\
  181. div.wrapper div.container{\n\
  182. min-width: 980px!important;\n\
  183. width:90%!important;}\n\
  184. div.wrapper div#js-repo-pjax-container{\n\
  185. min-width: 790px!important;\n\
  186. width: calc(100% - 200px)!important;\n\
  187. }/* */\n\
  188. \
  189. ');
  190.  
  191. dtStyle=stickStyle('\
  192. td.age span.css-truncate time{\
  193. visibility: hidden !important;\
  194. display: none !important;\
  195. }\
  196. td.age span.css-truncate .fsort-time {\
  197. visibility: visible !important;\
  198. display: inline !important;\
  199. }\
  200. ')
  201. }
  202.  
  203. function setC(n){
  204. for(var i=0,il=C.length; i<il; i++ ){
  205. if(i!=n) C[i].s= 0, C[i].d=d0[i];
  206. else C[i].s=1;
  207. oa[i].className='fsort-butt fsort-'+(C[i].d?'desc':'asc')+(C[i].s?' fsort-sel':'') ;
  208. //oa[i].title=C[i].d? '\u21ca' : '\u21c8';
  209. }
  210. }
  211.  
  212. function dd(s)
  213. { s=s.toString(); if(s.length<2)return'0'+s; return s}
  214. function d2s(n){
  215. var hs=dd(n.getHours())+':'+dd(n.getMinutes());
  216. return {
  217. d: n.getFullYear()+'-'+dd(n.getMonth()+1)+'-'+dd(n.getDate())+' <i>'+ hs+'</i>',
  218. t: hs+':'+dd(n.getSeconds())
  219. }
  220. }
  221.  
  222. function setDateTime(x){
  223. var dt,dtm,dta,dtd,tc,m,now,t;
  224. var DT=D.querySelectorAll('td.age span.css-truncate time');
  225. _l('sDT',x?'refresh':'create');
  226. try{
  227. now = new Date();
  228. for(var dl=DT.length, i=0; i<dl; i++){
  229. dta=DT[i].getAttribute('datetime');
  230. dtd=new Date(dta);
  231. dt= d2s(dtd); // 2014-07-24T17:06:11Z
  232. dtm=null;
  233. if(x){
  234. dtm=DT[i].parentNode.querySelector('.fsort-time');
  235. }
  236. if(!dtm){
  237. dtm=D.createElement('span');
  238. dtm.className='fsort-time';
  239. x=0;
  240. }
  241. if(!x || !dtm.title || dtm.title != DT[i].title)
  242. { dtm.title= DT[i].title;
  243. t= dt.d;
  244. if( (now.getTime() - dtd.getTime() < 12*3600*1000) ||
  245. ((now.getTime() - dtd.getTime() < 24*3600*1000) &&
  246. (now.getDate() == dtd.getDate()) )
  247. ) t=dt.t;
  248. dtm.innerHTML=t;
  249. }
  250. if(!x) insAfter(dtm,DT[i]);
  251. }
  252. }catch(e){(console.log(e+'\n*GHSFL* wrong datetime'+x))}
  253. }
  254.  
  255. function isDir(x){
  256. var c= TB.rows[x].cells[0].querySelector("span");
  257. if(c.className.indexOf("-directory")>0) return 0;
  258. if(c.className.indexOf("-file")>0) return -1;
  259. return 1;
  260. }
  261. function getCell(r,c,s,p){
  262. var rc=TB.rows[r].cells[c],q=null;
  263. if(typeof rc == "undefined") {
  264. _l('r:',r,'c:',c,'- ???' );
  265. }else
  266. q=rc.querySelector(s);
  267. if(q) q= p? q.getAttribute(p): q.textContent;
  268. if(q) return q;
  269. return "";
  270. }
  271. var sDir,sCells,sExts;
  272. var fa=[
  273. function(a){
  274. var r=getCell(a,1,'span.css-truncate-target a');
  275. return prefs.upc? r.toUpperCase(): r;
  276. },
  277. function(a){
  278. var r= getCell(a,2,'span.css-truncate');
  279. r=r.replace(/\s+/,' ').replace(/^\s|\s$/,'');
  280. return prefs.upc? r.toUpperCase(): r;
  281. },
  282. function(a){
  283. var c = getCell(a,3,'span.css-truncate>time','datetime');
  284. if(c) return c;
  285. return "2099-12-31T23:59:59Z"
  286. }
  287. ]
  288.  
  289. var b9='\x20\x20\x20'; b9+=b9+b9;
  290. function pad9(s){
  291. if(s.length<9) return (s+b9).substr(0,9);
  292. return s;
  293. }
  294. function sort_p(n){// prepare data for sorting
  295. sDir=[],sCells=[];
  296. for(var tl=TB.rows.length, a=0; a<tl; a++)
  297. sDir.push(isDir(a));
  298. if( n === 0 && prefs.ext ){
  299. for( a=0; a<tl; a++){ // f.x -> x.f
  300. var x=fa[n](a),
  301. m= x.match(/(.*)(\..*)$/);
  302. if(!m || !m[2]) m=['',x,''];
  303. x=pad9(m[2])+' '+m[1];
  304. sCells.push(x);
  305. }
  306. }else{
  307. for( a=0; a<tl; a++) sCells.push(fa[n](a));
  308. }
  309. }
  310.  
  311. function sort_fn(a,b){
  312. var x=sDir[a], y=sDir[b];
  313. if(x!=y) return ((x<y)? 1: -1);
  314. x= sCells[a], y= sCells[b];
  315. return x==y? 0: (((x>y)^ASC)<<1)-1;
  316. }
  317.  
  318. var CNn={content: 0, message: 1, age: 2}
  319.  
  320. function oClr(){
  321. var o= catcher.querySelectorAll('.fsort-butt,.fsort-time')
  322. for(var ol=o.length,i=0;i<ol;i++)
  323. o[i] && o[i].parentNode.removeChild(o[i]);
  324. }
  325. //
  326. function extclassName(){
  327. ext.className='fsort-butt'+ (prefs.ext? ' fsort-on': '' );
  328. }
  329. function clockclassName(){
  330. clock.className='fsort-butt'+ (prefs.dtStyle? '': ' fsort-on');
  331. }
  332. function upcclassName(){
  333. upc.className='fsort-butt'+ (prefs.upc? ' fsort-on': '' );
  334. }
  335. //
  336. function doSort(t){
  337. TB=outerNode(t,'TBODY');
  338. if(!TB){ _l( "*GHSFL* TBODY not found"); return; }
  339. var n = CNn[t.parentNode.className];
  340. if(typeof n=="undefined") n= CNn[t.parentNode.parentNode.className];
  341. if(typeof n=="undefined"){ _l( "*GHSFL* undefined col"); return; }
  342. if(t.id=='fsort-clock'){
  343. dtStyle.disabled = (prefs.dtStyle ^= 1);
  344. savePrefs();
  345. clockclassName();
  346. return;
  347. }
  348. if (t.id=='fsort-ext'){
  349. if(C[n].s) prefs.ext ^= 1;
  350. else prefs.ext= 1;
  351. savePrefs();
  352. extclassName();
  353. C[n].d^=C[n].s; // don't toggle dir on ext.click
  354. }else
  355. if (t.id=='fsort-upc'){
  356. if(C[n].s) prefs.upc ^= 1;
  357. else prefs.upc= 1;
  358. savePrefs();
  359. upcclassName();
  360. C[n].d^=C[n].s; // don't toggle case on upc.click
  361. }
  362. var tb=[],ix=[], i, tl,ti,tx;
  363. _l('n:'+n);
  364. tl=TB.rows.length;
  365. ASC=C[n].d^=C[n].s;
  366. for( i=0; i<tl; i++)
  367. ix.push(i);
  368. oClr();
  369. sort_p(n);
  370. ix.sort(sort_fn);
  371. for( i=0; i<tl; i++)
  372. tb.push(TB.rows[ix[i]]);
  373. for( i=tl-1; i>=0; i--)
  374. TB.removeChild(TB.rows[i]);
  375. for( i=0; i<tl; i++)
  376. TB.appendChild(tb[i]);
  377. setC(n);
  378. gitDir1(0);
  379. }
  380.  
  381. function onClik(e){doSort(e.target)}
  382.  
  383. function gitDir1(x){
  384. if(x && document.querySelector('.fsort-butt')) {
  385. _l('gitDir'+x+ '- already'); return;
  386. }
  387. _l('gitDir',x?'create':'refresh')
  388. var c,o;
  389. ca=[];
  390. c= D.querySelector('.file-wrap table.files td.content >span');
  391. if(!c){ _l( '*GHSFL* no content') ; return; }
  392. ca.push(c);
  393. c=D.querySelector('.file-wrap table.files td.message >span');
  394. if(!c){ _l( '*GHSFL* no messages'); return; }
  395. ca.push(c);
  396. c=D.querySelector('.file-wrap table.files td.age >span');
  397. if(!c){_l( '*GHSFL* no ages'); return; }
  398. ca.push(c);
  399. if(x){ oClr(); oa=[];
  400. o=D.createElement('span');
  401. o.textContent='';
  402. oa.push(o);
  403. o=o.cloneNode(true);
  404. oa.push(o);
  405. o=o.cloneNode(true);
  406. oa.push(o);
  407. clock=D.createElement('span');
  408. clock.id='fsort-clock'; clockclassName();
  409. ext=D.createElement('span');
  410. ext.id='fsort-ext'; extclassName();
  411. upc=D.createElement('span');
  412. upc.id='fsort-upc'; upcclassName();
  413. setDateTime();
  414. setC(-1);
  415. }
  416. o=insBefore(oa[0],ca[0]);
  417. o.appendChild(upc);
  418. o.appendChild(ext);
  419. insBefore(oa[1],ca[1]);
  420. o=insBefore(oa[2],ca[2]);
  421. o.appendChild(clock);
  422. }
  423.  
  424. function gitDir(){
  425. gitDir1(1);
  426. }
  427.  
  428. catcher= D.querySelector('#js-repo-pjax-container');
  429. if(!catcher){ _l( "*GHSFL* err0r"); return; }
  430.  
  431. catcher.addEventListener('mousedown',function(e){
  432. if(e.target.nodeName && e.target.nodeName=='SPAN' &&
  433. e.target.className.indexOf('fsort-butt')>-1)
  434. { onClik(e); }
  435. }
  436. ,false);
  437.  
  438. _l('startup()');
  439.  
  440. try {
  441. locStor = W.localStorage;
  442. tt=locStor.getItem("GHSFL");
  443. } catch(e){ locStor =null}
  444.  
  445. if(locStor && tt) try{
  446. var pa =JSON.parse(tt);
  447. for (var a in pa) prefs[a]=pa[a];
  448. _l('prefs:'+JSON.stringify(prefs));
  449. }catch(e){ console.log(e+"\n*GHSFL* bad prefs") }
  450.  
  451. css();
  452. dtStyle.disabled=(prefs.dtStyle===1);
  453.  
  454. gitDir();
  455. var target = catcher; //document.body; //D.querSelector('.file-wrap');
  456. var MO = window.MutationObserver;
  457. if(!MO) MO= window.WebKitMutationObserver;
  458. if(!MO) return;
  459. var __started=0;
  460. var mutI=0;
  461. var observer = new MO(function(mutations) {
  462.  
  463. for(var m,t, ml=mutations.length, i=0; i<ml; i++)
  464. {
  465. m=mutations[i],t = m.target;
  466. if( m.type=="attributes")
  467. {
  468. if( t.nodeName == 'DIV' &&
  469. t.className == "file-wrap"
  470. ){
  471. gitDir();
  472. return;
  473. }
  474. // patch for the very first page
  475. if( 0===__started && t.nodeName=='TIME' )
  476. {
  477. if( t.parentNode.parentNode.className=="age" )
  478. {
  479. //_l('T'+mutI++,ml,' T:' +m.type,'N:'+t.nodeName, 'C:',"age" ) ;
  480. if(!catcher.querySelector('.fsort-butt'))
  481. gitDir(1); //chrome ?!11
  482. setDateTime(1);
  483. __started=1;
  484. return;
  485. }
  486. else continue;
  487. }
  488. }
  489. if( m.type=="childList" )
  490. {
  491. if( t.className=='age' )
  492. {
  493. if(!catcher.querySelector('.fsort-butt'))
  494. gitDir(1); //chrome ?!11
  495. //_l('C'+mutI++,ml,' T:' +m.type,'N:'+t.nodeName,'C:'+ t.className,t.querySelector('TIME').textContent) ;
  496. setDateTime(1);
  497. return;
  498. }
  499. else continue;
  500. }
  501. }
  502. });
  503.  
  504. observer.observe(D.body, { attributes: true, childList: true, subtree: true } );
  505. /* attributes: true , childList: true, subtree: true,
  506. characterData: true, attributeOldValue:true, characterDataOldValue:true
  507. */
  508.  
  509. })()};