Balance of power
Last active
May 15, 2019 17:52
-
-
Save gabrielflorit/25bfa64c04495b3f3203bb8e830957d2 to your computer and use it in GitHub Desktop.
Balance of power
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
license: mit | |
height: 720 | |
border: no |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
.container{margin:1em;padding:0;position:relative}.container img{width:100%;opacity:1}span.label{display:inline-block;text-align:center;font-weight:700;width:2.75em}span.label.total{width:20em;text-align:left}input{width:100%}svg{display:block;border:1px solid red;width:50%;margin:0 auto}svg circle.ring{fill:none;stroke:grey;stroke-width:.01px}svg circle.dem{fill:#00f}svg circle.gop{fill:red}svg circle.ind{fill:green}svg circle.none{fill:grey;opacity:.25} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
var rows=[d3.range(23),d3.range(26),d3.range(30),d3.range(32),d3.range(36),d3.range(38),d3.range(42),d3.range(46),d3.range(50),d3.range(54),d3.range(58)],totalSeats=d3.merge(rows).length,RADIUS=2.25,outerWidth=200,outerHeight=outerWidth/2+RADIUS,width=outerWidth-2*RADIUS,height=outerHeight-RADIUS,gap=width/2*.4,svg=d3.select("svg").attr("viewBox","0 0 "+outerWidth+" "+outerHeight).append("g").attr("transform","translate("+outerWidth/2+", "+height+")"),updateElement=function(t,e){return document.querySelector(t).textContent=e},drawData=function(){var t=+document.querySelector("input.dem").value,e=+document.querySelector("input.ind").value,n=+document.querySelector("input.gop").value;updateElement("span.dem",t),updateElement("span.ind",e),updateElement("span.gop",n);var a=totalSeats-(t+e+n);if(a>=0){updateElement("span.total",t+e+n+" out of "+totalSeats);var r=t,o=e,i=n,l=rows.map(function(a){return{total:a.length,demPortion:Math.ceil(t*a.length/totalSeats),indPortion:Math.ceil(e*a.length/totalSeats),gopPortion:Math.ceil(n*a.length/totalSeats)}}),d=l.map(function(t,e,n){var a,l,d,u,s=t.total,c=t.demPortion,g=t.indPortion,p=t.gopPortion;if(e===n.length-1)a=r,l=o,d=i,u=s-(a+l+d);else{u=s;var m=r>0?Math.min(r,c):0,h=u>0?Math.min(u,m):0;a=h,u-=a,r-=a;var S=o>0?Math.min(o,g):0,v=u>0?Math.min(u,S):0;l=v,u-=l,o-=l;var f=i>0?Math.min(i,p):0,w=u>0?Math.min(u,f):0;d=w,u-=d,i-=d}return{demSeats:a,indSeats:l,noneSeats:u,gopSeats:d,total:s}}),u=d3.merge(d.map(function(t,e){return d3.merge([d3.range(t.demSeats).map(function(){return"dem"}),d3.range(t.indSeats).map(function(){return"ind"}),d3.range(t.noneSeats).map(function(){return"none"}),d3.range(t.gopSeats).map(function(){return"gop"})]).reverse().map(function(t,n,a){return{party:t,row:e,seat:n,rowSeats:a.length}})})),s=d.length,c=d3.scalePoint().range([gap,width/2]).domain(d3.range(s+1));svg.selectAll("*").remove();var g=(svg.selectAll("circle.ring").data(d).enter().append("circle").attr("class","ring").attr("r",function(t,e){return c(e+1)}).attr("cx",0).attr("cy",0),svg.selectAll("circle.seat").data(u).enter().append("circle").attr("class",function(t){return t.party+" seat"}).attr("r",RADIUS).attr("cx",function(t){return c(t.row+1)}).attr("cy",0).attr("transform",function(t,e){return"rotate(-"+180/(t.rowSeats-1)*t.seat+")"}),svg.selectAll("circle.seat.dem").size()),p=svg.selectAll("circle.seat.ind").size(),m=svg.selectAll("circle.seat.gop").size();g===t&&p===e&&m===n||(alert("error! see the console. slider input did not match graph dots."),console.error(JSON.stringify({dem:{input:t,output:g},ind:{input:e,output:p},gop:{input:n,output:m}},null,2)))}else updateElement("span.total","Total exceeds available seats!")};d3.selectAll("input").on("input",function(t){return drawData()}),drawData(); | |
//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["script.js"],"names":["const","rows","d3","range","totalSeats","merge","length","RADIUS","outerWidth","outerHeight","width","height","gap","svg","select","attr","append","updateElement","selector","value","document","querySelector","textContent","drawData","dem","ind","gop","none","let","demSeatsLeft","indSeatsLeft","gopSeatsLeft","rowProportions","map","row","total","demPortion","Math","ceil","indPortion","gopPortion","seatRows","index","array","demSeats","indSeats","gopSeats","rowSeatsLeft","demSeatsToAssign","min","demSeatsAvailable","indSeatsToAssign","indSeatsAvailable","gopSeatsToAssign","gopSeatsAvailable","noneSeats","seats","i","reverse","party","seat","a","rowSeats","ROWS","ringWidth","scalePoint","domain","selectAll","remove","demSelectionSize","data","enter","d","size","indSelectionSize","gopSelectionSize","alert","console","error","JSON","stringify","input","output","on"],"mappings":"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","file":"script.js","sourcesContent":["// This array of arrays represents our desired seating arrangement.\n// Each new row has more seats than the previous one.\nconst rows = [\n\td3.range(23),\n\td3.range(26),\n\td3.range(30),\n\td3.range(32),\n\td3.range(36),\n\td3.range(38),\n\td3.range(42),\n\td3.range(46),\n\td3.range(50),\n\td3.range(54),\n\td3.range(58),\n]\n\n// The total number of seats.\nconst totalSeats = d3.merge(rows).length\n\nconst RADIUS = 2.25\n\nconst outerWidth = 200\nconst outerHeight = (outerWidth / 2) + RADIUS\n\nconst width = outerWidth - RADIUS * 2\nconst height = outerHeight - RADIUS\n\nconst gap = (width / 2) * 0.4\n\nconst svg = d3.select('svg')\n\t\t.attr('viewBox', `0 0 ${outerWidth} ${outerHeight}`)\n\t.append('g')\n\t\t.attr('transform',\n\t\t\t`translate(${outerWidth/2}, ${height})`)\n\nconst updateElement = (selector, value) =>\n\tdocument.querySelector(selector).textContent = value\n\nconst drawData = () => {\n\n\tconst dem = +document.querySelector('input.dem').value\n\tconst ind = +document.querySelector('input.ind').value\n\tconst gop = +document.querySelector('input.gop').value\n\tupdateElement('span.dem', dem)\n\tupdateElement('span.ind', ind)\n\tupdateElement('span.gop', gop)\n\t\n\t// This is each party's seats.\n\tconst none = totalSeats - (dem + ind + gop)\n\n\tif (none >= 0) {\n\n\t\tupdateElement('span.total', `${(dem + ind + gop)} out of ${totalSeats}`)\n\n\t\t// We'll use these variables to track remaining seats.\n\t\tlet demSeatsLeft = dem\n\t\tlet indSeatsLeft = ind\n\t\tlet gopSeatsLeft = gop\n\n\t\t// STEP 1:\n\t\t// Calculate each row's ratio of party seats, rounded up.\n\t\tconst rowProportions = rows.map(row => ({\n\t\t\t\ttotal: row.length,\n\t\t\t\tdemPortion: Math.ceil(dem * row.length / totalSeats),\n\t\t\t\tindPortion: Math.ceil(ind * row.length / totalSeats),\n\t\t\t\tgopPortion: Math.ceil(gop * row.length / totalSeats),\n\t\t\t}))\n\n\t\t// STEP 2:\n\t\t// Assign seats until we run out of total party seats.\n\t\t// For each row,\n\t\tconst seatRows = rowProportions.map((row, index, array) => {\n\n\t\t\tconst { total, demPortion, indPortion, gopPortion } = row\n\t\t\tlet demSeats\n\t\t\tlet indSeats\n\t\t\tlet gopSeats\n\t\t\tlet rowSeatsLeft\n\n\t\t\t// If this is the last row, simply assign all leftover seats.\n\t\t\tif (index === array.length - 1) {\n\n\t\t\t\tdemSeats = demSeatsLeft\n\t\t\t\tindSeats = indSeatsLeft\n\t\t\t\tgopSeats = gopSeatsLeft\n\t\t\t\trowSeatsLeft = total - (demSeats + indSeats + gopSeats)\n\n\t\t\t} else {\n\n\t\t\t\t// If this is NOT the last row, let's proceed.\n\n\t\t\t\trowSeatsLeft = total\n\n\t\t\t\t// We want to assign dem seats to this row. Remember that we\n\t\t\t\t// have `demSeatsLeft`, all the seats we have left.\n\t\t\t\t// We want to assign as many of those as possible, given our remaining\n\t\t\t\t// slots.\n\t\t\t\t// So first, IF there are dem seats left, find the min of demSeatsLeft\n\t\t\t\t// and demProportion:\n\t\t\t\tconst demSeatsToAssign = demSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(demSeatsLeft, demPortion) : 0\n\n\t\t\t\t// Next, find the available slots left on this row, and seat as many as\n\t\t\t\t// possible, which is the min of rowSeatsLeft and demSeatsToAssign.\n\t\t\t\tconst demSeatsAvailable = rowSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(rowSeatsLeft, demSeatsToAssign) : 0\n\n\t\t\t\t// This will be the actual dem seats we assign to this row.\n\t\t\t\tdemSeats = demSeatsAvailable\n\n\t\t\t\t// Finally, update this row's seats left,\n\t\t\t\trowSeatsLeft -= demSeats\n\n\t\t\t\t// and update demSeatsLeft.\n\t\t\t\tdemSeatsLeft -= demSeats\n\n\t\t\t\t// We'll do the same thing for ind:\n\t\t\t\tconst indSeatsToAssign = indSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(indSeatsLeft, indPortion) : 0\n\t\t\t\tconst indSeatsAvailable = rowSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(rowSeatsLeft, indSeatsToAssign) : 0\n\t\t\t\tindSeats = indSeatsAvailable\n\t\t\t\trowSeatsLeft -= indSeats\n\t\t\t\tindSeatsLeft -= indSeats\n\n\t\t\t\t// and gop.\n\t\t\t\tconst gopSeatsToAssign = gopSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(gopSeatsLeft, gopPortion) : 0\n\t\t\t\tconst gopSeatsAvailable = rowSeatsLeft > 0 ?\n\t\t\t\t\tMath.min(rowSeatsLeft, gopSeatsToAssign) : 0\n\t\t\t\tgopSeats = gopSeatsAvailable\n\t\t\t\trowSeatsLeft -= gopSeats\n\t\t\t\tgopSeatsLeft -= gopSeats\n\n\t\t\t}\n\n\t\t\treturn {\n\t\t\t\tdemSeats,\n\t\t\t\tindSeats,\n\t\t\t\tnoneSeats: rowSeatsLeft,\n\t\t\t\tgopSeats,\n\t\t\t\ttotal,\n\t\t\t}\n\n\t\t})\n\n\t\t// Convert `seatRows` to the structure we need: an array of seats.\n\t\tconst seats = d3.merge(\n\t\t\tseatRows.map((row, i) =>\n\t\t\t\td3.merge([\n\t\t\t\t\td3.range(row.demSeats).map(() => 'dem'),\n\t\t\t\t\td3.range(row.indSeats).map(() => 'ind'),\n\t\t\t\t\td3.range(row.noneSeats).map(() => 'none'),\n\t\t\t\t\td3.range(row.gopSeats).map(() => 'gop'),\n\t\t\t\t])\n\t\t\t\t.reverse()\n\t\t\t\t.map((party, seat, a) => ({ party, row: i, seat, rowSeats: a.length }))\n\t\t\t)\n\t\t)\n\n\t\tconst ROWS = seatRows.length\n\n\t\tconst ringWidth = d3.scalePoint()\n\t\t\t.range([gap, width/2])\n\t\t\t.domain(d3.range(ROWS + 1))\n\n\t\tsvg.selectAll('*').remove()\n\t\t\n\t\tconst rings = svg.selectAll('circle.ring')\n\t\t\t\t.data(seatRows)\n\t\t\t.enter().append('circle')\n\t\t\t\t.attr('class', 'ring')\n\t\t\t\t.attr('r', (d, i) => ringWidth(i + 1))\n\t\t\t\t.attr('cx', 0)\n\t\t\t\t.attr('cy', 0)\n\n\t\tconst circleSeats = svg.selectAll('circle.seat')\n\t\t\t\t.data(seats)\n\t\t\t.enter().append('circle')\n\t\t\t\t.attr('class', d => `${d.party} seat`)\n\t\t\t\t.attr('r', RADIUS)\n\t\t\t\t.attr('cx', d => ringWidth(d.row + 1))\n\t\t\t\t.attr('cy', 0)\n\t\t\t\t.attr('transform', (d, i) => `rotate(-${180/(d.rowSeats - 1) * d.seat})`)\n\n\t\tconst demSelectionSize = svg.selectAll('circle.seat.dem').size()\n\t\tconst indSelectionSize = svg.selectAll('circle.seat.ind').size()\n\t\tconst gopSelectionSize = svg.selectAll('circle.seat.gop').size()\n\n\t\tif (!(demSelectionSize === dem &&\n\t\t\tindSelectionSize === ind &&\n\t\t\tgopSelectionSize === gop)) {\n\n\t\t\talert('error! see the console. slider input did not match graph dots.')\n\n\t\t\tconsole.error(JSON.stringify({\n\t\t\t\tdem: { input: dem, output: demSelectionSize },\n\t\t\t\tind: { input: ind, output: indSelectionSize },\n\t\t\t\tgop: { input: gop, output: gopSelectionSize },\n\t\t\t}, null, 2))\n\n\t\t}\n\n\t} else {\n\n\t\tupdateElement('span.total', `Total exceeds available seats!`)\n\n\t}\n\n}\n\nd3.selectAll('input').on('input', d => drawData())\ndrawData()\n"]} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
<!DOCTYPE html> | |
<title>Dot matrix</title> | |
<link href='dist.css' rel='stylesheet' /> | |
<body> | |
<p>US House balance of power</p> | |
<div class='container'> | |
<div class="intro"> | |
<p>Dem <span class="dem label">100</span><input class="dem" type="range" min="0" max="435" value="100"></span></p> | |
<p>Ind <span class="ind label">100</span><input class="ind" type="range" min="0" max="435" value="100"></span></p> | |
<p>GOP <span class="gop label">100</span><input class="gop" type="range" min="0" max="435" value="100"></span></p> | |
<p>Total: <span class="total label"></span></p> | |
</div> | |
<svg></svg> | |
</div> | |
<script src='https://cdn.jsdelivr.net/lodash/4.16.6/lodash.min.js'></script> | |
<script src='https://d3js.org/d3.v4.min.js'></script> | |
<script src='dist.js'></script> | |
</body> |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
// This array of arrays represents our desired seating arrangement. | |
// Each new row has more seats than the previous one. | |
const rows = [ | |
d3.range(23), | |
d3.range(26), | |
d3.range(30), | |
d3.range(32), | |
d3.range(36), | |
d3.range(38), | |
d3.range(42), | |
d3.range(46), | |
d3.range(50), | |
d3.range(54), | |
d3.range(58), | |
] | |
// The total number of seats. | |
const totalSeats = d3.merge(rows).length | |
const RADIUS = 2.25 | |
const outerWidth = 200 | |
const outerHeight = (outerWidth / 2) + RADIUS | |
const width = outerWidth - RADIUS * 2 | |
const height = outerHeight - RADIUS | |
const gap = (width / 2) * 0.4 | |
const svg = d3.select('svg') | |
.attr('viewBox', `0 0 ${outerWidth} ${outerHeight}`) | |
.append('g') | |
.attr('transform', | |
`translate(${outerWidth/2}, ${height})`) | |
const updateElement = (selector, value) => | |
document.querySelector(selector).textContent = value | |
const drawData = () => { | |
const dem = +document.querySelector('input.dem').value | |
const ind = +document.querySelector('input.ind').value | |
const gop = +document.querySelector('input.gop').value | |
updateElement('span.dem', dem) | |
updateElement('span.ind', ind) | |
updateElement('span.gop', gop) | |
// This is each party's seats. | |
const none = totalSeats - (dem + ind + gop) | |
if (none >= 0) { | |
updateElement('span.total', `${(dem + ind + gop)} out of ${totalSeats}`) | |
// We'll use these variables to track remaining seats. | |
let demSeatsLeft = dem | |
let indSeatsLeft = ind | |
let gopSeatsLeft = gop | |
// STEP 1: | |
// Calculate each row's ratio of party seats, rounded up. | |
const rowProportions = rows.map(row => ({ | |
total: row.length, | |
demPortion: Math.ceil(dem * row.length / totalSeats), | |
indPortion: Math.ceil(ind * row.length / totalSeats), | |
gopPortion: Math.ceil(gop * row.length / totalSeats), | |
})) | |
// STEP 2: | |
// Assign seats until we run out of total party seats. | |
// For each row, | |
const seatRows = rowProportions.map((row, index, array) => { | |
const { total, demPortion, indPortion, gopPortion } = row | |
let demSeats | |
let indSeats | |
let gopSeats | |
let rowSeatsLeft | |
// If this is the last row, simply assign all leftover seats. | |
if (index === array.length - 1) { | |
demSeats = demSeatsLeft | |
indSeats = indSeatsLeft | |
gopSeats = gopSeatsLeft | |
rowSeatsLeft = total - (demSeats + indSeats + gopSeats) | |
} else { | |
// If this is NOT the last row, let's proceed. | |
rowSeatsLeft = total | |
// We want to assign dem seats to this row. Remember that we | |
// have `demSeatsLeft`, all the seats we have left. | |
// We want to assign as many of those as possible, given our remaining | |
// slots. | |
// So first, IF there are dem seats left, find the min of demSeatsLeft | |
// and demProportion: | |
const demSeatsToAssign = demSeatsLeft > 0 ? | |
Math.min(demSeatsLeft, demPortion) : 0 | |
// Next, find the available slots left on this row, and seat as many as | |
// possible, which is the min of rowSeatsLeft and demSeatsToAssign. | |
const demSeatsAvailable = rowSeatsLeft > 0 ? | |
Math.min(rowSeatsLeft, demSeatsToAssign) : 0 | |
// This will be the actual dem seats we assign to this row. | |
demSeats = demSeatsAvailable | |
// Finally, update this row's seats left, | |
rowSeatsLeft -= demSeats | |
// and update demSeatsLeft. | |
demSeatsLeft -= demSeats | |
// We'll do the same thing for ind: | |
const indSeatsToAssign = indSeatsLeft > 0 ? | |
Math.min(indSeatsLeft, indPortion) : 0 | |
const indSeatsAvailable = rowSeatsLeft > 0 ? | |
Math.min(rowSeatsLeft, indSeatsToAssign) : 0 | |
indSeats = indSeatsAvailable | |
rowSeatsLeft -= indSeats | |
indSeatsLeft -= indSeats | |
// and gop. | |
const gopSeatsToAssign = gopSeatsLeft > 0 ? | |
Math.min(gopSeatsLeft, gopPortion) : 0 | |
const gopSeatsAvailable = rowSeatsLeft > 0 ? | |
Math.min(rowSeatsLeft, gopSeatsToAssign) : 0 | |
gopSeats = gopSeatsAvailable | |
rowSeatsLeft -= gopSeats | |
gopSeatsLeft -= gopSeats | |
} | |
return { | |
demSeats, | |
indSeats, | |
noneSeats: rowSeatsLeft, | |
gopSeats, | |
total, | |
} | |
}) | |
// Convert `seatRows` to the structure we need: an array of seats. | |
const seats = d3.merge( | |
seatRows.map((row, i) => | |
d3.merge([ | |
d3.range(row.demSeats).map(() => 'dem'), | |
d3.range(row.indSeats).map(() => 'ind'), | |
d3.range(row.noneSeats).map(() => 'none'), | |
d3.range(row.gopSeats).map(() => 'gop'), | |
]) | |
.reverse() | |
.map((party, seat, a) => ({ party, row: i, seat, rowSeats: a.length })) | |
) | |
) | |
const ROWS = seatRows.length | |
const ringWidth = d3.scalePoint() | |
.range([gap, width/2]) | |
.domain(d3.range(ROWS + 1)) | |
svg.selectAll('*').remove() | |
const rings = svg.selectAll('circle.ring') | |
.data(seatRows) | |
.enter().append('circle') | |
.attr('class', 'ring') | |
.attr('r', (d, i) => ringWidth(i + 1)) | |
.attr('cx', 0) | |
.attr('cy', 0) | |
const circleSeats = svg.selectAll('circle.seat') | |
.data(seats) | |
.enter().append('circle') | |
.attr('class', d => `${d.party} seat`) | |
.attr('r', RADIUS) | |
.attr('cx', d => ringWidth(d.row + 1)) | |
.attr('cy', 0) | |
.attr('transform', (d, i) => `rotate(-${180/(d.rowSeats - 1) * d.seat})`) | |
const demSelectionSize = svg.selectAll('circle.seat.dem').size() | |
const indSelectionSize = svg.selectAll('circle.seat.ind').size() | |
const gopSelectionSize = svg.selectAll('circle.seat.gop').size() | |
if (!(demSelectionSize === dem && | |
indSelectionSize === ind && | |
gopSelectionSize === gop)) { | |
alert('error! see the console. slider input did not match graph dots.') | |
console.error(JSON.stringify({ | |
dem: { input: dem, output: demSelectionSize }, | |
ind: { input: ind, output: indSelectionSize }, | |
gop: { input: gop, output: gopSelectionSize }, | |
}, null, 2)) | |
} | |
} else { | |
updateElement('span.total', `Total exceeds available seats!`) | |
} | |
} | |
d3.selectAll('input').on('input', d => drawData()) | |
drawData() |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
.container | |
margin 1em | |
padding 0 | |
position relative | |
img | |
width 100% | |
opacity 1 | |
span.label | |
display inline-block | |
text-align center | |
font-weight bold | |
width 2.75em | |
&.total | |
width 20em | |
text-align left | |
input | |
width 100% | |
svg | |
display block | |
border solid red 1px | |
width 50% | |
margin 0 auto | |
circle | |
&.ring | |
fill none | |
stroke grey | |
stroke-width 0.01px | |
&.dem | |
fill blue | |
&.gop | |
fill red | |
&.ind | |
fill green | |
&.none | |
fill grey | |
opacity 0.25 |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment