GitHub Sortable Filelist

appends sorting function to github directories

目前为 2014-11-20 提交的版本。查看 最新版本

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