Spaces:
Running
Running
| !(async function(){ | |
| var data = await util.getFile('mnist_train.csv') | |
| data.forEach(d => { | |
| delete d[''] | |
| d.i = +d.i | |
| }) | |
| var sel = d3.select('.umap-digit').html('') | |
| .at({role: 'graphics-document', 'aria-label': `Color coded UMAP of MNIST 1s showing that increasing privacy will misclassify slanted and serif β1β digits first.`}) | |
| var umapSel = sel.append('div') | |
| .append('div.chart-title').text('Sensitivity to higher privacy levels β') | |
| .parent() | |
| .st({maxWidth: 600, margin: '0 auto', marginBottom: 10}) | |
| .append('div') | |
| var buttonSel = sel.append('div.digit-button-container') | |
| .appendMany('div.button', d3.range(10)) | |
| .text(d => d) | |
| .on('click', d => drawDigitUmap(d)) | |
| drawDigitUmap(1) | |
| async function drawDigitUmap(digit){ | |
| buttonSel.classed('active', d => d == digit) | |
| // var umap = await util.getFile(`umap_train_${digit}.npy`) | |
| var umap = await util.getFile(`cns-cache/umap_train_784_${digit}.npy`) | |
| util.getFile(`cns-cache/mnist_train_raw_${digit}.npy`) | |
| var digitData = data | |
| .filter(d => d.y == digit) | |
| .map((d, i) => ({ | |
| rawPos: [umap.data[i*2 + 0], umap.data[i*2 + 1]], | |
| priv_order: d.priv_order, | |
| y: d.y, | |
| i: d.i | |
| })) | |
| var c = d3.conventions({ | |
| sel: umapSel.html(''), | |
| width: 600, | |
| height: 600, | |
| layers: 'sdc', | |
| margin: {top: 45} | |
| }) | |
| var nTicks = 200 | |
| c.svg.appendMany('rect', d3.range(nTicks)) | |
| .at({ | |
| height: 15, | |
| width: 1, | |
| fill: i => d3.interpolatePlasma(i/nTicks), | |
| }) | |
| .translate(i => [c.width/2 - nTicks/2 - 20 + i, -c.margin.top + 5]) | |
| c.x.domain(d3.extent(digitData, d => d.rawPos[0])) | |
| c.y.domain(d3.extent(digitData, d => d.rawPos[1]))//.range([0, c.height]) | |
| digitData.forEach(d => d.pos = [c.x(d.rawPos[0]), c.y(d.rawPos[1])]) | |
| c.sel.select('canvas').st({pointerEvents: 'none'}) | |
| var divSel = c.layers[1].st({pointerEvents: 'none'}) | |
| var ctx = c.layers[2] | |
| digitData.forEach(d => { | |
| ctx.beginPath() | |
| ctx.fillStyle = d3.interpolatePlasma(1 - d.priv_order/60000) | |
| ctx.rect(d.pos[0], d.pos[1], 2, 2) | |
| ctx.fill() | |
| }) | |
| var p = 10 | |
| c.svg | |
| .append('rect').at({width: c.width + p*2, height: c.height + p*2, x: -p, y: -p}) | |
| .parent() | |
| .call(d3.attachTooltip) | |
| .on('mousemove', function(){ | |
| var [px, py] = d3.mouse(this) | |
| var minPoint = _.minBy(digitData, d => { | |
| var dx = d.pos[0] - px | |
| var dy = d.pos[1] - py | |
| return dx*dx + dy*dy | |
| }) | |
| var s = 4 | |
| var c = d3.conventions({ | |
| sel: ttSel.html('').append('div'), | |
| width: 4*28, | |
| height: 4*28, | |
| layers: 'cs', | |
| margin: {top: 0, left: 0, right: 0, bottom: 0} | |
| }) | |
| // <div>Label: ${minPoint.y}</div> | |
| // ttSel.append('div').html(` | |
| // <div>Privacy Rank ${d3.format(',')(minPoint.priv_order)}</div> | |
| // `) | |
| ttSel.classed('tooltip-footnote', 0).st({width: 112}) | |
| util.drawDigit(c.layers[0], +minPoint.i, s) | |
| }) | |
| if (digit == 1){ | |
| var circleDigits = [ | |
| {r: 40, index: 1188}, | |
| {r: 53, index: 18698}, | |
| {r: 40, index: 1662} | |
| ] | |
| circleDigits.forEach(d => { | |
| d.pos = digitData.filter(e => e.priv_order == d.index)[0].pos | |
| }) | |
| c.svg.append('g') | |
| .appendMany('g', circleDigits) | |
| .translate(d => d.pos) | |
| .append('circle') | |
| .at({r: d => d.r, fill: 'none', stroke: '#fff', strokeDasharray: '2 3', strokeWidth: 1}) | |
| var {r, pos} = circleDigits[0] | |
| divSel | |
| .append('div').translate(pos) | |
| .append('div').translate([r + 20, -r + 10]) | |
| .st({width: 150, fontWeight: 300, fontSize: 14, color: '#fff', xbackground: 'rgba(255,0,0,.2)', lineHeight: '1.2em'}) | |
| .text('Increasing privacy will misclassify slanted and serif β1β digits first') | |
| } | |
| } | |
| })() | |