Skip to content

Instantly share code, notes, and snippets.

@nyem69
Last active July 7, 2022 16:14
Show Gist options
  • Star 0 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save nyem69/29896289104af5b5a5e27a61041756a9 to your computer and use it in GitHub Desktop.
Save nyem69/29896289104af5b5a5e27a61041756a9 to your computer and use it in GitHub Desktop.
multi sources
license: mit

d3.js table

  • load from selectable sources
  • select & sort columns
{
"rows": [
{
"alpha3": "BRN",
"name": "Brunei",
"alpha2": "BN",
"numeric": 99
},
{
"alpha3": "KHM",
"name": "Cambodia",
"alpha2": "KH",
"numeric": 116
},
{
"alpha3": "IDN",
"name": "Indonesia",
"alpha2": "ID",
"numeric": 360
},
{
"alpha3": "LAO",
"name": "Laos",
"alpha2": "LA",
"numeric": 418
},
{
"alpha3": "MYS",
"name": "Malaysia",
"alpha2": "MY",
"numeric": 458
},
{
"alpha3": "MMR",
"name": "Myanmar",
"alpha2": "MM",
"numeric": 458
},
{
"alpha3": "PHL",
"name": "Phillippines",
"alpha2": "PH",
"numeric": 608
},
{
"alpha3": "SGP",
"name": "Singapore",
"alpha2": "SG",
"numeric": 702
},
{
"alpha3": "THA",
"name": "Thailand",
"alpha2": "TH",
"numeric": 764
},
{
"alpha3": "TIL",
"name": "Timor Leste",
"alpha2": "TL",
"numeric": 603
},
{
"alpha3": "VNM",
"name": "Vietnam",
"alpha2": "VN",
"numeric": 704
}
]
}
[
{
"type": "json",
"url": "data.json",
"path": "rows"
},
{
"type": "json",
"url": "test.json"
},
{
"type": "json",
"url": "https://sandbox.darylphuah.com/invq1/data.php"
},
{
"type": "csv",
"url": "https://raw.githubusercontent.com/MoH-Malaysia/covid19-public/main/epidemic/deaths_state.csv"
},
{
"type": "csv",
"url": "https://raw.githubusercontent.com/MoH-Malaysia/covid19-public/main/epidemic/cases_age.csv"
}
]
<!DOCTYPE html>
<meta charset="utf-8">
<html>
<head>
<title>d3.js table</title>
<script type="text/javascript" src="//d3js.org/d3.v7.min.js"></script>
<style>
body {
font-family:sans-serif;
}
form, table, .error {
margin:12px;
}
</style>
</head>
<body>
<script>
d3.json('endpoints.json').then(endpoints=>{
d3.select('body')
.selectAll('select').data([endpoints])
.join('select')
.on('change',function(e,d){
let url = d3.select(this).property('value'),
endpoint = endpoints.find(d=>d.url==url);
console.log('url', url);
d3.selectAll('.error').data([])
.join('div');
d3.select('.root').remove();
window.setTimeout(()=>{
datasource(endpoint);
},300);
})
.selectAll('option').data(d=>d)
.join('option')
.html(d=>d.url);
datasource(
endpoints.find(d=>d.url==d3.select('select').property('value'))
);
});
function datasource(endpoint) {
console.log('endpoint', endpoint);
let links = [endpoint],
sources = links
.map(d=>d.type=='json' ? d3.json(d.url)
: d.type=='csv' ? d3.csv(d.url)
: d.type=='tsv' ? d3.tsv(d.url)
: d3.text(d.url)
);
console.log('links', links);
console.time('loading');
Promise.all(sources)
.then((raw)=>{
console.timeEnd('loading');
console.log('raw', raw);
let data = endpoint.path ? raw[0][endpoint.path] : raw[0];
console.log('data', data);
d3.select('body')
.selectAll('.root').data([data])
.join('div').attr('class','root')
.call(sel=>{
sel.selectAll('h1').data(d=>[d])
.join('h1')
.html('select & sort columns')
sel.selectAll('.rowsize').data(d=>[d])
.join('div').attr('class','rowsize')
.html(d=>'no of rows: '+ d3.format(',')(d.length) )
sel.selectAll('.colsize').data(d=>[d])
.join('div').attr('class','colsize')
.html(d=>'no of columns: '+ d3.format(',')( Object.entries(d[0]).length) )
sel.selectAll('form').data(d=>[d])
.join('form')
.style('display','flex')
.style('flex-wrap','wrap')
.style('justify-content','flex-start')
.selectAll('.inputs').data(d=>Object.entries(d[0]).map(d=>d[0]))
.join('div').attr('class','inputs')
.style('flex','1 1 auto')
.call(sel=>{
sel.selectAll('input').data(d=>[d])
.join('input')
.attr('type','checkbox')
.attr('checked', 'checked')
.on('input', function(e,d){
window.setTimeout(()=>{
d3.select('.root').call(renderTable);
},100);
})
.html(d=>d)
sel.selectAll('span').data(d=>[d])
.join('span')
.html(d=>d)
});
sel.call(renderTable)
function renderTable(sel) {
let columns = d3.selectAll('input:checked').data();
sel.selectAll('table').data(d=>[d])
.join('table')
.call(sel=>{
sel.selectAll('thead').data(d=>[d])
.join('thead')
.selectAll('tr').data(d=>[d])
.join('tr')
.selectAll('th').data(d=>{
return Object.entries(d[0])
.filter(d=>columns.includes(d[0]))
.map(d=>d[0])
})
.join('th')
.style('cursor','pointer')
.on('click', function(e,d){
let sel = d3.select(this),
order_descending = !sel.classed('order_descending');
sel.classed('order_descending', order_descending);
window.setTimeout(()=>{
d3.select(this.closest('table'))
.select('tbody')
.selectAll('tr')
.sort((a,b)=>order_descending
? d3.descending(a[d],b[d])
: d3.ascending(a[d],b[d])
)
},100);
})
.html(d=>d)
sel.selectAll('tbody').data(d=>[d]).join('tbody')
.selectAll('tr').data(d=>d).join('tr')
.selectAll('td').data(d=>{
return Object.entries(d)
.filter(d=>columns.includes(d[0]))
.map(d=>d[1])
}).join('td')
.call(sel=>{
sel.selectAll('img').data(d=>d&&d.toString().match(/^data:image/) ? [d] : [])
.join('img')
.attr('src', d=>d)
sel.selectAll('span').data(d=>!(d&&d.toString().match(/^data:image/)) ? [d] : [])
.join('span')
.html(d=>d)
})
});
}
});
})
.catch(function(err) {
d3.selectAll('.error').data([err])
.join('div').attr('class','error')
.style('color','red')
.html(d=>d.message)
});
}
</script>
</body>
[{"id":1,"name":"Jaxspan","city":"Busalangga","country":"Indonesia","description":"1E2","timezone":null,"image":null},
{"id":2,"name":"Flashset","city":"Velká nad Veličkou","country":"Czech Republic","description":"!@#$%^&*()","timezone":null,"image":null},
{"id":3,"name":"Skalith","city":"Skidal’","country":"Belarus","description":"test⁠test‫","timezone":"Europe/Minsk","image":""},
{"id":4,"name":"Plajo","city":"Cafarnaum","country":"Brazil","description":"社會科學院語學研究所","timezone":"America/Bahia","image":""},
{"id":5,"name":"Jabbersphere","city":"Canhestros","country":"Portugal","description":"ثم نفس سقطت وبالتحديد،, جزيرتي باستخدام أن دنو. إذ هنا؟ الستار وتنصيب كان. أهّل ايطاليا، بريطانيا-فرنسا قد أخذ. سليمان، إتفاقية بين ما, يذكر الحدود أي بعد, معاملة بولندا، الإطلاق عل إيو.","timezone":"Europe/Lisbon","image":""},
{"id":6,"name":"Reallinks","city":"Quatro Barras","country":"Brazil","description":null,"timezone":null,"image":null},
{"id":7,"name":"Zooxo","city":"Kolkhozobod","country":"Tajikistan","description":"../../../../../../../../../../../etc/passwd%00","timezone":null,"image":null},
{"id":8,"name":"Yadel","city":"Sangar Sarāy","country":"Afghanistan","description":"␢","timezone":null,"image":null},
{"id":9,"name":"Dynabox","city":"Tuchkovo","country":"Russia","description":"
test
","timezone":"Europe/Moscow","image":""},
{"id":10,"name":"Realblab","city":"Merke","country":"Kazakhstan","description":"`⁄€‹›fifl‡°·‚—±","timezone":"Asia/Almaty","image":""},
{"id":11,"name":"Linkbuzz","city":"Jutrosin","country":"Poland","description":"⁰⁴⁵","timezone":null,"image":null},
{"id":12,"name":"Lazzy","city":"Wŏnsan","country":"North Korea","description":"/dev/null; touch /tmp/blns.fail ; echo","timezone":"Asia/Pyongyang","image":""},
{"id":13,"name":"Zoombox","city":"Čoka","country":"Serbia","description":"0/0","timezone":"Europe/Belgrade","image":""},
{"id":14,"name":"Skajo","city":"Chernukha","country":"Russia","description":"`⁄€‹›fifl‡°·‚—±","timezone":null,"image":null},
{"id":15,"name":"Zava","city":"Manadhoo","country":"Maldives","description":"<>?:\"{}|_+","timezone":null,"image":null},
{"id":16,"name":"Katz","city":"Sapporo","country":"Japan","description":"١٢٣","timezone":"Asia/Tokyo","image":""},
{"id":17,"name":"Skinder","city":"Kaltungo","country":"Nigeria","description":"̗̺͖̹̯͓Ṯ̤͍̥͇͈h̲́e͏͓̼̗̙̼̣͔ ͇̜̱̠͓͍ͅN͕͠e̗̱z̘̝̜̺͙p̤̺̹͍̯͚e̠̻̠͜r̨̤͍̺̖͔̖̖d̠̟̭̬̝͟i̦͖̩͓͔̤a̠̗̬͉̙n͚͜ ̻̞̰͚ͅh̵͉i̳̞v̢͇ḙ͎͟-҉̭̩̼͔m̤̭̫i͕͇̝̦n̗͙ḍ̟ ̯̲͕͞ǫ̟̯̰̲͙̻̝f ̪̰̰̗̖̭̘͘c̦͍̲̞͍̩̙ḥ͚a̮͎̟̙͜ơ̩̹͎s̤.̝̝ ҉Z̡̖̜͖̰̣͉̜a͖̰͙̬͡l̲̫̳͍̩g̡̟̼̱͚̞̬ͅo̗͜.̟","timezone":"Africa/Lagos","image":""},
{"id":18,"name":"Dynazzy","city":"Dimitrovgrad","country":"Serbia","description":"田中さんにあげて下さい","timezone":"Europe/Belgrade","image":""},
{"id":19,"name":"Bubblemix","city":"Gjøvik","country":"Norway","description":"ÅÍÎÏ˝ÓÔÒÚÆ☃","timezone":"Europe/Oslo","image":""},
{"id":20,"name":"Jabbercube","city":"Semiluki","country":"Russia","description":"ثم نفس سقطت وبالتحديد،, جزيرتي باستخدام أن دنو. إذ هنا؟ الستار وتنصيب كان. أهّل ايطاليا، بريطانيا-فرنسا قد أخذ. سليمان، إتفاقية بين ما, يذكر الحدود أي بعد, معاملة بولندا، الإطلاق عل إيو.","timezone":"Europe/Moscow","image":""},
{"id":21,"name":"Midel","city":"Pulap","country":"Micronesia","description":"𠜎𠜱𠝹𠱓𠱸𠲖𠳏","timezone":null,"image":null},
{"id":22,"name":"Pixonyx","city":"Idvor","country":"Serbia","description":null,"timezone":null,"image":null},
{"id":23,"name":"Tagchat","city":"Pingdingbu","country":"China","description":"","timezone":"Asia/Harbin","image":""},
{"id":24,"name":"Thoughtbridge","city":"Tembau","country":"Indonesia","description":"1/0","timezone":null,"image":null},
{"id":25,"name":"Zoombox","city":"Villejuif","country":"France","description":"🐵 🙈 🙉 🙊","timezone":null,"image":null}]
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment