|
console.clear(); |
|
|
|
var shapeScale = 0.6; |
|
|
|
var keyedData = { |
|
pointiness_true: { |
|
name: "pointiness_true", |
|
isRounding: true, |
|
categoryName: "pointiness", |
|
categories: ["pointy", "round"], |
|
textPlacements: {}, |
|
}, |
|
pointiness_false: { |
|
name: "pointiness_false", |
|
isRounding: false, |
|
categoryName: "pointiness", |
|
categories: ["pointy", "round", "other"], |
|
textPlacements: {}, |
|
}, |
|
shape_name_true: { |
|
name: "shape_name_true", |
|
isRounding: true, |
|
categoryName: "shape_name", |
|
categories: ["circle", "triangle", "rect"], |
|
textPlacements: {}, |
|
}, |
|
shape_name_false: { |
|
name: "shape_name_false", |
|
isRounding: false, |
|
categoryName: "shape_name", |
|
categories: ["circle", "triangle", "rect", "other"], |
|
textPlacements: {}, |
|
}, |
|
size_true: { |
|
name: "size_true", |
|
isRounding: true, |
|
categoryName: "size", |
|
categories: ["small", "large"], |
|
textPlacements: {}, |
|
}, |
|
size_false: { |
|
name: "size_false", |
|
isRounding: false, |
|
categoryName: "size", |
|
categories: ["small", "large", "other"], |
|
textPlacements: {}, |
|
}, |
|
}; |
|
|
|
var data = []; |
|
for (var key in keyedData) { |
|
data.push(keyedData[key]); |
|
} |
|
|
|
var state = { |
|
selected: data[0], |
|
selectedTopIndex: 0, |
|
selectedBottomIndex: 0, |
|
}; |
|
|
|
function updateState( |
|
category, |
|
rounding, |
|
topIndex = undefined, |
|
bottomIndex = undefined |
|
) { |
|
var key = category + "_" + rounding; |
|
state.selected = keyedData[key]; |
|
state.selectedTopIndex = topIndex; |
|
state.selectedBottomIndex = bottomIndex; |
|
} |
|
|
|
|
|
var textPlacements = {}; |
|
|
|
var divHeight = 720; |
|
var divWidth = 850; |
|
|
|
var c = d3.conventions({ |
|
sel: d3.select(".shape-explainer").html(""), |
|
width: divWidth, |
|
height: divHeight, |
|
layers: "ds", |
|
}); |
|
|
|
var buttonHeight = 35; |
|
var buttonWidth = 200; |
|
var buttonBuffer = 15; |
|
var topRightShift = 200; |
|
var bottomRightShift = 270; |
|
|
|
function setActiveButton() { |
|
topExplainerButtonSel.classed( |
|
"explainer-active-button", |
|
(d, i) => i == state.selectedTopIndex |
|
); |
|
bottomExplainerButtonSel.classed( |
|
"explainer-active-button", |
|
(d, i) => i == state.selectedBottomIndex |
|
); |
|
} |
|
|
|
|
|
c.svg |
|
.append("text.top-explainer-text") |
|
.at({ |
|
textAnchor: "left", |
|
dominantBaseline: "top", |
|
dy: ".33em", |
|
}) |
|
.translate([0, buttonHeight / 2]) |
|
.text("All shapes are basically..."); |
|
|
|
c.svg |
|
.append("text.bottom-explainer-text") |
|
.at({ |
|
textAnchor: "left", |
|
dominantBaseline: "top", |
|
dy: ".33em", |
|
}) |
|
.translate([0, buttonHeight * 1.5 + buttonBuffer]) |
|
.text("Everything else should be labeled..."); |
|
|
|
|
|
var topExplainerButtonSel = c.svg |
|
.appendMany("g.explainer-button", ["pointiness", "shape_name", "size"]) |
|
.at({}) |
|
.translate((d, i) => [topRightShift + i * (buttonWidth + buttonBuffer), 0]) |
|
.on("click", function (d, i) { |
|
updateState( |
|
d, |
|
state.selected.isRounding, |
|
(topIndex = i), |
|
(bottomIndex = state.selectedBottomIndex) |
|
); |
|
setActiveButton(); |
|
moveShapes(); |
|
}); |
|
|
|
topExplainerButtonSel.append("rect").at({ |
|
height: buttonHeight, |
|
width: buttonWidth, |
|
class: "explainer-rect", |
|
}); |
|
|
|
topExplainerButtonSel |
|
.append("text") |
|
.at({ |
|
textAnchor: "middle", |
|
dy: ".33em", |
|
x: buttonWidth / 2, |
|
y: buttonHeight / 2, |
|
class: "dropdown", |
|
}) |
|
.text((d, i) => toShortValueStringDict[d]); |
|
|
|
var bottomExplainerButtonSel = c.svg |
|
.appendMany("g.explainer-button", ["true", "false"]) |
|
.at({}) |
|
.translate((d, i) => [ |
|
bottomRightShift + i * (buttonWidth + buttonBuffer), |
|
buttonHeight + buttonBuffer, |
|
]) |
|
.on("click", function (d, i) { |
|
updateState( |
|
state.selected.categoryName, |
|
d, |
|
(topIndex = state.selectedTopIndex), |
|
(bottomIndex = i) |
|
); |
|
setActiveButton(); |
|
moveShapes(); |
|
}); |
|
|
|
bottomExplainerButtonSel.append("rect").at({ |
|
height: buttonHeight, |
|
width: buttonWidth, |
|
class: "explainer-rect", |
|
}); |
|
|
|
bottomExplainerButtonSel |
|
.append("text") |
|
.at({ |
|
textAnchor: "middle", |
|
dy: ".33em", |
|
x: buttonWidth / 2, |
|
y: buttonHeight / 2, |
|
class: "dropdown", |
|
}) |
|
.text((d, i) => toDropdownValueRoundingStringDict[d]); |
|
|
|
var horizontalHeight = divHeight * (5 / 8); |
|
var horizontalBuffer = 50; |
|
|
|
p = d3.line()([ |
|
[horizontalBuffer, horizontalHeight], |
|
[divWidth - horizontalBuffer, horizontalHeight], |
|
]); |
|
|
|
var horizontal = c.svg |
|
.append("path") |
|
.at({ |
|
d: p, |
|
stroke: "black", |
|
strokeWidth: 1, |
|
}) |
|
.translate([0, 0]) |
|
.style("stroke-dasharray", "5, 5"); |
|
|
|
|
|
c.svg |
|
.append("text.label-correct") |
|
.at({ |
|
x: -400, |
|
y: 90, |
|
}) |
|
.text("correctly classified") |
|
.attr("transform", "rotate(-90)"); |
|
|
|
c.svg |
|
.append("text.label-correct") |
|
.at({ |
|
x: -630, |
|
y: 90, |
|
}) |
|
.text("incorrectly classified") |
|
.attr("transform", "rotate(-90)"); |
|
|
|
|
|
|
|
function getFineAdjustment(shape) { |
|
if ( |
|
shape.shape_name == "rt_rect" && |
|
shape.correctness == "incorrect" && |
|
shape.gt == "shaded" |
|
) { |
|
return 4; |
|
} |
|
if ( |
|
shape.shape_name == "rect" && |
|
shape.correctness == "incorrect" && |
|
shape.gt == "unshaded" |
|
) { |
|
return -10; |
|
} |
|
if ( |
|
shape.shape_name == "triangle" && |
|
shape.correctness == "incorrect" && |
|
shape.gt == "unshaded" |
|
) { |
|
return 0; |
|
} |
|
if ( |
|
shape.shape_name == "rt_circle" && |
|
shape.correctness == "incorrect" && |
|
shape.size == "small" |
|
) { |
|
return -20; |
|
} |
|
if ( |
|
shape.shape_name == "rt_triangle" && |
|
shape.correctness == "incorrect" && |
|
shape.size == "small" |
|
) { |
|
return -20; |
|
} |
|
return 0; |
|
} |
|
|
|
function getFinalCategory(labelName, isRounding) { |
|
if (isRounding == true) { |
|
return labelName.replace("rt_", ""); |
|
} else { |
|
if (labelName.includes("rt_")) { |
|
return "other"; |
|
} else { |
|
return labelName; |
|
} |
|
} |
|
} |
|
|
|
var startingCorrectHeight = horizontalHeight - 50; |
|
var startingIncorrectHeight = horizontalHeight + 50; |
|
var maxHeight = 180; |
|
var xRowAdjustment = 50; |
|
var heightBuffer = 10; |
|
|
|
function getPathHeight(inputPath) { |
|
var placeholder = c.svg.append("path").at({ |
|
d: scaleShapePath(inputPath, shapeScale), |
|
}); |
|
var height = placeholder.node().getBBox().height; |
|
placeholder.remove(); |
|
return height + heightBuffer; |
|
} |
|
|
|
|
|
function generatePlacements() { |
|
for (selectionCriteria of data) { |
|
|
|
var nCategories = selectionCriteria.categories.length; |
|
var centerX = []; |
|
for (var i = 0; i < nCategories; i++) { |
|
var startingX = divWidth * ((i + 1) / (nCategories + 1)); |
|
centerX.push(startingX); |
|
|
|
selectionCriteria["textPlacements"][ |
|
selectionCriteria.categories[i] |
|
] = startingX; |
|
} |
|
|
|
|
|
var locationParams = {}; |
|
for (categoryIdx in selectionCriteria.categories) { |
|
var categoryName = selectionCriteria.categories[categoryIdx]; |
|
locationParams[categoryName] = { |
|
correctX: centerX[categoryIdx], |
|
incorrectX: centerX[categoryIdx], |
|
lastCorrectY: startingCorrectHeight, |
|
lastIncorrectY: startingIncorrectHeight, |
|
}; |
|
} |
|
|
|
for (shape of shapeParams) { |
|
shapeCategory = getFinalCategory( |
|
shape[selectionCriteria.categoryName], |
|
selectionCriteria.isRounding |
|
); |
|
var shapeHeight = getPathHeight(shape.path); |
|
var shapeX, |
|
shapeY = 0; |
|
if (shape.correctness == "correct") { |
|
shapeY = locationParams[shapeCategory]["lastCorrectY"]; |
|
shapeX = locationParams[shapeCategory]["correctX"]; |
|
|
|
if ( |
|
startingCorrectHeight - |
|
locationParams[shapeCategory]["lastCorrectY"] >= |
|
maxHeight |
|
) { |
|
|
|
locationParams[shapeCategory]["lastCorrectY"] = |
|
startingCorrectHeight; |
|
|
|
locationParams[shapeCategory]["correctX"] = |
|
locationParams[shapeCategory]["correctX"] + |
|
xRowAdjustment; |
|
} else { |
|
locationParams[shapeCategory]["lastCorrectY"] += |
|
-1 * shapeHeight; |
|
} |
|
} else { |
|
shapeY = locationParams[shapeCategory]["lastIncorrectY"]; |
|
shapeX = locationParams[shapeCategory]["incorrectX"]; |
|
|
|
if ( |
|
locationParams[shapeCategory]["lastIncorrectY"] - |
|
startingIncorrectHeight >= |
|
maxHeight |
|
) { |
|
|
|
locationParams[shapeCategory]["lastIncorrectY"] = |
|
startingIncorrectHeight; |
|
|
|
locationParams[shapeCategory]["incorrectX"] = |
|
locationParams[shapeCategory]["incorrectX"] + |
|
xRowAdjustment; |
|
} else { |
|
locationParams[shapeCategory]["lastIncorrectY"] += |
|
shapeHeight; |
|
} |
|
} |
|
shapeY = shapeY + getFineAdjustment(shape); |
|
shape[selectionCriteria.name + "_X"] = shapeX; |
|
shape[selectionCriteria.name + "_Y"] = shapeY; |
|
} |
|
} |
|
} |
|
|
|
generatePlacements(); |
|
|
|
function getLocation(shape) { |
|
return [ |
|
shape[state.selected.name + "_X"], |
|
shape[state.selected.name + "_Y"], |
|
]; |
|
} |
|
|
|
function scaleShapePath(shapePath, factor = 0.5) { |
|
var newShapePath = ""; |
|
for (var token of shapePath.split(" ")) { |
|
if (parseInt(token)) { |
|
newShapePath = newShapePath + parseInt(token) * factor; |
|
} else { |
|
newShapePath = newShapePath + token; |
|
} |
|
newShapePath = newShapePath + " "; |
|
} |
|
return newShapePath; |
|
} |
|
|
|
|
|
var explainerShapeSel = c.svg |
|
.appendMany("path.shape", shapeParams) |
|
.at({ |
|
d: (d) => scaleShapePath(d.path, shapeScale), |
|
class: (d) => "gt-" + d.gt + " " + d.correctness, |
|
}) |
|
.translate(function (d) { |
|
return getLocation(d); |
|
}); |
|
|
|
explainerShapeSel.classed("is-classified", true); |
|
|
|
function getColor(d) { |
|
var scaleRowValue = d3.scaleLinear().domain([0.3, 1.0]).range([0, 1]); |
|
return d3.interpolateRdYlGn(scaleRowValue(d)); |
|
} |
|
|
|
|
|
function getResults() { |
|
return calculateResults( |
|
(property = state.selected.categoryName), |
|
(useGuess = state.selected.isRounding) |
|
); |
|
} |
|
|
|
function getCategoryAccuracy(results, category) { |
|
for (var key of results) { |
|
if (key.rawCategoryName == category) { |
|
return key.accuracy; |
|
} |
|
} |
|
} |
|
|
|
|
|
function toExplainerDisplayString(categoryName) { |
|
if (categoryName == "large") { |
|
return "big"; |
|
} |
|
if (categoryName == "rect") { |
|
return "rectangle"; |
|
} |
|
return categoryName; |
|
} |
|
|
|
function getExplainerTextColor(d, i) { |
|
console.log(d == "large"); |
|
if (d == "large" && state.selected.isRounding == false) { |
|
return "#ffccd8"; |
|
} else { |
|
return "#000000"; |
|
} |
|
} |
|
|
|
function updateText() { |
|
var explainerResults = getResults(); |
|
|
|
d3.selectAll(".explainer-label-text").html(""); |
|
d3.selectAll(".explainer-label-rect").remove(); |
|
|
|
var rectHeight = 30; |
|
var rectWidth = 80; |
|
var textRect = c.svg |
|
.appendMany("rect.column-text-rect", state.selected.categories) |
|
.at({ |
|
fill: (d) => getColor(getCategoryAccuracy(explainerResults, d)), |
|
height: rectHeight, |
|
width: rectWidth, |
|
class: "explainer-label-rect", |
|
}) |
|
.translate((d) => [ |
|
state.selected.textPlacements[d] - rectWidth / 2, |
|
horizontalHeight - rectHeight / 2, |
|
]); |
|
|
|
var text = c.svg |
|
.appendMany("text.column-text", state.selected.categories) |
|
.at({ |
|
textAnchor: "middle", |
|
dominantBaseline: "central", |
|
class: "explainer-label-text", |
|
}) |
|
.st({ |
|
fill: getExplainerTextColor, |
|
}) |
|
.text((d) => toExplainerDisplayString(d)) |
|
.translate((d) => [state.selected.textPlacements[d], horizontalHeight]); |
|
} |
|
|
|
function moveShapes() { |
|
explainerShapeSel |
|
.transition() |
|
.duration(500) |
|
.translate((d) => getLocation(d)); |
|
updateText(); |
|
} |
|
|
|
setActiveButton(); |
|
updateText(); |