Skip to content

Instantly share code, notes, and snippets.

@grybnicky
Last active March 24, 2016 16:12
Show Gist options
  • Save grybnicky/ed93d8d6810099269df9 to your computer and use it in GitHub Desktop.
Save grybnicky/ed93d8d6810099269df9 to your computer and use it in GitHub Desktop.
genome browser with GC tool tips and better json
[{"phagename": "L5",
"genomelength": 5000,
"genes": [{
"name": 1,
"start": 124,
"stop": 1410,
"direction": "forward",
"product": "dnaA",
"subnum": "13",
"sequence": "ATAGTCGATGCTAGTATGCGCATCCGGCGCGATCGTGCGTGCTAGC",
"translation": "amino acids"
},
{
"name": 2,
"start": 1413,
"stop": 2522,
"direction": "forward",
"product": "DNA Polymerase III",
"sequence": "ATAGTCGATGCTAGTATGCGCGTGCGTGCTAGC",
"subnum": "123",
"translation": "amino acids"
},
{
"name": 3,
"start": 2617,
"stop": 5070,
"direction": "forward",
"product": "DNA gyrase",
"sequence": "ATAGTCGATGCGTGCTAGC",
"subnum": "76",
"translation": "amino acids"
},
{
"name": 4,
"start": 5189,
"stop": 5917,
"direction": "reverse",
"product": "Transcriptional regulator",
"sequence": "ATAGTCGATCGCGGCGGCATGCATGCATGCGCGTGCTAGC",
"subnum": "654",
"translation": "amino acids"
},
{
"name": 5,
"start": 6100,
"stop": 7527,
"direction": "reverse",
"product": "probable amidase",
"sequence": "ATAGTCGATGCTAGTATGCATCGTACGATCCGAGCGCGCGATCGTGCGTGCTAGC",
"subnum": "82",
"translation": "amino acids"
},
{
"name": 6,
"start": 7566,
"stop": 8537,
"direction": "forward",
"product": "Tricarboxylate transport protein TctC",
"sequence": "ATAGTCGATGCTAGTATGCGCGCGCTATCGCGGCGCGCGCGCGCGGATCGTGCGTGCTAGC",
"subnum": "41",
"translation": "amino acids"
},
{
"name": 7,
"start": 8956,
"stop": 10119,
"direction": "forward",
"product": "L-lactate dehydrogenase",
"sequence": "ATAGTCGATGCTAGTATGCGCGCGCCGCGGCGCGGGGCGCGCGATCGTGCGTGCTAGC",
"subnum": "38",
"translation": "amino acids"
}]
}]
<!DOCTYPE html>
<head>
<meta charset="utf-8">
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/3.5.5/d3.min.js"></script>
<script src="//d3js.org/d3.v3.min.js" language="JavaScript"></script>
<script src="http://labratrevenge.com/d3-tip/javascripts/d3.tip.v0.6.3.js"></script>
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Roboto+Slab:400,100,300,700">
<link rel="stylesheet" type="text/css" href="http://fonts.googleapis.com/css?family=Roboto:400,100,100italic,300,300italic,400italic,500,500italic,700,700italic,900,900italic">
<style>
body {
margin: ;
position:fixed;
top:0;
right:0;
bottom:0;
left:2;
}
/* Tip box */
.d3-tip {
line-height: 1;
font-weight: bold;
padding: 11px;
background: rgba(0, 0, 0, 0.8);
color: #ff0505;
border-radius: 2px;
}
/* This is for the little triangle for the tooltip */
.d3-tip:after {
box-sizing: border-box;
display: inline;
font-size: 10px;
width: 100%;
line-height: 1;
color: rgba(0, 0, 0, 0.8);
content: "\25BC";
position: absolute;
text-align: center;
}
/* Style northward tooltips differently */
.d3-tip.n:after {
margin: -1px 0 0 0;
top: 100%;
left: 0;
}
/* Lower tip box */
.d3-hidden {
line-height: 1;
font-weight: bold;
padding: 11px;
background: rgba(1, 34, 34, 0.8);
color: #1fe429;
border-radius: 2px;
}
/* This is for the little triangle for the tooltip */
.d3-hidden:after {
box-sizing: border-box;
display: inline;
font-size: 20px;
width: 100%;
line-height: 1;
color: rgba(1, 0, 0, 0.8);
content: "\25BC";
position: absolute;
text-align: center;
}
/* Style northward tooltips differently */
.d3-hidden.n:after {
margin: -60px 0 0 0;
top: 100%;
left: 0;
}
a {
cursor: pointer;
text-decoration: underline;
color: black;
}
#download {
border: 1px solid silver;
position: absolute;
opacity: 0;
}
</style>
</head>
<body>
<script>
// Feel free to change or delete any of the code you see!
var genomelength = 7800;
var tickMarks = {thousand: [], fivehundred: [], onehundred: []};
var genome_positions = [];
for (var i = 1; i <= genomelength; i++) {
genome_positions.push(i);
}
genome_positions.forEach(function(currentValue, index, myArray){
if (currentValue % 1000 === 0) {
tickMarks.thousand.push(currentValue);
}
else if (currentValue % 500 === 0) {
tickMarks.fivehundred.push(currentValue);
}
else if (currentValue % 100 === 0) {
tickMarks.onehundred.push(currentValue);
}
});
var svg = d3.select("body").append("svg").attr({height: 500,width: genomelength/10});
svg.append("rect")
.attr({x: 0, y: 100, width: genomelength/10, height: 30})
.style({"stroke-width": "2px", "fill": "white", "stroke": "black"})
.attr("stroke-opacity", 0)
.transition().duration(1000)
.attr("stroke-opacity", 1);
d3.json("genes.json.txt", function(error, json) {
if (error) return console.warn(error);
var genes = svg.selectAll(".genes")
.data(json)
.enter()
.append("g");
//Hover box when you mouseover genes
var tip = d3.tip()
.attr('class', 'd3-tip')
.offset(function (d) {
if (d.direction == "forward") {
return [-20, -8];
}
else if (d.direction == "reverse") {
return [-20, -8];
}
})
.html(function(d) {
return "<b>Product:</b> <span style='color:white'>"
+ d.product + "</span>";
})
//Clickable tip box attr
var hidden = d3.tip()
.attr('class', 'd3-hidden')
.offset(function (d) {
if (d.direction == "forward") {
return [80, -8];
}
else if (d.direction == "reverse") {
return [80, -8];
}
})
.html(function(d) {
return "<a style=color:white target=_blank href=http://blast.ncbi.nlm.nih.gov/Blast.cgi>BLAST:</a><span style='color:white'>"
+ " " + d.translation + "</span>";
})
//call the "tool-tips"
svg.call(tip);
svg.call(hidden);
genes.append("rect")
.on('mouseover', tip.show).on('mouseout', tip.hide)
.on('click', hidden.show).on('dblclick', hidden.hide)
.attr("y", function (d) {
if (d.direction == "forward") {
if (d.name % 2 === 0) {
return 70;
}
else { return 40;}
}
else if (d.direction == "reverse") {
if (d.name % 2 === 0) {
return 160;
}
else { return 130; }
}
})
.attr("x", function (d) {
if (d.direction === "forward") {
return (0 - ((d.stop-d.start)/10)) - 2;
}
else if (d.direction === "reverse") {
return (genomelength/10) + 2;
}
})
.attr("height", function (d) {return 30;})
.attr("width", function (d) { return (d.stop-d.start)/10; })
.style({"stroke":"black", "stroke-width": "2px"})
.attr("fill", function (d) {
if (d.direction == "forward") {
return "green";
}
else if (d.direction == "reverse") {
return "red";
}
else {
return "black";
}
})
.transition().delay(1000).duration(1500).attr("x", function (d) { return d.start/10; });
genes.append("text") // GC
.attr("x", function(d) { return ((d.start + d.stop)/2)/10;})
.attr("y", function (d) {
if (d.direction == "forward") {
if (d.name % 2 === 0) {
return 90;
}
else { return 60;}
}
else if (d.direction == "reverse") {
if (d.name % 2 === 0) {
return 180;
}
else { return 150;}
}
})
.style({"text-anchor": "middle", "fill": "black"})
.attr("font-family", "sans-serif")
.text(function(d) {
return ((((d.sequence.match(/G/gi).length)+(d.sequence.match(/C/gi).length))/(d.sequence.length))*100).toPrecision(3)+"%";
})
.attr("fill-opacity", 0)
.transition().delay(2000).duration(1500)
.attr("fill-opacity", 1);
genes.append("text") // pham name
.attr("x", function(d) {
return ((d.start + d.stop)/2)/10;
})
.attr("y", function (d) {
if (d.direction == "forward") {
if (d.name % 2 === 0) {
return 60;
}
else { return 30;}
}
else if (d.direction == "reverse") {
if (d.name % 2 === 0) {
return 210;
}
else { return 180;}
}
})
.style({"text-anchor": "middle", "fill": "black"})
.attr("font-family", "sans-serif")
.text(function(d) {
return d.pham
})
.attr("fill-opacity", 0)
.transition().delay(3500).duration(1500)
.attr("fill-opacity", 1);
});
var group = svg.selectAll(".thousandticks")
.data(tickMarks.thousand)
.enter()
.append("g");
group.append("rect")
.style({"fill": "black"})
//.attr({x: 0, y: 100, width: "1px", height: 30})
.attr({x: function (d) { return d/10; }, y: 100, width: "1px", height: 30})
.attr({"fill-opacity": 0})
.transition().duration(1500)
.attr({"fill-opacity": 1});
//.attr("transform", function (d) { return "translate(" + d/10 + ",0)"; });
group.append("text")
.attr("x", function(d) {return (d/10) + 3;})
.attr("y", (90) + 22)
.attr("font-family", "sans-serif")
.attr("font-size", "14px")
.attr("fill", "green")
.style("text-anchor", "start")
.text(function(d) { return d/1000; })
.attr({"fill-opacity": 0})
.transition().duration(1500)
.attr({"fill-opacity": 1});
var group2 = svg.selectAll(".fivehundredticks")
.data(tickMarks.fivehundred)
.enter()
.append("g");
group2.append("rect")
.style({"fill": "black"})
.attr({x: function(d) {return d/10;}, y: 100, width: "1px", height: 15})
.attr({"fill-opacity": 0})
.transition().duration(1500)
.attr({"fill-opacity": 1});
//.attr("transform", function (d) { return "translate(" + d/10 + ",0)"; });
var group3 = svg.selectAll(".onehundredticks")
.data(tickMarks.onehundred)
.enter()
.append("g");
group3.append("rect")
.style({"fill": "black"})
.attr({x: function (d) { return d/10; }, y: 115, width: "1px", height: 15})
.attr("fill-opacity", 0)
.transition().duration(1500)
.attr("fill-opacity", 1);
//.attr("transform", function (d) { return "translate(" + d/10 + ",0)"; });
</script>
</body>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment