Skip to content

Instantly share code, notes, and snippets.

@artzub
Last active December 10, 2015 23:28
Show Gist options
  • Save artzub/4509035 to your computer and use it in GitHub Desktop.
Save artzub/4509035 to your computer and use it in GitHub Desktop.
Work with github api.
.idea/
*.iml
*.iws
/**
* User: ArtZub
* Date: 15.01.13
* Time: 2:13
*/
'use strict';
(function(window) {
var lastTime = 0;
var vendors = ['ms', 'moz', 'webkit', 'o'];
for(var x = 0; x < vendors.length && !window.requestAnimationFrame; ++x) {
window.requestAnimationFrame = window[vendors[x]+'RequestAnimationFrame'];
window.cancelAnimationFrame =
window[vendors[x]+'CancelAnimationFrame'] || window[vendors[x]+'CancelRequestAnimationFrame'];
}
if (!window.requestAnimationFrame)
window.requestAnimationFrame = function(callback, element) {
var currTime = new Date().getTime();
var timeToCall = Math.max(0, 16 - (currTime - lastTime));
var id = window.setTimeout(function() { callback(currTime + timeToCall); },
timeToCall);
lastTime = currTime + timeToCall;
return id;
};
if (!window.cancelAnimationFrame)
window.cancelAnimationFrame = function(id) {
clearTimeout(id);
};
})(window);
var ONE_SECOND = 1000,
ONE_MINUTE = 60 * ONE_SECOND,
ONE_HOUR = 60 * ONE_MINUTE,
ONE_DAY = 24 * ONE_HOUR,
PI_CIRCLE = 2 * Math.PI;
var ghcs = {
users: {},
states: {cur:0, max:0},
limits: {
commits : 100,
stepShow : 1,
stepType : ONE_DAY
},
settings : {
access : {
client_id : "c45417c5d6249959a91d",
client_secret : "4634b3aa7549c3d6306961e819e5ec9b355a6548"
},
cs : {
fileLife : 255 // number of steps of life a file
, userLife : 255 // number of steps of life a user
, edgeLife : 255 // number of steps of life a edge
, showCountExt : true // show table of file's extension
, onlyShownExt : true // show only extension which is shown
, showHistogram : true // displaying histogram of changed files
, showHalo : true // show a file's halo
, padding : 25 // padding around a user
, rateOpacity : .5 // rate of decrease of opacity
, rateFlash : 2.5 // rate of decrease of flash
, sizeFile : 2 // size of file
, sizeUser : 24 // size of user
, showPaddingCircle : false // show circle of padding
, useAvatar : true // show user's avatar
, showEdge : true // show a edge
, showFile : true // show a file
, showUser : true // show a user
, showLabel : true // show user name
, showFilename : true // show file name TODO: надо-ли?
, labelPattern : "%n <%e>" // pattern for label of user
, showCommitMessage : false // show commit message
, skipEmptyDate : true // skip empty date
}
},
asyncForEach: function(items, fn, time) {
if (!(items instanceof Array))
return;
var workArr = items.reverse().concat();
function loop() {
if (workArr.length > 0)
fn(workArr.shift(), workArr);
if (workArr.length > 0)
setTimeout(loop, time || 1);
}
loop();
},
imageHash : d3.map({})
};
(function(ghcs) {
ghcs.storage = sessionStorage;
if (ghcs.storage) {
Storage.prototype.set = function(key, value) {
this.setItem(key, JSON.stringify(value));
};
Storage.prototype.get = function(key) {
var res = this.getItem(key);
return res ? JSON.parse(res) : res;
};
Storage.prototype.setImageData = setImageData;
}
else {
ghcs.storage = d3.map({});
ghcs.storage.clear = function() {
var ks = ghcs.storage.keys,
l = ks.length;
while(--l > -1)
ghcs.storage.remove(ks[l]);
};
ghcs.storage.setImageData = setImageData;
}
var canvas = document.createElement("canvas"),
ctx = canvas.getContext("2d");
/**
* @param url {String}
* @param image {Image}
*/
function setImageData(url, image) {
canvas.width = image.width;
canvas.height = image.height;
ctx.drawImage(image, 0, 0);
ghcs.storage.set(url, canvas.toDataURL("image/png"));
canvas.width = 0;
canvas.height = 0;
}
})(ghcs);
/**
* User: ArtZub
* Date: 15.01.13
* Time: 1:36
*/
'use strict';
var TYPE_REQUEST = {
repos : 1,
commits : 2
}
, TYPE_STATUS_FILE = {
removed : 0,
modified : 1,
added : 2,
renamed : 3
}
;
function makeUrl(url, type, limit) {
limit = limit || 0;
var sec = "client_id=" + ghcs.settings.access.client_id + "&client_secret=" + ghcs.settings.access.client_secret;
if (type == TYPE_REQUEST.repos) {
sec += (ghcs.rot ? "&per_page=100&type=" + ghcs.rot : "" );
}
else if(type == TYPE_REQUEST.commits) {
limit = limit < 0 ? ghcs.limits.commits : limit;
sec += "&per_page=" + (limit > 100 ? 100 : limit)
}
return url ? (url + (url.indexOf('?') === -1 ? '?' : '&') + sec) : url;
}
function crossUrl(url, rt) {
return url;
/*var result = url, arr = /(.*)?\?(.*)/.exec(url);
if (arr) {
result = "http://artzub.com/cross/?"
+ ( rt ? "rt=" + rt + "&" : "" )
+ "u=" + arr[1];
if (arr.length > 2)
result += "&d=" + encodeURIComponent(arr[2]);
}
return result;*/
}
function randTrue() {
return Math.round((Math.random() * 2) % 2);
}
function getDataFromRequest(req) {
return req && req.meta && req.meta.status == 200 && req.data ? req.data : (log(req) && null);
}
function parseCommit(org_commit, commit){
if (!commit || !org_commit || commit.sha != org_commit.sha)
return;
var s = commit.stats = {
f : {
m : 0,
a : 0,
d : 0
},
changes : 0,
additions : 0,
deletions : 0
};
commit.files = org_commit.files.map(function(f) {
if (TYPE_STATUS_FILE[f.status] == undefined)
console.log(f.status);
f.status = TYPE_STATUS_FILE[f.status];
if (f.changes > 0) {
s.changes += f.changes;
s.additions += f.additions;
s.deletions += f.deletions;
}
else if(f.status) {
commit.stats.changes++;
commit.stats.additions++;
}
else if(!f.status) {
commit.stats.changes -= commit.stats.changes ? 1 : 0;
commit.stats.additions -= commit.stats.additions ? 1 : 0;
}
(f.status == TYPE_STATUS_FILE.modified || f.status == TYPE_STATUS_FILE.renamed) && s.f.m++;
f.status == TYPE_STATUS_FILE.added && s.f.a++;
!f.status && s.f.d++;
return {
name : f.filename.toLowerCase(),
changes : f.changes || 0,
additions : f.additions || 0,
deletions : f.deletions || 0,
status : f.status
}
});
ghcs.repo.stats = ghcs.repo.stats || {};
ghcs.repo.stats.changes = d3.max([ghcs.repo.stats.changes || 0, commit.stats.deletions, commit.stats.additions]);
ghcs.repo.stats.files = d3.max([ghcs.repo.stats.files || 0, s.f.a + s.f.m, s.f.d]);
}
function upCommits() {
redrawStats();
updateStatus(ghcs.states.cur++);
ldrTop.show();
psBar.show();
checkCompleted();
}
function preloadImage(url) {
var ava, image;
image = ghcs.imageHash.get(url);
if (!image) {
image = new Image();
//ava = ghcs.storage.get(url);
//if (!ava) {
image.onerror = function () {
return console.log(this);
};
/*image.onload = (function (url) {
return function () {
if (url)
ghcs.storage.setImageData(url, this);
};
})(url);*/
image.src = crossUrl((url || (url = "https://secure.gravatar.com/avatar/" + Date.now() + Date.now() + "?d=identicon&f=y")) + "&s=48", "image");
/*}
else {
image.src = ava;
}*/
ghcs.imageHash.set(url, image);
}
return image;
}
function parseCommits(commits) {
ghcs.repo.commits = ghcs.repo.commits || d3.map({});
if (commits && commits.length) {
updateStatus(ghcs.states.cur);
psBar.show();
ldrTop.show();
commits.forEach(function(d, i) {
var obj = ghcs.repo.commits.get(d.sha);
if (!obj) {
obj = {
url : d.url,
sha : d.sha,
author : {
name : d.commit.author.name,
email : d.commit.author.email,
login : d.author && d.author.login ? d.author.login : d.commit.author.email
},
committer : {
name : d.commit.committer.name,
email : d.commit.committer.email,
login : d.committer && d.committer.login ? d.committer.login : d.commit.committer.email
},
date : Date.parse(d.commit.author.date),
avatar_url : (d.author && d.author.avatar_url ? d.author.avatar_url : null),
message : d.commit.message,
parents : d.parents
};
ghcs.repo.commits.set(obj.sha, obj);
obj.author.avatar = preloadImage(obj.avatar_url);
}
if (!obj.files) {
JSONP(makeUrl(obj.url), (function(c) {
ghcs.repo.dates.push(c.date);
ghcs.repo.dates.sort(d3.ascending);
return function(req) {
parseCommit(getDataFromRequest(req), ghcs.repo.commits.get(c.sha));
upCommits();
};
})(obj), {
onerror : function(err) {
console.log(err);
upCommits();
}
});
}
else {
upCommits();
}
});
}
else {
upCommits();
}
}
function clearUserRepos() {
if (ghcs.users.hasOwnProperty(ghcs.login) && ghcs.users[ghcs.login].hasOwnProperty("repos"))
delete ghcs.users[ghcs.login]["repos"]
}
function parseRepos(data) {
if (data) {
ghcs.users[ghcs.login] = ghcs.users[ghcs.login] || {};
ghcs.users[ghcs.login].repos = (ghcs.users[ghcs.login].repos || []).concat(
data.filter(function (d) {
return !d.private && !d.hasOwnProperty("nodeValue");
}).map(function (d) {
return {
x : (Math.random() * w) || 1,
y : (Math.random() * h) || 1,
nodeValue : {
id: d.id,
name: d.name,
url: d.url,
html_url: d.html_url,
commits_url : d.commits_url.replace(/{.*$/, ""),
size : d.size,
date : Date.parse(d.pushed_at || d.updated_at),
cdate : Date.parse(d.created_at),
desc : d.description,
lang : d.language || "Multi"
}
};
})
);
ghcs.states.cur = ghcs.users[ghcs.login].repos.length;
}
updateStatus(ghcs.states.cur, "loading ...");
ldrTop.show();
psBar.show();
checkCompleted();
redrawRepos();
}
function chSelect(e) {
if (e) {
ghcs.repo = e.nodeValue;
ghcs.repo.dates = [];
ghcs.repo.changes = [];
runBtn.enable();
curRep.setName(e);
}
else {
ghcs.repo = null;
runBtn.disable();
curRep.setName(null);
stepsBar.secondStep();
}
}
function chUser() {
if (ghcs.chUserTimer) {
clearTimeout(ghcs.chUserTimer);
delete ghcs.chUserTimer;
}
ghcs.chUserTimer = setTimeout((function (login) {
return function () {
if (login) {
stepsBar.firstStep();
curRep.setName(null);
vis.clearRepos();
userTxt.disable();
showBtn.disable();
runBtn.disable();
ghcs.login = login;
ghcs.repo = null;
redrawStats();
ghcs.states.complete = function() {
stepsBar.secondStep();
ldrTop.hide();
setTimeout(nextStepApplyParams, 500);
};
cbDlr.check();
if (!ghcs.users.hasOwnProperty(login) || !ghcs.users[login].hasOwnProperty("repos")) {
ldrTop.show();
JSONP(makeUrl("https://api.github.com/users/" + login), function (req) {
var data = getDataFromRequest(req);
if (!data) {
parseRepos(null);
return;
}
var u = ghcs.users[data.login] = {info: data};
u.info.avatar = new Image();//preloadeImage(u.info.avatar_url);
u.info.avatar.src = u.info.avatar_url;
ghcs.login = data.login;
ghcs.states.max = +u.info.public_repos;
ghcs.states.cur = 0;
updateStatus(ghcs.states.cur, "loading ...");
psBar.show();
if (data.repos_url)
JSONP(makeUrl(data.repos_url, TYPE_REQUEST.repos), function getAll(req) {
parseRepos(getDataFromRequest(req));
getNext(req, function(next) {
if (next) {
ldrTop.show();
JSONP(next, getAll);
}
});
divStat.updateInfo();
}, {
onerror : function(err) {
console.log(err);
}
});
else
parseRepos(null);
divStat.updateInfo();
});
}
else {
ghcs.states.max = ghcs.users[login].repos ? ghcs.users[login].repos.length : 0;
parseRepos(ghcs.users[login].repos);
}
divStat.updateInfo();
userTxt.enable();
}
}
})(userTxt.property("value")), 300);
}
function getNext(req, fn) {
var next;
if (req && req.meta && req.meta.Link) {
next = req.meta.Link.reduce(function (a, b) {
if (!a && b[1].rel == "next")
return b[0];
return a;
}, null);
}
fn && fn(next);
}
function analyseCommits() {
runBtn.disable();
ldrTop.show();
ghcs.states.max = ghcs.limits.commits;
ghcs.states.cur = 0;
ghcs.states.loaded = 0;
ghcs.states.complete = function() {
stepsBar.thirdStep();
runBtn.enable();
ldrTop.hide();
visBtn.enable();
if (ghcs.repo && ghcs.repo.commits)
ghcs.repo.commitsCount = ghcs.repo.commits.values().filter(function(d) {
return !!d.files;
}).length;
vis.redrawStat(ghcs.repo);
setTimeout(nextStepApplyParams, 500);
};
vis.layers.stat.toFront();
cbDlsr.check();
if (!ghcs.repo || !ghcs.repo.commits_url || ghcs.repo.loadedAll) {
updateStatus(ghcs.states.cur = ghcs.states.max);
checkCompleted();
return;
}
JSONP(makeUrl(ghcs.repo.commits_url, TYPE_REQUEST.commits, ghcs.limits.commits), function getAll(req) {
getNext(req, function(next) {
var l = req && req.data && req.data instanceof Array ? req.data.length : 0;
ghcs.states.loaded += l;
l = ghcs.states.max - ghcs.states.loaded;
if (next && l > 0) {
updateStatus(ghcs.states.cur);
psBar.show();
ldrTop.show();
JSONP(next.replace("per_page=100", "per_page=" + (l > 100 ? 100 : l)),
getAll, {
onerror : (function(len) {
return function() {
ghcs.states.max -= len;
};
})(l)
});
}
else {
if (!next)
ghcs.repo.loadedAll = true;
ghcs.states.max =
ghcs.states.loaded;
}
});
parseCommits(getDataFromRequest(req));
});
}
Display the source blob
Display the rendered blob
Raw
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
<!DOCTYPE html>
<html xmlns="http://www.w3.org/1999/html">
<head>
<meta charset="utf-8">
<link rel="stylesheet" href="style.css">
<script src="http://d3js.org/d3.v3.js"></script>
<script src="jsonp.js"></script>
<script src="ghcs.js"></script>
<script src="main.js"></script>
<script src="github.js"></script>
<script src="vis.js"></script>
<script src="stat.js"></script>
<script src="repo.js"></script>
<script src="langhg.js"></script>
<script src="show.js"></script>
</head>
<body onload="init()">
<header>
<div id="controls" class="popup">
<div class="row">
<ul class="steps sfirst">
<li style="z-index: 2" class="first">
<label><strong>1.</strong></label>
<input placeholder="User" type="text" id="user">
<button id="showBtn" title="Press for loading data about user repositories." class="btn" disabled="disabled">Show</button>
<div class="sub">
<div class="row">
<blockquote>Enter user's login of github.</blockquote>
<div class="row">
<ul class="setting">
<li class="field"><h1>Display:</h1>
<ul class="group">
<li class="field">
<input id="cb-dllh" type="checkbox" checked="checked">
<label for="cb-dllh">Layer histogram languages</label>
</li>
</ul>
</li>
</ul>
</div>
</div>
</div>
</li>
<li style="z-index: 1" class="second">
<label><strong>2.</strong></label>
<label id="curRep"><span>Select Repo...</span></label>
<button id="runBtn" title="Press for analyse repository." class="btn" disabled="disabled">Analyse</button>
<div class="sub">
<div class="row">
<blockquote>Click to repo in graph of repos.</blockquote>
<div class="row">
<ul class="setting">
<li class="field"><h1>Display:</h1>
<ul class="group">
<li class="field">
<input id="cb-dlr" type="checkbox" checked="checked">
<label for="cb-dlr">Layer repos</label>
</li>
<li class="field">
<input id="cb-dlsr" type="checkbox" checked="checked">
<label for="cb-dlsr">Layer stat repo</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input id="cb-dlcmrs" type="checkbox" checked="checked">
<label for="cb-dlcmrs">Layer committers</label>
</li>
</ul>
</li>
<li class="field"><h1>Settings:</h1>
<ul class="group">
<li class="field">
<label for="txt-lc">Limit of commits</label>
<input id="txt-lc" type="number" value="100" max="10000000" min="0" step="1">
</li>
</ul>
</li>
</ul>
</div>
</div>
</div>
</li>
<li class="third">
<label><strong>3.</strong></label>
<button id="visBtn" title="Press for dynamic visualization." class="btn" disabled="disabled"><span class="mono">►</span> Run</button>
<button id="visBtnPause" title="Pause" class="btn" style="display: none" disabled="disabled"><span class="mono">▌▌</span>Pause</button>
<button id="visBtnStop" title="Stop" class="btn" style="display: none" disabled="disabled"><span class="mono">■</span> Stop</button>
<button id="visBtnResume" title="Restart" class="btn" style="display: none" disabled="disabled"><span class="mini-icon mini-icon-history"></span> Restart</button>
<div class="sub">
<div class="row">
<blockquote>Select type visualization, change setting and press button run.</blockquote>
<div class="row">
<ul class="setting">
<li class="field"><h1>Display:</h1>
<ul class="group">
<li class="field">
<input id="cb-dlvml" type="checkbox" checked="checked">
<label for="cb-dlvml">Layer canvas</label>
</li>
</ul>
</li>
<li class="field"><h1>Type visualization:</h1>
<ul class="group">
<li class="field">
<input name="typeVis" id="rb_lacs" checked="checked" type="radio">
<label for="rb_lacs">Code_swarm</label>
</li>
<li class="field">
<input name="typeVis" id="rb_lag" type="radio">
<label for="rb_lag">Gource</label>
</li>
</ul>
</li>
<li class="field"><h1>Settings:</h1>
<ul class="setting" id="st-cs">
<li class="field"><h1>Code_swarm:</h1>
<ul class="setting">
<li class="field"><h1>The number of cycles of life element:</h1>
<ul class="group">
<li class="field">
<label for="n-cs-userLife">User</label>
<input data-ns="cs" id="n-cs-userLife" type="number" step="1" min="0" max="1000000">
</li>
<li class="field">
<input data-ns="cs" id="cb-cs-showUser" type="checkbox" checked="checked">
<label for="cb-cs-showUser">visible</label>
</li>
</ul>
<ul class="group">
<li class="field">
<label for="n-cs-fileLife">&nbsp;&nbsp;File</label>
<input data-ns="cs" id="n-cs-fileLife" type="number" step="1" min="0" max="1000000">
</li>
<li class="field">
<input data-ns="cs" id="cb-cs-showFile" type="checkbox" checked="checked">
<label for="cb-cs-showFile">visible</label>
</li>
</ul>
<ul class="group">
<li class="field">
<label for="n-cs-edgeLife">Edge</label>
<input data-ns="cs" id="n-cs-edgeLife" type="number" step="1" min="0" max="1000000">
</li>
<li class="field">
<input data-ns="cs" id="cb-cs-showEdge" type="checkbox">
<label for="cb-cs-showEdge">visible</label>
</li>
</ul>
</li>
</ul>
<ul class="setting">
<li class="field"><h1>Display:</h1>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-showCountExt" type="checkbox" checked="checked">
<label for="cb-cs-showCountExt">Legend extension</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-showHistogram" type="checkbox" checked="checked">
<label for="cb-cs-showHistogram">Changes Histogram</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-showCommitMessage" type="checkbox" checked="checked">
<label for="cb-cs-showCommitMessage">Commit Messages</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-onlyShownExt" type="checkbox">
<label for="cb-cs-onlyShownExt">Only current commit extensions</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-showLabel" type="checkbox" checked="checked">
<label for="cb-cs-showLabel">User's name</label>
</li>
<li class="field">
<input data-ns="cs" id="cb-cs-showHalo" type="checkbox" checked="checked">
<label for="cb-cs-showHalo">Halo</label>
</li>
</ul>
<ul class="group">
<!--<li class="field">
<input id="cb-cs-showFilename" type="checkbox" checked="checked">
<label for="cb-cs-showFilename">Show file's name</label>
</li>-->
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-showPaddingCircle" type="checkbox">
<label for="cb-cs-showPaddingCircle">Circle of padding</label>
</li>
</ul>
</li>
</ul>
<ul class="setting">
<li class="field"><h1>Rate of decrease of:</h1>
<ul class="group">
<li class="field">
<label for="n-cs-rateFlash">Flash</label>
<input data-ns="cs" id="n-cs-rateFlash" type="number" step="0.5" min="0" max="100">
</li>
</ul>
<ul class="group">
<li class="field">
<label for="n-cs-rateOpacity">Opacity</label>
<input data-ns="cs" id="n-cs-rateOpacity" type="number" step="0.5" min="0" max="100">
</li>
</ul>
</li>
</ul>
<ul class="setting">
<li class="field"><h1>Size of:</h1>
<ul class="group">
<li class="field">
<label for="n-cs-sizeUser">User</label>
<input data-ns="cs" id="n-cs-sizeUser" type="number" step="0.1" min="0.1" max="100">
</li>
</ul>
<ul class="group">
<li class="field">
<label for="n-cs-sizeFile">File (initial)</label>
<input data-ns="cs" id="n-cs-sizeFile" type="number" step="0.1" min="0.1" max="100">
</li>
</ul>
</li>
</ul>
<ul class="group">
<li class="field">
<label for="n-cs-padding">Padding around a user</label>
<input data-ns="cs" id="n-cs-padding" type="number" step="0.1" min="0" max="100">
</li>
</ul>
<ul class="group">
<li class="field">
<label for="txt-cs-labelPattern">Pattern for label of user</label>
<input data-ns="cs" id="txt-cs-labelPattern" type="text" value="%n <%e>">
<blockquote>%n - name; %e - email; %l - login</blockquote>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-useAvatar" type="checkbox" checked="checked">
<label for="cb-cs-useAvatar">Use user's avatar</label>
</li>
</ul>
<ul class="group">
<li class="field">
<input data-ns="cs" id="cb-cs-skipEmptyDate" type="checkbox" checked="checked">
<label for="cb-cs-skipEmptyDate">Skip empty date</label>
</li>
</ul>
</li>
</ul>
</li>
</ul>
</div>
</div>
</div>
</li>
</ul>
</div>
</div>
<div id="userInfo" class="popup tooltip">
<div class="row" id="divStat"></div>
</div>
</header>
<!--https://github.com/images/spinners/octocat-spinner-128.gif-->
<!---->
<div id="ldrCont" class="popup" style="display: none"><img id="ldrTop" src="octocat-spinner-128.gif"></div>
<div id="about" class="popup" style="display: none;">
<div class="row">
<h1>The visualization the history of repositories from the Github</h1>
</div>
<div class="row">
</div>
<div class="row">
<hr>
Create by <a href="http://artzub.com">Artem Zubkov</a> (<a href="mailto:artzub@gmail.com" title="Write me...">artzub@gmail.com</a>)
<br />
Powered by <a href="http://d3js.org" title="Perfect javascript framework">D3.js</a>
</div>
</div>
<div id="tooltip" class="popup tooltip"></div>
<div id="layers">
<div id="canvas"></div>
<div id="svg"></div>
</div>
<footer>
<div id="statusBar" class="popup">
<ul>
<li class="progressBar">
<div class="meter orange" style="display: none">
<span id="progressBar" style="width: 0%"></span>
</div>
</li>
</ul>
</div>
<div id="console" class="popup"></div>
</footer>
</body>
</html>
/**
* repo https://github.com/artzub/JSONP
*/
(function(window) {
if (!window || !window.document)
return;
var seq = 0, // sequent
document = window.document;
/**
* JSONP
* @param {String} uri The URL you are requesting with the JSON data (you may include URL params).
* @param {Function} callback The callback function which you want to execute for JSON data (JSON response is first argument).
* @param {Object} params The params contains data about callback param's name, onload function and onerror function.
* Params have next structure:
* params = {
* callbackParam : '', default is callback
* onerror_callback : function() {},
* onload_callback : function() {},
* script_order : 'defer' | 'async' (is default)
*}
*/
window.JSONP = function(uri, callback, params) {
if (!arguments.length || arguments.length < 2)
return;
uri = uri || '';
callback = callback || function() {};
params = params || {};
params.callbackParam = params.callbackParam || 'callback'
uri += uri.indexOf('?') === -1 ? '?' : '&';
function clear() {
try {
delete window[id];
} catch(e) {
window[id] = null;
}
document.documentElement.removeChild(script);
}
function response() {
clear();
callback.apply(this, arguments);
}
function doError() {
clear();
params.onerror && params.onerror.apply(this, arguments);
}
function doLoad() {
params.onload && params.onload.apply(this, arguments);
}
var id = '_JSONP_' + seq++,
script = document.createElement('script');
window[id] = response;
params.script_order = params.script_order || 'async';
script.onload = doLoad;
script.onerror = doError;
script.setAttribute(params.script_order, params.script_order);
script.setAttribute('src', uri + params.callbackParam + '=' + id);
document.documentElement.insertBefore(
script,
document.documentElement.lastChild
);
}
})(window);
/**
* Created with IntelliJ IDEA.
* User: ArtZub
* Date: 23.01.13
* Time: 13:32
* To change this template use File | Settings | File Templates.
*/
'use strict';
(function(vis) {
vis.clearLangHg = function() {
vis.layers && vis.layers.repo && vis.layers.repo.langHg && vis.layers.repo.langHg.selectAll("*")
.transition()
.duration(750)
.ease("elastic")
.remove();
};
vis.redrawLangHg = function(data, layout) {
layout = layout || vis.layers.repo;
if (!data) {
vis.clearLangHg();
return;
}
data = data.sort(function(a, b) {
return d3.ascending(a.key, b.key);
});
var w_hg = w / 4,
h_hg = h / 6,
m = {left : 10, top : 10, right : 10, bottom : 10},
pos = {top : h - h_hg - m.bottom - m.top - margin.bottom, left : margin.left};
var x = d3.scale.ordinal()
.rangeRoundBands([0, 28 * data.length], .2)
.domain(data.map(function(d) { return d.key; }));
var xc = x.rangeBand() / 2;
var y = d3.scale.linear()
.domain([0, d3.max(data, function(d) { return d.values.length; })])
.range([h_hg, 0]);
if (layout.langHg && layout.langHg.empty())
layout.langHg = null;
layout.langHg = (layout.langHg || layout.insert("g", ":first-child"))
.attr("class", "langHg")
.attr("width", w_hg + m.left + m.right)
.attr("height", h_hg + m.top + m.bottom)
.attr("transform", "translate(" + pos.left + "," + pos.top + ")");
function mei(d) {
d._g
&& d._g.selectAll("circle")
.style("stroke", d3.rgb(colors.decolor).darker())
.style("fill", toRgba(colors.decolor, vis.forceRep.opt(vis.forceRep.radO(d))))
&& d._g.selectAll("text")
.style("fill", d3.rgb(colors.decolor).darker());
}
function moi(d) {
d._g
&& d._g.selectAll("circle")
.style("stroke", d3.rgb(vis.forceRep.colors(d.nodeValue.lang)))
.style("fill", toRgba(vis.forceRep.colors(d.nodeValue.lang), vis.forceRep.opt(vis.forceRep.radO(d))))
&& d._g.selectAll("text")
.style("fill", d3.rgb(vis.forceRep.colors(d.nodeValue.lang)).brighter());
}
function me(d) {
vis.forceRep.nodes().filter(function(k) {
return k.nodeValue.lang != d.key;
}).forEach(mei);
}
function mo(d) {
vis.forceRep.nodes().filter(function(k) {
return k.nodeValue.lang != d.key;
}).forEach(moi);
}
function appendItems(g) {
g.append("path")
.attr("class", "hLine")
.style("stroke", "rgba(255, 255, 255, .3)");
g.append("text")
.attr("class", "tLang")
.style("fill", function(d) { return d3.rgb(vis.forceRep.colors(d.key)); })
.attr("dy", ".33em")
.attr("dx", "-6px")
.attr("transform", "rotate(90)")
.style("text-anchor", "end")
.text(function(d) { return d.key; })
.each(function() {
var pr = d3.select(this.parentNode);
pr.insert("rect", ":first-child")
.attr("class", "barSelect")
.attr("fill", "rgba(244, 244, 244, .2)");
});
var gg = g.append("g")
.attr("class", "barChain");
gg.append("path")
.attr("class", "vLine")
.style("stroke", function(d) { return d3.rgb(vis.forceRep.colors(d.key)).darker(); })
.attr("d", "M0,0 L0,0");
gg.append("circle")
.style("fill", function(d) { return d3.rgb(vis.forceRep.colors(d.key)).darker(); })
.attr("r", 2);
var dg = gg.append("g")
.attr("class", "dCircle")
.attr("transform", "translate(" + [0 , 0] + ")");
dg.append("circle")
.style("fill", function(d) { return d3.rgb(vis.forceRep.colors(d.key)); })
.style("stoke", function(d) { return d3.rgb(vis.forceRep.colors(d.key)).darker(); })
;
dg.append("text")
.attr("text-anchor", "middle")
.attr("dy", ".32em")
.style("fill", function(d) { return d3.rgb(vis.forceRep.colors(d.key)).brighter(); })
;
}
var bar = layout.langHg.selectAll(".barLang")
.data(data, function(d) { return d.key; });
bar.exit().remove();
bar.enter()
.append("g")
.attr("transform", "translate(" + [0 , -xc * 2] + ")")
.attr("class", "barLang")
.on("mouseover", me)
.on("mouseout", mo)
.call(appendItems);
bar.transition()
.duration(3500)
.ease("elastic")
.attr("transform", function(d) { return "translate(" + [x(d.key) , -xc * 2] + ")"; })
bar.each(function(k) {
d3.select(this).selectAll("*")
.datum(k);
});
bar.selectAll("path.hLine")
.attr("d", "M0,0 L" + (xc * 2) + ",0");
bar.selectAll("text.tLang")
.attr("y", -xc)
.each(function(d) {
var pr = d3.select(this.parentNode);
pr.selectAll("rect.barSelect")
.attr("transform", "translate(" + [-xc * .2 , -(this.clientWidth + 6 + xc * .4) ] + ")")
.attr("width", xc * 2.4 )
.attr("height", (this.clientWidth + 6 + xc * .4) + h_hg - y(d.values.length) + xc * 2.4);
});
var gg = bar.selectAll("g.barChain")
.attr("transform", "translate(" + [x.rangeBand() / 2 , 0] + ")");
var dg = gg.selectAll("g.dCircle");
dg.selectAll("circle")
.attr("r", xc);
dg.selectAll("text")
.text(function(d) { return d.values.length; });
dg.transition()
.delay(100)
.duration(3500)
.ease("elastic")
.attr("transform", function(d) {
return "translate(" + [0 , h_hg - y(d.values.length) + xc] + ")";
});
gg.selectAll("path.vLine").transition()
.duration(3500)
.ease("elastic")
.attr("d", function(d) { return "M0,0 L0," + (h_hg - y(d.values.length) + xc); });
}
})(vis || (vis = {}));
/**
* User: ArtZub
* Date: 11.01.13
* Time: 21:41
*/
'use strict';
var timeFormat = (function() {
var fd = d3.time.format("%b %d, %Y");
return function(ms) {
return fd(new Date(ms));
}
})();
var log;
var cs, svg_cs, svg,
margin = {top: 20, right: 20, bottom: 20, left: 20},
w, h, stackLoad = 0,
psBar, runBtn, ldrTop, toolTip, showBtn, visBtn,
userTxt, curRep, divStat, stepsBar, cbDlr, cbDlsr;
function updateStatus(pos, label) {
pos = pos > ghcs.states.max ? ghcs.states.max : pos;
psBar.setPos((pos * 100 / (ghcs.states.max || 1)) + "%")
.setLabel(label || "Completed " + pos + " of " + ghcs.states.max + " commits ...");
}
d3.select(window).on("hashchange", applyParams);
function parseParams(hash) {
var params = {};
hash.replace(/^#/, "").split("&").forEach(function(item) {
var values = item.split("=");
var key = values[0].toLowerCase();
params[key] = values.length > 1 ? values[1] : "";
});
ghcs.params = params;
ghcs.params.climit = params.climit || ghcs.limits.commits;
}
function rewriteHash() {
var step,
hash = [];
if (this == showBtn.node() && ghcs.params) {
step = 0;
}
else if (this == runBtn.node() && ghcs.params) {
step = 1;
}
switch (step) {
case 1:
ghcs.params.repo = ghcs.repo ? ghcs.repo.name : null;
ghcs.params.repo && hash.push("repo=" + ghcs.params.repo);
ghcs.params.climit > 0 && hash.push("climit=" + ghcs.params.climit);
case 0:
ghcs.params.user && hash.push("user=" + ghcs.params.user);
ghcs.params.rot && hash.push("rot=" + ghcs.params.rot);
break;
}
document.location.hash = "#" + hash.join("&");
}
function applyParams() {
d3.event && d3.event.preventDefault();
parseParams(document.location.hash);
stackLoad = stackLoad-- < 1 ? 0 : stackLoad;
if (ghcs.rot != ghcs.params.rot || ghcs.login != ghcs.params.user) {
ghcs.rot = ghcs.params.rot;
userTxt.property("value", ghcs.params.user);
ghcs.user = null;
if (ghcs.params.repo)
stackLoad++;
chUser();
}
else if (ghcs.user && ghcs.user.repos) {
// && (!ghcs.repo || ghcs.repo.name != ghcs.params.repo || (ghcs.params.climit > 0 && ghcs.limits.commits != ghcs.params.climit))
var r;
ghcs.limits.commits = ghcs.params.climit || ghcs.limits.commits;
d3.select("#txt-lc").property("value", ghcs.limits.commits);
if (!ghcs.repo
|| ghcs.repo.name != ghcs.params.repo
|| !ghcs.repo.commits
|| (!ghcs.repo.loadedAll && ghcs.repo.commitsCount < ghcs.limits.commits)) {
if (ghcs.repo && ghcs.repo.name == ghcs.params.repo)
r = ghcs.repo;
else
r = ghcs.user.repos.reduce(function(a, b) {
if (!a && b.nodeValue.name == ghcs.params.repo)
a = b;
return a;
}, null)
;
}
if (r) {
if (ghcs.repo != r) {
vis.meRepo(r);
vis.clRepo(r);
vis.mlRepo(r);
}
}
analyseCommits();
}
}
function nextStepApplyParams() {
if (stackLoad)
applyParams();
}
function chRadio(d) {
switch(this.name) {
case "participation" : {
}
}
}
function chCheckbox(d) {
var ln;
d = d3.select(this);
switch(d.attr("id")) {
case "cb-dlr":
case "cb-dlsr":
ln = d.datum();
vis.layers[ln]
&& ((d.property("checked") && vis.layers[ln].show()) || vis.layers[ln].hide());
break;
case "cb-dllh":
vis.layers.repo
&& vis.layers.repo.langHg
&& vis.layers.repo.langHg.style("display", d.property("checked") ? null : "none");
break;
case "cb-dlvml":
d.property("checked") ? cs.show() : cs.hide();
break;
default :
(ln = d.datum()) && ln.ns
&& (ln.ns[ln.key] = d.property("checked"));
break;
}
}
function chValue(d) {
var ln;
d = d3.select(this);
(ln = d.datum()) && ln.ns
&& (ln.ns[ln.key] = d.property("value"));
}
function checkCompleted() {
if (ghcs.states.cur >= ghcs.states.max) {
psBar.setPos("100%").hide();
if (ghcs.states.complete)
ghcs.states.complete();
return true;
}
return false;
}
function redrawStats() {
if (ghcs.redrawStatsTimer) {
clearTimeout(ghcs.redrawStatsTimer);
ghcs.redrawStatsTimer = null;
}
ghcs.redrawStatsTimer = setTimeout(function () {
vis.redrawStat(ghcs.repo);
ghcs.redrawStatsTimer = null;
}, 100);
}
function runShow() {
if (ghcs.repo && ghcs.repo.commits) {
visBtn.disable();
vis.runShow(ghcs.repo);
}
}
function redrawRepos() {
if (ghcs.redrawReposTimer) {
clearTimeout(ghcs.redrawReposTimer);
ghcs.redrawReposTimer = null;
}
ghcs.redrawReposTimer = setTimeout(function () {
vis.redrawRepos(ghcs.users
&& ghcs.users[ghcs.login]
&& ghcs.users[ghcs.login].repos
? ghcs.users[ghcs.login].repos
: null
);
vis.redrawLangHg(ghcs.users
&& ghcs.users[ghcs.login]
&& ghcs.users[ghcs.login].repos
? d3.nest().key(function(d) { return d.nodeValue.lang; }).entries(ghcs.users[ghcs.login].repos)
: null);
ghcs.redrawReposTimer = null;
}, 100);
}
function init() {
log = (function () {
var logCont = d3.select("#console")
.append("ul");
return function (msg) {
logCont.append("li").text(msg instanceof Object ? JSON.stringify(msg) : msg);
}
})();
cs = d3.select("#canvas");
cs.hide = function() {
this.style("display", "none");
vis.inited && vis.layers.show.hide();
return this;
};
cs.show = function() {
this.style("display", null);
vis.inited && vis.layers.show.show();
return this;
};
svg_cs = d3.select("#svg");
svg = svg_cs.append("svg");
w = svg.property("clientWidth") || document.body.clientWidth;
h = svg.property("clientHeight")|| document.body.clientHeight;
svg.attr("width", w).attr("height", h);
d3.selectAll("input").datum(function() {
var obj = null;
if (this.dataset && this.dataset.ns) {
var reg = new RegExp("(cb|n|txt)-" + this.dataset.ns + "-");
obj = {
ns : ghcs.settings[this.dataset.ns],
key : this.id.replace(reg, "")
};
if (this.type == "checkbox") {
this.checked = obj.ns[obj.key];
}
else {
this.value = obj.ns[obj.key];
}
}
return obj;
});
d3.selectAll("input[type=checkbox]").on("change", chCheckbox);
d3.selectAll("input[type=number], input[type=text]").on("change", chValue);
psBar = d3.select("#progressBar");
psBar.pntNode = d3.select(psBar.node().parentNode);
psBar.show = function() {
this.pntNode.style("display", null);
return this;
};
psBar.hide = function() {
this.pntNode.style("display", "none");
return this;
};
psBar.setLabel = function(lbl) {
this.text(lbl);
return this;
};
psBar.setPos = function(pos) {
this.style("width", pos);
return this;
};
stepsBar = d3.select(".steps");
stepsBar.firstStep = function() {
this.attr("class", "steps sfirst");
return this;
};
stepsBar.secondStep = function() {
this.attr("class", "steps ssecond");
return this;
};
stepsBar.thirdStep = function() {
this.attr("class", "steps");
return this;
};
runBtn = d3.select("#runBtn");
showBtn = d3.select("#showBtn");
visBtn = d3.select("#visBtn");
userTxt = d3.select("#user").on("change", function() {
stepsBar.firstStep();
showBtn.disable();
if (this.value) {
if (this.value != ghcs.login)
showBtn.enable();
else
stepsBar.secondStep();
}
(ghcs.params || (ghcs.params = {})).user = this.value;
});
[runBtn, showBtn, userTxt, visBtn].forEach(function(item) {
item.enable = function () {
this.attr("disabled", null);
return this;
};
item.disable = function () {
this.attr("disabled", "disabled");
return this;
};
});
runBtn.on("click", rewriteHash);
showBtn.on("click", rewriteHash);
visBtn.on("click", runShow);
ldrTop = d3.select("#ldrTop");
ldrTop.pntNode = d3.select(ldrTop.node().parentNode);
ldrTop.show = function () {
this.pntNode.style("display", null);
return this;
};
ldrTop.hide = function () {
this.pntNode.style("display", "none");
return this;
};
toolTip = d3.select("#tooltip");
toolTip.show = function () {
this.style("display", "block");
return this;
};
toolTip.hide = function () {
this.style("display", null);
return this;
};
cbDlr = d3.select("#cb-dlr").datum("repo");
cbDlsr = d3.select("#cb-dlsr").datum("stat");
[cbDlr, cbDlsr].forEach(function(item) {
item.check = function() {
this.property("checked", true);
chCheckbox.apply(this.node());
};
item.uncheck = function() {
this.property("checked", false);
chCheckbox.apply(this.node());
};
item.trigger = function() {
this.property("checked", !this.property("checked"));
chCheckbox.apply(this.node());
};
});
d3.select("#txt-lc").on("change", function() {
(ghcs.params || (ghcs.params = {})).climit = +this.value;
if (ghcs.params.climit < 1)
ghcs.params.climit = ghcs.limits.commits;
});
initGraphics(svg);
curRep = d3.select("#curRep")
.on("mouseover", function(d) {
if (d) {
vis.meRepo(d);
vis.mtt(d, null, null, {pageX : d.x, pageY : d.y});
}
})
.on("mouseout", function(d) {
if (d)
vis.mlRepo(d);
});
curRep.setName = function(r) {
this.selectAll("*").remove();
this.datum(r);
if (!r)
this.append("span")
.text("Select Repo...");
else {
this.append("span")
.style("color", d3.rgb(vis.forceRep.colors(r.nodeValue.lang)).brighter())
.attr("class", "mega-icon mega-icon-public-repo");
this.append("strong")
.style("margin-right", "5px")
.style("text-shadow", "0 0 3px rgba(0, 0, 0, 1)")
.style("color", d3.rgb(vis.forceRep.colors(r.nodeValue.lang)).brighter())
.text((r.nodeValue.name || ""));
this.append("a")
.attr("target", "_blank")
.attr("title", "Go to Github")
.attr("href", (r.nodeValue.html_url || "#"))
.attr("class", "mega-icon mini-icon-link a-icon");
}
return this;
};
divStat = d3.select("#divStat");
divStat.updateInfo = function() {
var user;
if (ghcs.login && (user = ghcs.user = ghcs.users[ghcs.login]) && user.info) {
divStat.selectAll("*").remove();
user.info.avatar && divStat.node().appendChild(user.info.avatar);
divStat.append("ul")
.call(function(ul) {
(user.info.name || user.info.login) && ul.append("li").call(function(li) {
li.append("h1")
.text((user.info.name || user.info.login))
.append("a")
.attr("class", "a-icon")
.attr("target", "_blank")
.attr("title", "Go to GitHub")
.attr("href", user.info.html_url)
.append("span")
.attr("class", "mini-icon mini-icon-octocat")
;
li.append("hr");
});
user.info.location && ul.append("li")
.html("<span class='mini-icon mini-icon-location'></span><strong>" + user.info.location + "</strong>")
user.info.blog && ul.append("li")
.call(function(li) {
li.append("span")
.attr("class", "mini-icon mini-icon-link");
li.append("a")
.attr("target", "_blank")
.attr("href", user.info.blog)
.text(user.info.blog)
});
ul.append("li")
.call(function(li) {
li.append("span")
.attr("class", "mini-icon mini-icon-public-repo")
li.append("strong")
.text(user.info.public_repos)
});
user.info.updated_at && ul.append("li")
.call(function(li) {
li.append("span")
.attr("class", "mini-icon mini-icon-time");
li.append("strong")
.text(timeFormat(Date.parse(user.info.updated_at)))
});
})
}
};
applyParams();
}
/**
* User: ArtZub
* Date: 23.01.13
* Time: 13:29
*/
'use strict';
(function(vis) {
vis.mdRepo = function(d) {
};
vis.meRepo = function(d) {
vis.layers.repo.langHg
&& vis.layers.repo.langHg.style("pointer-events", "none");
vis.layers.repo.toFront();
d._tg = (d._g || (d._g = d3.select(this)));
d._tg.selectAll("circle")
.style("fill", d3.rgb(vis.forceRep.colors(d.nodeValue.lang)).brighter());
d._tg.selectAll("text")
.style("fill", function(d) {
return d3.rgb(vis.forceRep.colors(d.nodeValue.lang)).darker();
})
.style("visibility", "visible");
toolTip.selectAll("*").remove();
toolTip.append("h1")
.text(d.nodeValue.name);
toolTip.append("hr");
d.nodeValue.desc
&& toolTip.append("blockquote")
.text(d.nodeValue.desc)
&& toolTip.append("br")
;
toolTip.append("span")
.attr("class", "mini-icon mini-icon-time")
toolTip.append("strong")
.style("margin-left", "5px")
.text(timeFormat(d.nodeValue.date));
toolTip.append("br");
toolTip.append("span")
.text("Primary language: ")
.append("strong")
.style("color", vis.forceRep.colors(d.nodeValue.lang))
.style("text-shadow", "0 0 3px rgba(0, 0, 0, 0.8)")
.text(d.nodeValue.lang);
toolTip.show();
};
vis.mlRepo = function(d, i) {
if (vis.forceRep.selected && vis.forceRep.selected == d && i !== "deselect") {
vis.muRepo(d);
}
else {
var g = d._tg || d._g;
if (!g)
return;
g.selectAll("circle")
.style("fill", toRgba(vis.forceRep.colors(d.nodeValue.lang), vis.forceRep.opt(vis.forceRep.radO(d))));
g.selectAll("text")
.style("fill", function(d) {
return d3.rgb(vis.forceRep.colors(d.nodeValue.lang)).brighter();
})
.style("visibility", vis.forceRep.visible);
d._tg = null;
}
vis.layers.repo.langHg
&& vis.layers.repo.langHg.style("pointer-events", "all");
toolTip.hide();
};
vis.clRepo = function(d) {
if (vis.forceRep.selected && vis.forceRep.selected == d) {
vis.forceRep.selected = null;
d && (d.fixed = 4);
}
else {
if (vis.forceRep.selected) {
vis.forceRep.selected.fixed = 0;
vis.mlRepo(vis.forceRep.selected, "deselect");
toolTip.show();
vis.layers.repo.langHg
&& vis.layers.repo.langHg.style("pointer-events", "none");
}
vis.forceRep.selected = d;
d && (d.fixed = true);
}
chSelect(vis.forceRep.selected);
};
vis.muRepo = function(d){
};
vis.clearRepos = function() {
if (vis.forceRep) {
vis.forceRep.stop().nodes([]);
delete vis.forceRep;
}
vis.layers && vis.layers.repo && vis.layers.repo.selectAll("*")
.transition()
.duration(750)
.ease("elastic")
.remove();
vis.layers.repo.langHg = null;
};
vis.redrawRepos = function(data, layout) {
layout = layout || vis.layers.repo;
if (!data) {
vis.clearRepos();
return;
}
function tr(d) {
return "translate(" + [d.x, d.y] + ")";
}
vis.forceRep = vis.forceRep || d3.layout.force()
.size([w, h])
.friction(.99)
.gravity(.005)
.charge(function(d) { return -vis.forceRep.radius(vis.forceRep.rad(d)) / 2; })
.on("tick", tick)
;
vis.forceRep.dateNow = Date.now();
vis.forceRep.rad = vis.forceRep.rad || function (d) {
return d.nodeValue.cdate;
};
vis.forceRep.radO = vis.forceRep.radO || function (d) {
return d.nodeValue.date - vis.forceRep.dateNow;
};
data = data.sort(function(a, b) { return a.nodeValue.date - b.nodeValue.date; });
var kof = (h > w ? h : w) / (4 * ((Math.log(data.length || 1) / Math.log(1.5)) || 1));
var r = [kof / 5, kof];
(vis.forceRep.radius || (vis.forceRep.radius = d3.scale.linear()))
.range(r)
.domain(d3.extent(data, vis.forceRep.rad));
data.length == 1 && vis.forceRep.radius.domain([vis.forceRep.radius.domain()[0] - 1, vis.forceRep.radius.domain()[1]]);
(vis.forceRep.opt || (vis.forceRep.opt = d3.scale.log().range([.01,.9])))
.domain(
d3.extent(data, vis.forceRep.radO)
// [d3.min(data, vis.forceRep.radO), vis.forceRep.dateNow]
);
vis.forceRep.colors = vis.reposColors || (vis.reposColors = d3.scale.category20());
vis.forceRep.visible = vis.forceRep.visible || function(d) {
return this.clientWidth < vis.forceRep.radius(vis.forceRep.rad(d)) * 2.1 ? null : "hidden";
};
vis.forceRep.appCT = vis.forceRep.appCT || function(g) {
g.each(function(d) {
d._g = d3.select(this);
});
g.append("circle")
.attr("r", 0);
g.append("text")
.attr("text-anchor", "middle")
.attr("dy", ".31em")
.text(function(d) { return d.nodeValue.name; });
};
vis.forceRep.upCT = vis.forceRep.upCT || function(g) {
g.selectAll("circle")
.style("stroke-width", 1)
.style("stroke", function(d) { return d3.rgb(vis.forceRep.colors(d.nodeValue.lang)); })
.style("fill", function(d) { return toRgba(d3.rgb(vis.forceRep.colors(d.nodeValue.lang)), vis.forceRep.opt(vis.forceRep.radO(d))); })
.transition()
.duration(2500)
.ease("elastic")
.attr("r", function(d) { return vis.forceRep.radius(vis.forceRep.rad(d)); })
g.selectAll("text")
.style("fill", function(d) {
return d3.rgb(vis.forceRep.colors(d.nodeValue.lang)).brighter();
})
.style("visibility", vis.forceRep.visible);
};
vis.forceRep
.stop()
.nodes(data)
.start()
;
vis.forceRep.circle = layout.selectAll(".cRepo")
.data(data, function(d) { return d.nodeValue.id })
;
vis.forceRep.circle.enter()
.append("g")
.attr("class", "cRepo")
.attr("transform", tr)
.call(vis.forceRep.drag)
.on("mouseover.select", vis.meRepo)
.on("mouseout.select", vis.mlRepo)
.on("mousedown.select", vis.mdRepo)
.on("mouseup.select", vis.muRepo)
.on("mousemove.mtt", vis.mtt)
.on("click.select", vis.clRepo)
.call(vis.forceRep.appCT);
vis.forceRep.circle.call(vis.forceRep.upCT);
vis.forceRep.circle.exit().remove();
function tick(e) {
var quadtree = d3.geom.quadtree(vis.forceRep.nodes());
vis.forceRep.circle
.each(cluster(.025/*10 * e.alpha * e.alpha*/))
.each(collide(.5, quadtree))
.attr("transform", tr);
vis.forceRep.resume();
}
// Move d to be adjacent to the cluster node.
function cluster(alpha) {
vis.forceRep.cenralNodes = vis.forceRep.cenralNodes || {};
// Find the largest node for each cluster.
vis.forceRep.nodes().forEach(function(d, n) {
n = vis.forceRep.cenralNodes[d.nodeValue.lang];
(!n || vis.forceRep.radO(d) > vis.forceRep.radO(n)) &&
(vis.forceRep.cenralNodes[d.nodeValue.lang] = d);
});
return function(d) {
var node = vis.forceRep.cenralNodes[d.nodeValue.lang],
l,
r,
x,
y;
if (node == d) return;
x = d.x - node.x;
y = d.y - node.y;
l = Math.sqrt(x * x + y * y);
r = vis.forceRep.radius(vis.forceRep.rad(d)) + vis.forceRep.radius(vis.forceRep.rad(node)) * 1.5;
if (l != r) {
l = (l - r) / (l || 1) * (alpha || 1);
x *= l;
y *= l;
//if (!d.fixed) {
if (true) {
d.x -= x;
d.y -= y;
}
//if (!node.fixed) {
if (true) {
node.x += x;
node.y += y;
}
}
};
}
// Resolves collisions between d and all other circles.
function collide(alpha, quadtree) {
return function(d) {
var padding = vis.forceRep.radius.range()[1] / 2,
r = vis.forceRep.radius(vis.forceRep.rad(d)) + 3 * padding,
nx1 = d.x - r,
nx2 = d.x + r,
ny1 = d.y - r,
ny2 = d.y + r;
quadtree.visit(function(quad, x1, y1, x2, y2) {
if (quad.point && (quad.point !== d)) {
var x = d.x - quad.point.x,
y = d.y - quad.point.y,
l = Math.sqrt(x * x + y * y),
r = (vis.forceRep.radius(vis.forceRep.rad(d)) + vis.forceRep.radius(vis.forceRep.rad(quad.point))) * 1.02 /*+ (d.nodeValue.lang !== quad.point.nodeValue.lang) * padding*/;
if (l < r) {
l = (l - r) / (l || 1) * (alpha || 1);
x *= l;
y *= l;
//if (!d.fixed) {
if (true) {
d.x -= x;
d.y -= y;
}
//if (!quad.point.fixed) {
if (true) {
quad.point.x += x;
quad.point.y += y;
}
}
}
return x1 > nx2
|| x2 < nx1
|| y1 > ny2
|| y2 < ny1;
});
};
}
};
})(vis || (vis = {}));
/**
* User: ArtZub
* Date: 23.01.13
* Time: 15:00
*/
"use strict";
(function(vis) {
var _worker,
_data,
nodes,
dateRange,
authorHash,
filesHash,
extHash,
extMax,
_force,
_forceAuthor,
links,
lCom, lLeg, lHis,
canvas, ctx,
bufCanvas, bufCtx,
layer,
valid,
pause,
stop,
particle,
defImg,
lastEvent,
zoomScale,
xW,
yH,
setting,
rd3 = d3.random.irwinHall(8);
var extColor = d3.scale.category20(),
userColor = d3.scale.category20b();
var typeNode = {
author : 0,
file : 1
};
function reCalc(d) {
if (stop)
return;
lCom.showCommitMessage(d.message);
appendExtLegend(d.sha);
var l = d.nodes.length,
n, a, fn;
a = d.cuserNode ? d.cuserNode : d.userNode;
a.fixed = false;
if (!l)
console.log(d);
else {
a.alive = setting.userLife > 0 ? setting.userLife : 1;
a.opacity = 100;
a.flash = 100;
a.visible = true;
}
while(--l > -1) {
n = d.nodes[l];
if (n.fixed) {
n.x = xW(n.x);
n.y = yH(n.y);
}
n.size += 2;
n.fixed = false;
n.author = a.nodeValue.email;
n.visible = !!n.statuses[d.sha].status;
fn = n.nodeValue.name.toLowerCase();
if (n.visible) {
n.ext.now.indexOf(fn) < 0
&& n.ext.now.push(fn);
n.flash = 100;
n.alive = setting.fileLife > 0 ? setting.fileLife : 1;
n.opacity = 100;
}
else {
(fn = n.ext.now.indexOf(fn)) > -1
&& n.ext.now.splice(fn, 1);
n.alive = (setting.fileLife > 0 ? setting.fileLife : 1) * .2;
n.opacity = 50;
}
/*data.push(n);
if (d.cuserNode) {
links.push({
source : n,
target : d.cuserNode
});
}
links.push({
source : n,
target : d.userNode
});*/
}
updateLegend(d.sha);
_force.nodes(nodes.filter(function(d) {
return d.type != typeNode.author && (d.visible || d.opacity);
}).sort(sortBySize)
).start();
_forceAuthor.nodes(nodes.filter(function(d) {
return d.type == typeNode.author && (d.visible || d.opacity);
})).start();
}
function loop() {
if (pause)
return;
var dl, dr;
dl = dateRange[0];
dr = dl + ghcs.limits.stepShow * ghcs.limits.stepType;
dateRange[0] = dr;
var visTurn = _data.filter(function (d) {
return d.date >= dl && d.date < dr;
});
ghcs.asyncForEach(visTurn, reCalc, ONE_SECOND / (visTurn.length > 1 ? visTurn.length : ONE_SECOND));
//visTurn.forEach(reCalc);
updateStatus(ghcs.states.cur += ghcs.limits.stepShow * ghcs.limits.stepType, timeFormat(new Date(dr)));
if (dl >= dateRange[1]) {
if (_worker)
clearInterval(_worker);
} else {
if (!visTurn.length && setting.skipEmptyDate)
loop();
}
updateExtHistogram();
}
function run() {
if (_worker)
clearInterval(_worker);
anim();
_worker = setInterval(loop, ONE_SECOND);
}
function nr(d) {
return d.size > 0 ? d.size : 0;
}
function ncb(d) {
return d3.rgb(d.color).brighter().brighter();
}
function nc(d) {
return d.color;
}
function randomTrue() {
return Math.floor(rd3() * 8) % 2;
}
function radius(d) {
return Math.sqrt(d);
}
function node(d, type) {
var c = type == typeNode.file ? d.name : userColor(d.email),
ext, x, y,
w2 = w/2,
w5 = w/5,
h2 = h/2,
h5 = h/5;
if (type == typeNode.file) {
c = c && c.match(/.*(\.\w+)$/) ? c.replace(/.*(\.\w+)$/, "$1").toLowerCase() : "Mics";
ext = extHash.get(c);
if (!ext) {
ext = {
all : 0,
currents : {},
color : extColor(c),
now : []
};
extHash.set(c, ext);
}
ext.all++;
c = ext.color;
}
x = w * Math.random();
y = h * Math.random();
if (type == typeNode.author) {
if (randomTrue()) {
x = x > w5 && x < w2
? x / 5
: x > w2 && x < w - w5
? w - x / 5
: x
;
}
else {
y = y > h5 && y < h2
? y / 5
: y > h2 && y < h - h5
? h - y / 5
: y
;
}
}
return {
x : x,
y : y,
id : type + (type == typeNode.file ? d.name : d.email),
size : type != typeNode.file ? 24 : 2,
weight : type != typeNode.file ? 24 : 2,
fixed : true,
visible : false,
links : 0,
type : type,
color : c,
ext : ext,
author : type == typeNode.author ? d.email : null,
img : type == typeNode.author ? d.avatar : null,
nodeValue : d
}
}
function getAuthor(d) {
if (!d || !d.author)
return null;
var n = authorHash.get(d.author.email);
if (!n) {
n = node(d.author, typeNode.author);
authorHash.set(d.author.email, n);
}
return n;
}
function getFile(d) {
if (!d || !d.name)
return null;
var n = filesHash.get(d.name);
if (!n) {
n = node(d, typeNode.file);
n.links = 1;
filesHash.set(d.name, n);
}
return n;
}
function initNodes(data) {
var ns = [],
i, j, n, d, df;
authorHash = d3.map({});
filesHash = d3.map({});
extHash = d3.map({});
extMax = 0;
if (data) {
i = data.length;
while(--i > -1) {
d = data[i];
d.nodes = [];
if (!d) continue;
n = getAuthor(d);
d.userNode = n;
!n.inserted && (n.inserted = ns.push(n));
if (d.author.login != d.committer.login) {
n = getAuthor(d);
d.cuserNode = n;
!n.inserted && (n.inserted = ns.push(n));
/*links.push({
source : d.userNode,
target : d.cuserNode
})*/
}
if (!d.files) continue;
j = d.files.length;
while(--j > -1) {
df = d.files[j];
if (!df) continue;
n = getFile(df);
d.nodes.push(n);
n.size = 2;
n.statuses = n.statuses || {};
n.statuses[d.sha] = df;
n.ext.currents[d.sha] = (n.ext.currents[d.sha] || 0);
n.ext.currents[d.sha]++;
!n.inserted && (n.inserted = ns.push(n));
}
j = extHash.values().reduce(function(a, b) {
return a += b.currents[d.sha] || 0;
}, null);
extMax = j > extMax ? j : extMax;
}
}
return ns;
}
var tempCanvas, tempFileCanvas;
function setOpacity(img, a, f) {
if (!img || !img.width)
img = defImg;
return img;
}
function colorize(img, r, g, b, a) {
if (!img)
return img;
if (!tempFileCanvas) {
tempFileCanvas = document.createElement("canvas");
tempFileCanvas.width = img.width;
tempFileCanvas.height = img.height;
}
var imgCtx = tempFileCanvas.getContext("2d"), imgdata, i;
imgCtx.clearRect(0, 0, img.width, img.height);
imgCtx.save();
imgCtx.drawImage(img, 0, 0);
imgdata = imgCtx.getImageData(0, 0, img.width, img.height);
i = imgdata.data.length;
while((i -= 4) > -1) {
imgdata.data[i + 3] = imgdata.data[i] * a;
if (imgdata.data[i + 3]) {
imgdata.data[i] = r;
imgdata.data[i + 1] = g;
imgdata.data[i + 2] = b;
}
}
imgCtx.putImageData(imgdata, 0, 0);
imgCtx.restore();
return tempFileCanvas;
}
function blink(d, aliveCheck, i, l) {
d.flash = (d.flash -= setting.rateFlash) > 0 ? d.flash : 0;
!d.flash && aliveCheck
&& (d.alive = (d.alive-- > 0 ? d.alive : 0))
;
d.opacity = !d.alive
? ((d.opacity -= setting.rateOpacity) > 0 ? d.opacity : 0)
: d.opacity
;
d.visible && !d.opacity
&& (d.visible = false);
}
function sortBySize(a, b) {
return d3.descending(b.size, a.size);
}
function checkVisible(d, offsetx, offsety) {
var tx = lastEvent.translate[0]/lastEvent.scale,
ty = lastEvent.translate[1]/lastEvent.scale
;
offsetx = offsetx || 0;
if (!(offsetx instanceof Array))
offsetx = [offsetx, offsetx];
offsety = offsety || 0;
if (!(offsety instanceof Array))
offsety = [offsety, offsety];
return (
d.x + d.size > -tx + offsetx[0]
&& d.x - d.size < -tx + offsetx[1] + w/lastEvent.scale
&& d.y + d.size > -ty + offsety[0]
&& d.y - d.size < -ty + offsety[1] + h/lastEvent.scale
);
}
function redrawCanvas() {
bufCtx.save();
bufCtx.clearRect(0, 0, w, h);
bufCtx.translate(lastEvent.translate[0], lastEvent.translate[1]);
bufCtx.scale(lastEvent.scale, lastEvent.scale);
var n, l, i, j,
img,
d, d1, d2,
c, x, y, s;
if (setting.showFile) {
n = d3.nest()
.key(function(d) {
return d.opacity;
})
.key(function(d) {
return d.flash ? ncb(d) : d3.rgb(nc(d));
})
.entries(_force.nodes().filter(function(d) { return checkVisible(d) && (d.visible || d.alive); }));
l = n.length;
while(--l > -1) {
d1 = n[l];
i = d1.values.length;
while(--i > -1) {
d2 = d1.values[i];
j = d2.values.length;
c = d3.rgb(d2.key);
if (!setting.showHalo) {
bufCtx.beginPath();
bufCtx.strokeStyle = "none";
bufCtx.fillStyle = toRgba(c, d1.key * .01);
}
else
img = colorize(particle, c.r, c.g, c.b, d1.key * .01);
while(--j > -1) {
d = d2.values[j];
if (d.visible || d.alive) {
//blink(d, setting.fileLife > 0);
x = Math.floor(d.x);
y = Math.floor(d.y);
s = radius(nr(d)) * (setting.showHalo ? 8 : 1);
setting.showHalo
? bufCtx.drawImage(img, x - s / 2, y - s / 2, s, s)
: bufCtx.arc(x, y, s, 0, PI_CIRCLE, true)
;
}
}
if (!setting.showHalo) {
bufCtx.fill();
bufCtx.stroke();
bufCtx.closePath();
}
}
}
}
if (setting.showUser || setting.showLabel) {
n = _forceAuthor.nodes();
l = n.length;
while(--l > -1) {
d = n[l];
if (checkVisible(d) && (d.visible || d.opacity)) {
//blink(d, !d.links && setting.userLife > 0);
x = Math.floor(d.x);
y = Math.floor(d.y);
if (setting.showUser) {
c = d.flash ? ncb(d) : d3.rgb(nc(d));
bufCtx.save();
if (setting.showPaddingCircle) {
bufCtx.beginPath();
bufCtx.strokeStyle = "none";
bufCtx.fillStyle = toRgba("#ff0000", .1);
bufCtx.arc(x, y, nr(d) + setting.padding, 0, PI_CIRCLE, true);
bufCtx.fill();
bufCtx.stroke();
}
bufCtx.beginPath();
bufCtx.strokeStyle = "none";
bufCtx.fillStyle = setting.useAvatar ? "none" : toRgba(c, d.opacity * .01);
bufCtx.arc(x, y, nr(d), 0, PI_CIRCLE, true);
bufCtx.fill();
bufCtx.stroke();
if (setting.useAvatar && d.img) {
bufCtx.clip();
bufCtx.drawImage(setOpacity(d.img, d.opacity * .01, d.flash), x - nr(d), y - nr(d), nr(d) * 2, nr(d) * 2);
}
bufCtx.closePath();
bufCtx.restore();
}
if (setting.showLabel) {
c = d.flash ? "white" : "gray";
bufCtx.save();
bufCtx.fillStyle = toRgba(c, d.opacity * .01);
bufCtx.fillText(setting.labelPattern
.replace("%l", d.nodeValue.login)
.replace("%n", d.nodeValue.name != "unknown" ? d.nodeValue.name : d.nodeValue.login)
.replace("%e", d.nodeValue.email), x, y + nr(d) * 1.5);
bufCtx.restore();
}
}
}
}
bufCtx.restore();
}
function anim() {
requestAnimationFrame(anim);
lHis && lHis.style("display", setting.showHistogram ? null : "none");
lLeg && lLeg.style("display", setting.showCountExt ? null : "none");
if (valid)
return;
valid = true;
ctx.save();
ctx.clearRect(0, 0, w, h);
redrawCanvas();
ctx.drawImage(bufCanvas, 0, 0);
ctx.restore();
valid = false;
}
function tick() {
if (_force.nodes()) {
_force.nodes()
.forEach(cluster(0.025));
_forceAuthor.nodes(
_forceAuthor.nodes()
.filter(function(d, i) {
blink(d, !d.links && setting.userLife > 0, i, _forceAuthor.nodes().length);
if (d.visible && d.links === 0 && setting.userLife > 0) {
d.flash = 0;
d.alive = d.alive / 10;
}
return d.visible;
})
);
}
_forceAuthor.resume();
_force.resume();
}
// Move d to be adjacent to the cluster node.
function cluster(alpha) {
authorHash.forEach(function(k, d) {
d.links = 0;
});
return function(d, i) {
blink(d, setting.fileLife > 0, i, _force.nodes().length);
if (!d.author || !d.visible)
return;
var node = authorHash.get(d.author),
l,
r,
x,
y;
if (node == d) return;
node.links++;
x = d.x - node.x;
y = d.y - node.y;
l = Math.sqrt(x * x + y * y);
r = radius(nr(d)) / 2 + (nr(node) + setting.padding);
if (l != r) {
l = (l - r) / (l || 1) * (alpha || 1);
x *= l;
y *= l;
d.x -= x;
d.y -= y;
}
};
}
function appendExtLegend(sha){
if (!layer)
return;
var data = [],
w3 = w / 3,
ml = w * .01,
mb = 18,
h2 = (h / 2) - mb,
bw = 2,
i, ny
;
var y = d3.scale.linear()
.range([0, h2])
.domain([0, extMax]);
lHis = (lHis || layer.append("g"))
.attr("width", w3)
.attr("height", h2)
.attr("transform", "translate(" + [ ml , h - h2 - mb ] + ")");
if (!sha)
return;
ny = h2;
extHash.forEach(function(k, d) {
var obj = {
key : k,
h : y(d.currents[sha] || 0),
color : d.color
};
obj.y = ny -= obj.h;
data.push(obj);
});
updateExtHistogram();
var g = lHis.append("g")
.attr("class", "colStack")
.datum({ x : w3, w : bw })
.style("opacity", 0);
g.selectAll("rect")
.data(data)
.enter()
.append("rect")
.attr("x", 0)
.attr("y", function(d) { return d.y; })
.attr("width", bw)
.attr("height", function(d) { return d.h; })
.attr("fill", function(d) { return d.color; })
;
g.style("opacity", 1)
.attr("transform", function(d) {
return "translate(" + [ d.x, 0] + ")";
});
}
function updateExtHistogram() {
if (!lHis || lHis.selectAll(".colStack").empty())
return;
lHis.selectAll(".colStack")
.attr("transform", function(d) {
return "translate(" + [ d.x -= d.w - 1, 0] + ")";
})
.filter(function(d) {
return d.x < 0;
})
.remove();
}
function initLegend() {
if (!layer)
return;
var mt = 48,
ml = w * .01,
h2 = h / 2 - mt,
w3 = w / 3
;
lLeg = (lLeg || layer.append("g"))
.attr("width", w3)
.attr("height", h2)
.attr("transform", "translate(" + [ml, mt] + ")");
lLeg.selectAll("*").remove();
var g = lLeg.selectAll(".gLeg")
.data(extHash.entries(), function(d) { return d.key; });
g.exit().remove();
g.enter().append("g")
.attr("class", "gLeg")
.attr("transform", function(d, i) {
return "translate(" + [0, i * 18] + ")";
})
.style("visibility", "hidden")
;
g.append("rect")
.attr("height", 16)
.style("fill", function(d) { return d.value.color; })
;
g.append("text")
.attr("class", "gttLeg")
.style("font-size", "13px")
.text(function(d) { return d.key.substr(1); })
.style("fill", function(d) { return d3.rgb(d.value.color).brighter().brighter(); })
;
g.append("text")
.attr("class", "gtLeg")
.style("font-size", "11px")
.attr("transform", "translate(" + [2, 12] + ")")
;
}
function sortLeg(b, a) {
return d3.ascending(a.value.now.length, b.value.now.length);
}
function sortLegK(b, a) {
return d3.ascending(a.key, b.key);
}
function updateLegend(sha) {
if (!lLeg || lLeg.empty())
return;
var g = lLeg.selectAll(".gLeg");
function wl(d) {
return d.value.now.length;
}
var wb = 9 * d3.max(g.data(), function(d) {
return ((wl(d) || "") + "").length;
});
g.selectAll("rect")
.attr("width", wb)
;
g.selectAll(".gttLeg")
.attr("transform", "translate(" + [wb + 2, 12] + ")")
;
g.selectAll(".gtLeg")
.text(wl)
;
g.sort(sortLegK).sort(sortLeg)
.style("visibility", function(d, i) {
return !wl(d) || i * 18 > lLeg.attr("height") ? "hidden" : "visible";
})
.transition()
.duration(500)
.attr("transform", function(d, i) {
return "translate(" + [0, i * 18] + ")";
})
;
}
vis.runShow = function(data, svg, params) {
if (_worker)
clearInterval(_worker);
_data = data && data.commits ? data.commits.values().sort(vis.sC) : null;
if (!_data || !_data.length)
return;
setting = ghcs.settings.cs;
vis.layers.repo && cbDlr.uncheck();
vis.layers.stat && cbDlsr.uncheck();
dateRange = d3.extent(data.dates);
layer = d3.select("#canvas");
layer.select("#mainCanvas").remove();
lastEvent = {
translate: [0, 0],
scale : 1
};
xW = d3.scale.linear()
.range([0, w])
.domain([0, w]);
yH = d3.scale.linear()
.range([0, h])
.domain([0, h]);
var zoom = d3.behavior.zoom()
.scaleExtent([.1, 8])
.scale(1)
.translate([0, 0])
.on("zoom", function() {
lastEvent.translate = d3.event.translate.slice(0);
lastEvent.scale = d3.event.scale;
var tl = lastEvent.translate[0] / lastEvent.scale,
tt = lastEvent.translate[1] / lastEvent.scale;
xW.range([-tl, -tl + w / lastEvent.scale])
.domain([0, w]);
yH.range([-tt, -tt + h / lastEvent.scale])
.domain([0, h]);
valid = false;
});
canvas = layer.append("canvas")
.text("This browser don't support element type of Canvas.")
.attr("id", "mainCanvas")
.attr("width", w)
.attr("height", h)
.call(zoom)
.node();
ctx = canvas.getContext("2d");
bufCanvas = document.createElement("canvas");
bufCanvas.width = w;
bufCanvas.height = h;
bufCtx = bufCanvas.getContext("2d");
bufCtx.font = "normal normal " + setting.sizeUser / 2 + "px Tahoma";
bufCtx.textAlign = "center";
layer = svg || vis.layers.show;
layer && layer.show && layer.show();
layer.append("g")
.call(zoom)
.append("rect")
.attr("width", w)
.attr("height", h)
.attr("x", 0)
.attr("y", 0)
.style("fill", "#ffffff")
.style("fill-opacity", 0);
lHis && lHis.selectAll("*").remove();
lCom = (
lCom || layer.append("g")
.attr("width", 10)
.attr("height", 14)
)
.attr("transform", "translate(" + [w/2, h - 18] + ")")
;
lCom.visible = !setting.showCommitMessage;
lCom.selectAll("text").remove();
lCom.showCommitMessage = lCom.showCommitMessage || function(text) {
if (setting.showCommitMessage && !lCom.visible) {
lCom.visible = true;
lCom.style("display", null);
}
else if (!setting.showCommitMessage && lCom.visible) {
lCom.visible = false;
lCom.style("display", "none");
}
lCom.append("text")
.attr("text-anchor", "middle")
.attr("class", "com-mess")
.attr("transform", "translate("+ [0, -lCom.node().childElementCount * 14] +")")
.text(text.split("\n")[0].substr(0, 100))
.transition()
.delay(500)
.duration(2000)
.style("fill-opacity", 1)
.duration(200)
.style("font-size", "11.2pt")
.transition()
.duration(1500)
.style("fill-opacity", .3)
.style("font-size", "11pt")
.each("end", function() {
lCom.selectAll("text").each(function(d, i) {
d3.select(this)
.attr("transform", "translate("+ [0, -i * 14] +")");
});
})
.remove();
};
psBar.show();
ghcs.states.cur = 0;
ghcs.states.max = dateRange[1] - dateRange[0];
updateStatus(ghcs.states.cur, timeFormat(new Date(dateRange[0])));
links = [];
nodes = initNodes(_data);
defImg = new Image();
defImg.src = "default.png";
particle = new Image();
particle.src = "particle.png";
_force = (_force || d3.layout.force()
.stop()
.size([w, h])
.friction(.75)
.gravity(0)
.charge(function(d) {return -3 * radius(nr(d)); } )
.on("tick", tick))
.nodes([])
;
zoomScale = d3.scale.linear()
.range([5, 1])
.domain([.1, 1]);
_forceAuthor = (_forceAuthor || d3.layout.force()
.stop()
.size([w, h])
.gravity(setting.padding * .001)
.charge(function(d) { return -(setting.padding + d.size) * 8
* (Math.sqrt(d.links / lastEvent.scale) || 1)
;
}))
.nodes([])
;
initLegend();
stop = false;
pause = false;
run();
_force.start();
_forceAuthor.start();
};
vis.pauseShow = function() {
pause = true;
};
vis.stopShow = function() {
stop = true;
if (_worker)
clearInterval(_worker);
};
vis.resumeShow = function() {
pause = false;
};
})(vis || (vis = {}));
/**
* User: ArtZub
* Date: 23.01.13
* Time: 12:54
*/
'use strict';
(function (vis) {
var textFormat = d3.format(",");
vis.meArc = function(d) {
vis.layers.stat.toFront();
if (!d._g)
return;
d._g.selectAll("path.bar")
.style({
"fill-opacity" : .6,
"stroke-width" : 1,
"stroke" : function() { return toRgba("#000000", 1); }
});
var self = d._g.node();
self.barDel && self.barDel
.transition()
.attr("d", self.delArc.startAngle(PIdiv2 + smallRad )
.endAngle(PIdiv2 * 3 - smallRad)())
;
self.barAdd && self.barAdd
.transition()
.attr("d", self.addArc.startAngle(-PIdiv2 + smallRad )
.endAngle(PIdiv2 - smallRad)())
;
d._g.selectAll("text.del, text.add")
.style("visibility", "visible");
toolTip.selectAll("*").remove();
toolTip.append("div").attr("class", "row")
.append("blockquote")
.text(d.message.split("\n")[0]);
toolTip.append("div").attr("class", "row userInfo open").call(function(div) {
div = div.append("div").attr("class", "statInfo");
div.node().appendChild(d.author.avatar);
div.append("ul").call(function(ul) {
(d.author.name || d.author.login) && ul.append("li").call(function(li) {
li.append("h1")
.text((d.author.name || d.author.login))
;
li.append("hr");
});
ul.append("li").call(function(li) {
li.append("span")
.attr("class", "mini-icon mini-icon-time");
li.append("strong")
.text(timeFormat(d.date))
});
d.files.length && ul.append("li")
.call(function(li) {
var stat = d.files.reduce(function(a, b) {
for(var key in TYPE_STATUS_FILE) {
if (TYPE_STATUS_FILE.hasOwnProperty(key)
&& b.status == TYPE_STATUS_FILE[key]) {
a[key] = (a[key] || 0);
a[key]++;
break;
}
}
return a;
}, {});
li = li.append("ul")
.attr("class", "setting");
li = li.append("li").attr("class", "field");
li.append("h1")
.text("Changed files:");
for(var key in stat) {
stat.hasOwnProperty(key)
&& li.append("ul")
.attr("class", "group")
.append("li")
.attr("class", "field")
.append("span")
.text(key + ": ")
.append("strong")
.text(stat[key]);
}
});
ul.append("li")
.call(function(li) {
li = li.append("ul")
.attr("class", "setting");
li = li.append("li").attr("class", "field");
li.append("h1")
.text("Changed lines:");
var stat = {changes : "", additions : " + ", deletions : " - "};
for(var key in stat) {
d.stats.hasOwnProperty(key)
&& li.append("ul")
.attr("class", "group")
.append("li")
.attr("class", "field")
.append("span")
.text(key + ": ")
.append("strong")
.style("color", d3.rgb(colors[key]).darker(.2))
.text(stat[key] + textFormat(d.stats[key]));
}
});
});
});
toolTip.show();
};
vis.mlArc = function(d) {
if (!d._g)
return;
d._g.selectAll("path.bar")
.style({
"fill-opacity" : .3,
"stroke" : null
});
var self = d._g.node();
if (!self)
return;
self.barDel && self.barDel
.transition()
.duration(750)
.ease("elastic")
.attr("d", self.delArc.startAngle(Math.PI - smallRad)
.endAngle(Math.PI + smallRad)())
;
self.barAdd && self.barAdd
.transition()
.duration(750)
.ease("elastic")
.attr("d", self.addArc.startAngle(- smallRad )
.endAngle(smallRad)())
;
d._g.selectAll("text.del")
.style("visibility", d.stats && d.stats.delTextVis ? d.stats.delTextVis : "hidden");
d._g.selectAll("text.add")
.style("visibility", d.stats && d.stats.addTextVis ? d.stats.addTextVis : "hidden");
toolTip.hide();
};
vis.mmArc = function(d) {
vis.mtt(d);
};
vis.clearStat = function() {
if (vis.layers && vis.layers.stat) {
vis.layers.stat.selectAll("*").remove();
vis.layers.stat.cont && (vis.layers.stat.cont = null);
}
};
vis.redrawStat = function(data, layout) {
layout = layout || vis.layers.stat;
var _commits = data && data.commits ? data.commits.values() : null;
if (!_commits || !_commits.length) {
vis.clearStat();
return;
}
var bd = d3.extent(data.dates);
var delta = (bd[1] - bd[0]) * 0.1;
delta = delta || ONE_DAY;
bd = [bd[0] - delta, bd[1] + delta];
var x = d3.time.scale()
.domain(d3.extent(bd))
.range([0, w - margin.left - margin.right])
;
var h6 = h/6;
var y = d3.scale.linear()
.range([2, h6 * 2])
.domain([0, data.stats.changes || 1]);
var sorted = _commits.slice(0).concat([
{ date : bd[0] + delta / 2, f : { d : 0, a : 0, m : 0 } },
{ date : bd[1] - delta / 2, f : { d : 0, a : 0, m : 0 } }
]).sort(vis.sC);
var layers =
[
{
color: colors.deletedFile,
values: sorted.map(function (d) {
return {t : 1, x: d.date, y0 : 0, y: (d.stats ? -d.stats.f.d : 0)}
})
},
{
color: colors.modifiedFile,
values: sorted.map(function (d) {
return {x: d.date, y0 : 0, y: (d.stats ? d.stats.f.m : 0)}
})
},
{
color: colors.addedFile,
values: sorted.map(function (d) {
return {x: d.date, y0: (d.stats ? d.stats.f.m : 0), y : (d.stats ? d.stats.f.a : 0)}
})
}
]
;
function interpolateSankey(points) {
var x0 = points[0][0], y0 = points[0][1], x1, y1, x2,
path = [x0, ",", y0],
i = 0,
n = points.length;
while (++i < n) {
x1 = points[i][0];
y1 = points[i][1];
x2 = (x0 + x1) / 2;
path.push("C", x2, ",", y0, " ", x2, ",", y1, " ", x1, ",", y1);
x0 = x1;
y0 = y1;
}
return path.join("");
}
var y1 = d3.scale.linear()
.range([h6 * 4.5, h6 * 3, h6 * 1.5])
.domain([-data.stats.files, 0, data.stats.files]),
area = d3.svg.area()
.interpolate(interpolateSankey /*true ? "linear" : "basis"*/)
.x(function(d) { return x(d.x); })
.y0(function(d) { return y1(d.y0); })
.y1(function(d) { return y1(d.y0 + d.y); })
;
var xAxis = d3.svg.axis()
.scale(x)
;
layout.cont = layout.cont || layout
.append("g")
.attr("transform", "translate(" + margin.left + "," + margin.top + ")")
;
layout.cont.selectAll("path.areaFile").remove();
layout.cont.selectAll("path.areaFile")
.data(layers)
.enter()
.insert("path", ":first-child")
.attr("class", "areaFile")
.style("fill-opacity",.1)
.style("fill", function(d) { return d.color; })
.transition()
.duration(750)
.attr("d", function(d) { return area(d.values); })
;
layout.cont.axis = layout.cont.axis || layout.cont.append("g")
.attr("class", "x axis")
.attr("transform", "translate(0," + h/2 + ")")
;
layout.cont.axis
.selectAll("*").remove();
layout.cont.axis
.call(xAxis)
.selectAll("text")
.attr("y", 0)
.attr("x", 9)
.attr("dy", ".35em")
.attr("transform", "rotate(90)")
.style("text-anchor", "start")
;
layout.cont.points = layout.cont.points || layout.cont.append("g")
.attr("transform", "translate(0," + h/2 + ")")
;
var cData = layout.cont.points.selectAll("g.com")
.data(_commits, function(d) { return d.sha; });
cData.enter()
.append("g")
.attr("class", "com")
.on("mouseover", vis.meArc)
.on("mouseout", vis.mlArc)
.on("mousemove", vis.mmArc)
;
cData.each(function(d, g) {
g = d._g = d3.select(this);
g.transition()
.duration(1500)
.ease("elastic")
.attr("transform", "translate(" + [ x(d.date), 0] + ")");
this.center = this.center || g.append("circle")
.attr("r", 2)
.style("fill", colors.center)
;
if (!d.stats)
return;
var add = this.addArc = d3.svg.arc()
.innerRadius(1)
.outerRadius(1)
.startAngle(- smallRad )
.endAngle(smallRad)
;
var del = this.delArc = d3.svg.arc()
.innerRadius(1)
.outerRadius(1)
.startAngle(Math.PI - smallRad)
.endAngle(Math.PI + smallRad)
;
(this.barAdd || (
this.barAdd = g.append("path")
.attr("class", "bar")
.style({
"fill" : colors.additions,
"fill-opacity" : .3
})
.attr("d", add())
))
.transition()
.duration(750)
.ease("elastic")
.attr("d", add.outerRadius(y(d.stats.additions))())
;
(this.barDel || (
this.barDel = g.append("path")
.attr("class", "bar")
.style({
"fill" : colors.deletions,
"fill-opacity" : .3
})
.attr("d", del())
))
.transition()
.duration(750)
.ease("elastic")
.attr("d", del.outerRadius(y(d.stats.deletions))())
;
function sized(s, k) {
return s * (1 + k);
}
var addTop = this.addArcTop = d3.svg.arc()
.innerRadius(sized(y(d.stats.additions), .015))
.outerRadius(sized(y(d.stats.additions), .025))
.startAngle(- smallRad )
.endAngle(smallRad)
;
var delTop = this.delArcTop = d3.svg.arc()
.innerRadius(sized(y(d.stats.deletions), .015))
.outerRadius(sized(y(d.stats.deletions), .025))
.startAngle(Math.PI - smallRad)
.endAngle(Math.PI + smallRad)
;
(this.barAddTop || (
this.barAddTop = g.append("path")
.style("fill", toRgba(colors.additions))
))
.attr("d", addTop())
;
(this.barDelTop || (
this.barDelTop = g.append("path")
.style("fill", toRgba(colors.deletions))
))
.attr("d", delTop())
;
if (d.stats.additions) {
d.stats.addTextVis = sized(y(d.stats.additions), .029) > h6 / 2 ? "visible" : "hidden";
(this.labelAdd || (
this.labelAdd = g.append("text")
.attr("class", "add")
.attr("dy", "-.31em")
.attr("text-anchor", "middle")
.style("fill", colors.additions)
.text(" + " + textFormat(d.stats.additions))
))
.attr("transform", "translate(" + [0, -sized(y(d.stats.additions), .027)] + ")")
.attr("visibility", d.stats.addTextVis)
;
}
if (d.stats.deletions) {
d.stats.delTextVis = sized(y(d.stats.deletions), .029) > h6 / 2 ? "visible" : "hidden";
(this.labelDel || (
this.labelDel = g.append("text")
.attr("class", "del")
.attr("dy", ".93em")
.attr("text-anchor", "middle")
.style("fill", colors.deletions)
.text(" - " + textFormat(d.stats.deletions))
))
.attr("transform", "translate(" + [0, sized(y(d.stats.deletions), .027)] + ")")
//rotate(180)
.attr("visibility", d.stats.delTextVis)
;
}
});
cData.sort(function(a, b) {
return a.stats && b.stats ? d3.ascending(b.stats.changes, a.stats.changes) : 0;
});
cData.exit().remove();
};
})(vis || (vis = {}));
/* html5doctor.com Reset v1.6.1 - http://cssreset.com */
html,body,div,span,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,abbr,address,cite,code,del,dfn,em,img,ins,kbd,q,samp,small,strong,sub,sup,var,b,i,dl,dt,dd,ol,ul,li,fieldset,form,label,legend,table,caption,tbody,tfoot,thead,tr,th,td,article,aside,canvas,details,figcaption,figure,footer,header,hgroup,menu,nav,section,summary,time,mark,audio,video{margin:0;padding:0;border:0;outline:0;font-size:100%;vertical-align:baseline;background:transparent}body{line-height:1}article,aside,details,figcaption,figure,footer,header,hgroup,menu,nav,section{display:block}nav ul{list-style:none}blockquote,q{quotes:none}blockquote:before,blockquote:after,q:before,q:after{content:none}a{margin:0;padding:0;font-size:100%;vertical-align:baseline;background:transparent}ins{background-color:#ff9;color:#000;text-decoration:none}mark{background-color:#ff9;color:#000;font-style:italic;font-weight:bold}del{text-decoration:line-through}abbr[title],dfn[title]{border-bottom:1px dotted;cursor:help}table{border-collapse:collapse;border-spacing:0}hr{display:block;height:1px;border:0;border-top:1px solid #ccc;margin:1em 0;padding:0}input,select{vertical-align:middle}
/** взято с github.com */
@font-face {
font-family: 'Octicons Regular';
src: url(github_webfont.eot);
src: url(github_webfont_ie.eot) format("embedded-opentype"),
url(github_webfont.woff) format("woff"),
url(github_webfont.ttf) format("truetype"),
url(github_webfont.svg) format("svg");
font-weight: normal;
font-style: normal
}
.mini-icon, .mega-icon {
font-family: 'Octicons Regular';
font-weight: normal;
font-style: normal;
display: inline-block;
text-decoration: inherit;
line-height: 1;
-webkit-font-smoothing: antialiased;
line-height: 1;
text-decoration: none;
}
.mini-icon {
font-size: 16px;
width: 16px;
height: 16px;
}
.mega-icon {
font-size: 32px;
width: 32px;
height: 32px;
}
.mini-icon-octocat:before {
content: "\f008";
}
.mini-icon-public-repo:before {
content: "\f001";
}
.mini-icon-time:before {
content: "\f046";
}
.mini-icon-link:before {
content: "\f05c";
}
.mini-icon-location:before {
content: "\f060";
}
.mini-icon-history:before {
content: "\f07e";
}
.mega-icon-public-repo:before {
content: "\f201";
}
/** end */
html,
body {
width: 100%;
height: 100%;
overflow: hidden;
}
body {
background: url('');
}
label {
color: #f9f9f9;
}
.popup {
position: absolute;
overflow: hidden;
z-index: 10;
}
#controls {
left: -1px;
top: -1px;
z-index: 11;
padding: 5px;
vertical-align: middle;
border-radius: 0 0 3px 0;
width: 100%;
}
#controls input,
#controls button,
#controls label {
display: inline-block;
}
input[type=text],
input[type=number] {
border: 1px solid #b9b9b9;
display: block;
padding: 5px;
font-size: 14px;
margin: 0;
outline: 0;
}
input[type=text]:hover,
input[type=number]:hover,
input[type=text]:focus,
input[type=number]:focus {
border-color: #000;
-webkit-box-shadow: 0px 0px 3px rgba(0, 0, 0, 0.3);
-moz-box-shadow: 0px 0px 3px rgba(0, 0, 0, 0.3);
-o-box-shadow: 0px 0px 3px rgba(0, 0, 0, 0.3);
-ms-box-shadow: 0px 0px 3px rgba(0, 0, 0, 0.3);
box-shadow: 0px 0px 3px rgba(0, 0, 0, 0.3);
}
#controls input {
font-family: sans-serif;
font-weight: bolder;
}
#console {
bottom: 0;
left: 0;
z-index: 100;
}
/* progress bar */
.meter {
overflow: hidden;
position: relative;
background: #555;
padding: 10px;
-webkit-box-shadow: inset 0 -1px 1px rgba(255, 255, 255, 0.3);
-moz-box-shadow: inset 0 -1px 1px rgba(255, 255, 255, 0.3);
box-shadow: inset 0 -1px 1px rgba(255, 255, 255, 0.3);
}
.meter > span {
display: block;
background-color: rgb(43, 194, 83);
/*background-image: -webkit-gradient(linear, left bottom, left top, color-stop(0, rgb(43, 194, 83)), color-stop(1, rgb(84, 240, 84)));
background-image: -moz-linear-gradient(center bottom, rgb(43, 194, 83) 37%, rgb(84, 240, 84) 69%);*/
-webkit-box-shadow: inset 0 2px 9px rgba(255, 255, 255, 0.3), inset 0 -2px 6px rgba(0, 0, 0, 0.4);
-moz-box-shadow: inset 0 2px 9px rgba(255, 255, 255, 0.3), inset 0 -2px 6px rgba(0, 0, 0, 0.4);
box-shadow: inset 0 2px 9px rgba(255, 255, 255, 0.3), inset 0 -2px 6px rgba(0, 0, 0, 0.4);
position: relative;
overflow: hidden;
min-height: 10px
}
.orange > span {
background-color: #f1a165;
background: rgb(241,161,101);
/*background: url();
background: -moz-linear-gradient(top, rgba(241,161,101,1) 0%, rgba(243,109,10,1) 100%);
background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,rgba(241,161,101,1)), color-stop(100%,rgba(243,109,10,1)));
background: -webkit-linear-gradient(top, rgba(241,161,101,1) 0%,rgba(243,109,10,1) 100%);
background: -o-linear-gradient(top, rgba(241,161,101,1) 0%,rgba(243,109,10,1) 100%);
background: -ms-linear-gradient(top, rgba(241,161,101,1) 0%,rgba(243,109,10,1) 100%);
background: linear-gradient(to bottom, rgba(241,161,101,1) 0%,rgba(243,109,10,1) 100%);
filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#f1a165', endColorstr='#f36d0a',GradientType=0 );*/
}
.red > span {
background-color: #f0a3a3;
/*background-image: -moz-linear-gradient(top, #f0a3a3, #f42323);
background-image: -webkit-gradient(linear, left top, left bottom, color-stop(0, #f0a3a3), color-stop(1, #f42323));
background-image: -webkit-linear-gradient(#f0a3a3, #f42323);*/
}
.progressBar {
font-size: 11px;
width: 100%;
}
.progressBar .meter {
margin: 0;
padding: 1px;
}
/*.progressBar .meter > span*/
#progressBar {
text-align: right;
vertical-align: middle;
color: white;
text-shadow: 0 1px 1px black;
word-wrap: normal;
white-space: nowrap;
text-overflow: ellipsis;
}
/* end progress bar */
#statusBar {
bottom: 0;
left: 0;
z-index: 12;
width: 100%;
position: fixed;
border: 0;
border-top: 1px solid #CACACA;
background: rgb(239,239,239);
/*background: url();
background: -moz-linear-gradient(top, rgba(239,239,239,1) 0%, rgba(221,221,221,1) 41%, rgba(202,202,202,1) 100%);
background: -webkit-gradient(linear, left top, left bottom, color-stop(0%,rgba(239,239,239,1)), color-stop(41%,rgba(221,221,221,1)), color-stop(100%,rgba(202,202,202,1)));
background: -webkit-linear-gradient(top, rgba(239,239,239,1) 0%,rgba(221,221,221,1) 41%,rgba(202,202,202,1) 100%);
background: -o-linear-gradient(top, rgba(239,239,239,1) 0%,rgba(221,221,221,1) 41%,rgba(202,202,202,1) 100%);
background: -ms-linear-gradient(top, rgba(239,239,239,1) 0%,rgba(221,221,221,1) 41%,rgba(202,202,202,1) 100%);
background: linear-gradient(to bottom, rgba(239,239,239,1) 0%,rgba(221,221,221,1) 41%,rgba(202,202,202,1) 100%);
filter: progid:DXImageTransform.Microsoft.gradient( startColorstr='#efefef', endColorstr='#cacaca',GradientType=0 );*/
}
#statusBar ul {
padding: 2px 3px;
overflow: hidden;
margin: 0;
list-style: none;
}
#statusBar li {
display: block;
vertical-align: middle;
float: left;
margin-right: 2px;
border: 0;
border-right: 1px solid #444;
}
#statusBar li:last-child {
margin-right: 0;
border-right: 0;
}
#layers,
#canvas,
#svg,
canvas,
svg {
position: relative;
width: 100%;
height: 100%;
}
#svg, #canvas {
position: absolute;
top: 0;
left : 0;
}
#svg {
z-index: 1;
}
#canvas {
z-index: 0;
}
#layers {
background: rgba(224, 224, 224, .1);
}
.axis text {
font: 10px sans-serif;
fill: #ccc;
}
.axis line,
.axis path {
fill: none;
stroke: #ccc;
shape-rendering: crispEdges;
}
#ldrCont {
z-index: 100;
top: 0;
left: 0;
text-align: center;
width: 100%;
height: 100%;
}
#ldrCont > img {
display: inline-block;
position:relative;
top:49%;
width: 48px;
height: auto;
}
.lineFiles {
fill: none;
shape-rendering: crispEdges;
stroke-width: 2px;
}
.lineFiles .M {
stroke: rgba(255, 184, 119, .5);
}
.lineFiles .A {
stroke: rgba(165, 236, 110, .5);
}
.lineFiles .D {
stroke: rgba(255, 119, 255, .5);
}
.cRepo {
cursor: pointer;
}
.cRepo text {
text-shadow: 0 1px 2px rgba(0, 0, 0, 0.5);
pointer-events: none;
}
.cRepo:hover text {
text-shadow: 0 1px 2px rgba(255, 255, 255, 1);
}
.tooltip {
color: #f9f9f9;
text-shadow: 0 1px 2px rgba(0, 0, 0, 0.5);
background: rgba(110, 130, 163, 0.7);
border: 1px dotted #8D9DB6;
padding: 5px;
border-radius: 3px;
}
#tooltip {
z-index: 50;
display: none;
max-width: 25%;
pointer-events: none;
}
/* http://xbreaker.github.com/plusstrap/base-css.html#buttons */
.btn {
display: inline-block;
*display: inline;
*zoom: 1;
padding: 0px 8px;
font-weight: bold;
margin-bottom: 0;
font-size: 11px;
line-height: 22px;
*line-height: 22px;
text-align: center;
vertical-align: middle;
cursor: default;
color: #333333;
text-shadow: 0 1px 1px rgba(255, 255, 255, 0.75);
background-color: #f3f3f3;
background-image: -moz-linear-gradient(top, #f5f5f5, #f1f1f1);
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#f5f5f5), to(#f1f1f1));
background-image: -webkit-linear-gradient(top, #f5f5f5, #f1f1f1);
background-image: -o-linear-gradient(top, #f5f5f5, #f1f1f1);
background-image: linear-gradient(to bottom, #f5f5f5, #f1f1f1);
background-repeat: repeat-x;
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff5f5f5', endColorstr='#fff1f1f1', GradientType=0);
border-color: #e4e4e4;
*background-color: #f1f1f1;
filter: progid:DXImageTransform.Microsoft.gradient(enabled = false);
border: 1px solid rgba(0, 0, 0, 0.1);
*border: 0;
border-bottom-color: rgba(0, 0, 0, 0.1);
-webkit-border-radius: 2px;
-moz-border-radius: 2px;
border-radius: 2px;
*margin-left: .3em;
}
.btn:not(.active):hover {
background-color: #f5f5f5;
background-image: -moz-linear-gradient(top, #f9f9f9, #eeeeee);
background-image: -webkit-gradient(linear, 0 0, 0 100%, from(#f9f9f9), to(#eeeeee));
background-image: -webkit-linear-gradient(top, #f9f9f9, #eeeeee);
background-image: -o-linear-gradient(top, #f9f9f9, #eeeeee);
background-image: linear-gradient(to bottom, #f9f9f9, #eeeeee);
background-repeat: repeat-x;
filter: progid:DXImageTransform.Microsoft.gradient(startColorstr='#fff9f9f9', endColorstr='#ffeeeeee', GradientType=0);
}
.btn:hover,
.btn:active,
.btn.active,
.btn.disabled,
.btn[disabled] {
color: #333333;
background-color: #eeeeee;
*background-color: #e4e4e4;
-webkit-box-shadow: 0 1px 0px #dedede;
-moz-box-shadow: 0 1px 0px #dedede;
box-shadow: 0 1px 0px #dedede;
border-color: #cbcbcb;
}
.btn:active,
.btn.active {
background-color: #d8d8d8 \9;
-webkit-box-shadow: inset 0 1px 2px rgba(0, 0, 0, .2);
-moz-box-shadow: inset 0 1px 2px rgba(0, 0, 0, .2);
box-shadow: inset 0 1px 2px rgba(0, 0, 0, .2);
}
.btn:first-child {
*margin-left: 0;
}
.btn:hover {
color: #333333;
text-decoration: none;
background-color: #e6e6e6;
*background-color: #d9d9d9;
-webkit-transition: background-position 0.1s linear;
-moz-transition: background-position 0.1s linear;
-o-transition: background-position 0.1s linear;
transition: background-position 0.1s linear;
}
.btn:focus {
outline: thin dotted #333;
outline: 5px auto -webkit-focus-ring-color;
outline-offset: -2px;
}
.btn.active,
.btn:active {
background-color: #e6e6e6;
background-color: #d9d9d9 \9;
background-image: none;
outline: 0;
}
.btn.disabled,
.btn[disabled] {
cursor: default;
background-color: #e6e6e6;
background-image: none;
opacity: 0.65;
filter: alpha(opacity=65);
-webkit-box-shadow: none;
-moz-box-shadow: none;
box-shadow: none;
}
/** end **/
#controls .btn {
padding: 2px 8px;
margin: 0;
}
.row {
display: block;
max-width: 100%;
}
#divStat img,
.statInfo img {
float: left;
margin: 0 0 0 0;
padding: 1px;
background: #fff;
box-shadow: 0 1px 2px rgba(0,0,0,0.15);
}
#divStat ul,
.statInfo ul {
overflow: hidden;
padding: 0;
list-style: none;
}
#divStat li,
.statInfo li {
color: #ccc;
}
hr {
margin: 5px 0;
}
#divStat li strong,
#divStat li a,
.statInfo li strong,
.statInfo li a {
margin-left: 5px;
}
#divStat li strong,
#divStat h1,
.statInfo li strong,
.statInfo h1 {
color: #f9f9f9;
}
#curRep .a-icon,
#divStat .a-icon,
.statInfo .a-icon {
text-decoration: none;
color: rgba(172, 213, 229, 1);
}
#divStat a,
.statInfo a {
color: rgba(172, 213, 229, 1);
}
#curRep .a-icon:hover,
#divStat .a-icon:hover,
#divStat a:hover,
.statInfo .a-icon:hover,
.statInfo a:hover {
color: rgba(221, 246, 255, 1);
}
#curRep {
vertical-align: middle;
text-align: left;
}
#curRep .mega-icon {
font-size: 24px;
width: 24px;
height: inherit;
position: relative;
}
#curRep .mini-icon-link:before,
#curRep .mega-icon-public-repo:before {
position: absolute;
top: -16px;
}
#userInfo,
.userInfo {
right: -2px;
top: -2px;
padding: 7px 7px 5px 5px;
z-index: 52;
}
#userInfo ul,
.userInfo ul {
white-space: nowrap;
-webkit-transition: all .4s;
-moz-transition: all .4s;
-o-transition: all .4s;
transition: all .4s;
max-width: 0;
max-height: 0;
}
#userInfo img,
.userInfo img {
-webkit-transition: all .5s ease-in-out .4s;
-moz-transition: all .5s ease-in-out .4s;
-o-transition: all .5s ease-in-out .4s;
transition: all .5s ease-in-out .4s;
width : 48px;
height : auto;
}
#userInfo.open img,
#userInfo:hover img,
.userInfo.open img,
.userInfo:hover img {
-webkit-transition: all .5s;
-moz-transition: all .5s;
-o-transition: all .5s;
transition: all .5s;
margin-right: 10px;
width : 96px;
height : auto;
}
#userInfo.open ul,
#userInfo:hover ul,
.userInfo.open ul,
.userInfo:hover ul {
-webkit-transition: all .5s ease-in-out .5s;
-moz-transition: all .5s ease-in-out .5s;
-o-transition: all .5s ease-in-out .5s;
transition: all .5s ease-in-out .5s;
max-width: 960px;
max-height: 1280px;
}
/** stepbar */
.steps {
list-style: none;
margin: 0;
}
.steps > li {
display: block;
float: left;
position: relative;
padding: 4px;
border: 1px solid white;
border-right-style: dotted;
border-left-style: dotted;
margin-right: 1em;
background: #50617F;
padding-right: 2px;
margin-left: -17px;
padding-left: 24px;
/*text-align: right;*/
box-shadow: 2px 4px 5px rgba(0, 0, 0, .5);
}
.steps > li .sub {
text-align: left;
display: block;
overflow: hidden;
margin-left: -22px;
}
.steps > li:first-child .sub {
margin-left: 0;
}
.steps > li .sub {
max-height: 0;
max-width: 0;
-webkit-transition: max-height .7s ease-in-out .2s, max-width .2s ease-in-out .9s;
-moz-transition: max-height .7s ease-in-out .2s, max-width .2s ease-in-out .9s;
-o-transition: max-height .7s ease-in-out .2s, max-width .2s ease-in-out .9s;
transition: max-height .7s ease-in-out .2s, max-width .2s ease-in-out .9s;
}
.steps > li:hover .sub {
max-height: 960px;
max-width: 960px;
-webkit-transition: max-height .5s ease-in-out .3s, max-width .2s ease-in-out .1s;
-moz-transition: max-height .5s ease-in-out .3s, max-width .2s ease-in-out .1s;
-o-transition: max-height .5s ease-in-out .3s, max-width .2s ease-in-out .1s;
transition: max-height .5s ease-in-out .3s, max-width .2s ease-in-out .1s;
}
.steps > li:first-child {
border-radius: 5px 0 0 5px;
border-left: 1px solid #ffffff;
margin-left: 0;
padding-left: 4px;
}
.steps > li:hover {
border-right-style: solid;
border-left-style: solid;
}
.steps > li:after,
.steps > li:before {
width: 0;
height: 0;
content: "";
display: inline-block;
position: absolute;
}
.steps > li:after {
border: 1.248em solid transparent;
border-left-color: #ffffff;
top: -0.05em;
right: -2.45em;
z-index: 0;
}
.steps > li:before {
border: 1.13em solid transparent;
border-left-color: #50617F;
top: .02em;
right: -2.25em;
z-index: 1;
}
.steps.sfirst > li:not(.first) {
border-color: #ccc;
background: #858689;
}
.steps.sfirst > li:not(.first):after {
border-left-color: #ccc;
}
.steps.sfirst > li:not(.first):before {
border-left-color: #858689;
}
.steps.ssecond > li.third {
border-color: #ccc;
background: #858689;
}
.steps.ssecond > li.third:after {
border-left-color: #ccc;
}
.steps.ssecond > li.third:before {
border-left-color: #858689;
}
.steps > li > label {
vertical-align: middle;
}
.sub > div {
padding: 5px;
}
.sub blockquote {
padding: 3px;
margin: 2px 0;
background: rgba(255, 252, 196, .6);
border-left: 2px solid rgba(255, 252, 196, 1);
font-style: italic;
color: rgba(255, 252, 196, 1);
text-shadow: 0 1px 2px rgba(0, 0, 0, .5);
}
blockquote.green {
/*background: rgba(161, 237, 144, .3);
color: rgb(198, 244, 188);
border-left: 2px solid rgb(198, 244, 188);*/
background: rgba(198, 245, 188, .6);
color: rgb(31, 166, 31);
border-left: 2px solid rgb(31, 166, 31);
}
/** end stepbar */
.barLang circle,
.barLang path {
}
.barLang path {
fill: none;
stroke-width: 1px;
}
.barLang text {
pointer-events: none;
text-shadow : 0 0 3px rgba(0, 0, 0, 1);
}
.barLang .barSelect {
stroke: #000;
stroke-opacity: 0;
stroke-width: 1px;
fill: #f9f9f9;
fill-opacity: 0;
-webkit-transition: all 1.5s;
-moz-transition: all 1.5s;
-o-transition: all 1.5s;
transition: all 1.5s;
}
.barLang:hover .barSelect {
fill-opacity: .6;
stroke-opacity: .6;
-webkit-transition: all .5s;
-moz-transition: all .5s;
-o-transition: all .5s;
transition: all .5s;
}
.field {
padding: 3px;
}
.setting {
list-style: none;
text-shadow: 0 1px 2px rgba(0, 0, 0, .5);
}
.setting input {
padding: 0;
margin: 0;
}
.setting h1 {
color: rgba(0, 0, 0, .8);
text-shadow: 0 1px 1px rgba(255, 255, 255, .5);
}
.setting > li {
border: 1px dotted rgba(255, 255, 255, .5);
background: rgba(255, 255, 255, .2);
}
.setting > li .group {
list-style: none;
vertical-align: middle;
}
.group > li {
display: inline-block;
}
.com-mess {
font-size: 11pt;
font-family: Verdana,serif;
fill: #ffffff;
fill-opacity: .3;
}
.mono {
font-family: monospace;
font-size: 10pt;
}
#visBtnResume .mini-icon-history {
position: relative;
height: 0;
}
#visBtnResume .mini-icon-history:before {
position: absolute;
top: -12px;
left: 0;
}
.gtLeg, .gttLeg {
fill : #f9f9f9;
text-shadow: 0 1px 2px rgba(0, 0, 0, .5);
}
/**
* User: ArtZub
* Date: 15.01.13
* Time: 1:34
*/
'use strict';
var PIdiv2 = Math.PI / 2,
smallRad = PIdiv2 * (1 / 15),
TT_PAD = 8
;
var colors = {
additions : "#ddffdd",
deletions : "#ffdddd",
changes : "#FFF1DD",
center : "#FF6600",
stroked : "#f9f9f9",
addedFile : "#A5EC6E",
modifiedFile : "#FFB877",
deletedFile : "#FF77B5",
decolor : "#888888"
};
function toRgba(color, a) {
color = d3.rgb(color);
return "rgba(" + [color.r, color.g, color.b, a || 1] + ")";
}
var vis = {
sC : function(a, b) {
return d3.ascending(b.date, a.date);
},
mtt : function(d) {
if (toolTip) {
var e = d3.event;
if (arguments.length > 3)
e = arguments[3];
toolTip
.style("top", e.pageY > h / 2 ? (e.pageY - toolTip.node().clientHeight - TT_PAD) + "px" : (e.pageY + TT_PAD) + "px")
.style("left", e.pageX > w / 2 ? (e.pageX - toolTip.node().clientWidth - TT_PAD) + "px" : (e.pageX + TT_PAD) + "px")
;
}
}
};
function initGraphics(svg) {
vis.layers = (function(data) {
var ls = { _data : data };
svg.selectAll("g.layer")
.data(data, function(d) {return d.name})
.enter()
.append("g")
.each(function(d) {
ls[d.name] = d3.select(this).attr("class", "layer").attr("width", w).attr("height", h);
ls[d.name].getOrder = function() {
return d.order;
return this;
};
ls[d.name].toFront = function() {
d.order &&
vis.layers.ordering(this, 0);
return this;
};
ls[d.name].hide = function() {
ls[d.name].visible = false;
this.style("display", "none");
return this;
};
ls[d.name].show = function() {
ls[d.name].visible = true;
this.style("display", null);
return this;
};
ls[d.name].visible = true;
});
return ls;
})([
{name : "repo", order : 2},
{name : "stat", order : 1},
{name : "show", order : 0}
]);
vis.layers.ordering = function(layer, order) {
function s(a, b) {
return d3.ascending(b.order, a.order);
}
var _d = (layer instanceof Array || layer instanceof Object ? layer : this[layer]);
_d = _d ? (_d instanceof Array ? _d.datum() : _d) : null;
if (_d) {
this._data.forEach(function(d) {
_d != d && d.order >= order && d.order++;
});
_d.order = order;
svg.selectAll("g.layer").sort(s);
}
return this;
};
vis.resources = {
};
vis.inited = true;
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment