forked from Open-CT/openct-tasks
261 lines
45 KiB
JavaScript
261 lines
45 KiB
JavaScript
function initTask(subTask) {
|
|
var state = {};
|
|
var level;
|
|
var answer = null;
|
|
var data = {
|
|
easy: {
|
|
cellSize: 40,
|
|
nbCol: 12,
|
|
nbRows: 5,
|
|
xShift: 0,
|
|
yShift: 0,
|
|
// converted on https://www.base64-image.de/
|
|
imgURL: ""
|
|
},
|
|
medium: {
|
|
cellSize: 30,
|
|
nbCol: 24,
|
|
nbRows: 10,
|
|
xShift: 7,
|
|
yShift: -5,
|
|
imgURL: ""
|
|
},
|
|
hard: {
|
|
cellSize: 30,
|
|
nbCol: 20,
|
|
nbRows: 15,
|
|
xShift: 0,
|
|
yShift: 0,
|
|
imgURL: ""
|
|
// imgURL: " "
|
|
// imgURL: " "
|
|
}
|
|
};
|
|
|
|
var paper;
|
|
var paperW;
|
|
var paperH;
|
|
|
|
var cellSize;
|
|
var nbCol;
|
|
var nbRows;
|
|
var xShift;
|
|
var yShift;
|
|
var threshold = 250; // cell above this value will be rounded to 255
|
|
|
|
var grid;
|
|
|
|
var imgURL;
|
|
|
|
var gridLineAttr = {
|
|
stroke: "grey",
|
|
"stroke-width": 1
|
|
};
|
|
var altGridLineAttr = {
|
|
stroke: "grey",
|
|
"stroke-width": 2
|
|
};
|
|
var cellAttr = {
|
|
stroke: "none",
|
|
// fill: "black"
|
|
};
|
|
|
|
subTask.loadLevel = function (curLevel) {
|
|
level = curLevel;
|
|
cellSize = data[level].cellSize;
|
|
nbCol = data[level].nbCol;
|
|
nbRows = data[level].nbRows;
|
|
imgURL = data[level].imgURL;
|
|
xShift = data[level].xShift;
|
|
yShift = data[level].yShift;
|
|
paperW = cellSize * nbCol;
|
|
paperH = cellSize * nbRows;
|
|
// rng = new RandomGenerator(subTask.taskParams.randomSeed);
|
|
};
|
|
|
|
subTask.getStateObject = function () {
|
|
return state;
|
|
};
|
|
|
|
subTask.reloadAnswerObject = function (answerObj) {
|
|
answer = answerObj;
|
|
if(answer){
|
|
// rng.reset(answer.seed);
|
|
}
|
|
};
|
|
|
|
subTask.resetDisplay = function () {
|
|
displayError("");
|
|
initPaper();
|
|
initCanvas();
|
|
displayHelper.customValidate = checkResult;
|
|
};
|
|
|
|
subTask.getAnswerObject = function () {
|
|
return answer;
|
|
};
|
|
|
|
subTask.getDefaultAnswerObject = function () {
|
|
var defaultAnswer = { grid: [], cellShade: [] };
|
|
for(var iRow = 0; iRow < nbRows; iRow++){
|
|
defaultAnswer.grid[iRow] = [];
|
|
defaultAnswer.cellShade[iRow] = [];
|
|
for(var iCol = 0; iCol < nbCol; iCol++){
|
|
defaultAnswer.grid[iRow][iCol] = 0;
|
|
defaultAnswer.cellShade[iRow][iCol] = 0;
|
|
}
|
|
}
|
|
return defaultAnswer;
|
|
};
|
|
|
|
function getResultAndMessage() {
|
|
var result = checkResult(true);
|
|
return result;
|
|
}
|
|
|
|
subTask.unloadLevel = function (callback) {
|
|
callback();
|
|
};
|
|
|
|
subTask.getGrade = function (callback) {
|
|
callback(getResultAndMessage());
|
|
};
|
|
|
|
function initPaper() {
|
|
paper = subTask.raphaelFactory.create("paper","paper",paperW,paperH);
|
|
$("#paper, #container").css("width", paperW);
|
|
initGrid();
|
|
};
|
|
|
|
function initGrid() {
|
|
if(level == "easy"){
|
|
var attr = gridLineAttr;
|
|
}else{
|
|
var attr = altGridLineAttr;
|
|
}
|
|
grid = new Grid.fromArray("paper",paper,answer.grid,cellFiller,cellSize,cellSize,0,0,attr);
|
|
grid.clickCell(clickCell);
|
|
};
|
|
|
|
function initCanvas() {
|
|
$("canvas").attr({"width":paperW,"height":paperH});
|
|
var ctx = $("canvas")[0].getContext("2d");
|
|
var img = new Image();
|
|
img.addEventListener('load', function() {
|
|
ctx.fillStyle = "white";
|
|
ctx.fillRect(0,0,paperW,paperH);
|
|
ctx.drawImage(img,xShift,yShift,paperW,paperH);
|
|
for(var iRow = 0; iRow < nbRows; iRow++){
|
|
for(var iCol = 0; iCol < nbCol; iCol++){
|
|
var x = iCol * cellSize;
|
|
var y = iRow * cellSize;
|
|
var imageData = ctx.getImageData(x,y,cellSize,cellSize);
|
|
var data = imageData.data;
|
|
var cellShade = 0;
|
|
for(var iPixel = 0; iPixel < data.length/4; iPixel++){
|
|
var index = iPixel*4;
|
|
if(level != "hard"){
|
|
var shade = (data[index] + data[index + 1] + data[index + 2])/3;
|
|
}else{
|
|
var shade = data[index];
|
|
}
|
|
cellShade += shade;
|
|
}
|
|
var average = cellShade/(data.length/4);
|
|
if(average < threshold || average == 255){
|
|
answer.cellShade[iRow][iCol] = average;
|
|
}else{
|
|
answer.cellShade[iRow][iCol] = 255;
|
|
ctx.fillRect(x, y,cellSize,cellSize);
|
|
}
|
|
}
|
|
}
|
|
}, false);
|
|
img.src = imgURL;
|
|
// console.log(answer)
|
|
};
|
|
|
|
function cellFiller(data,paper) {
|
|
var x = data.xPos;
|
|
var y = data.yPos;
|
|
var entry = data.entry;
|
|
var shade = 255*(1 - entry);
|
|
var fill = "rgb("+shade+","+shade+","+shade+")";
|
|
var rect = paper.rect(x,y,cellSize,cellSize).attr(cellAttr).attr("fill",fill);
|
|
if(!entry){
|
|
rect.attr("opacity",0);
|
|
}
|
|
return [rect];
|
|
};
|
|
|
|
function clickCell(ev) {
|
|
displayError("");
|
|
unhighlightAllCells();
|
|
var row = ev.data.row;
|
|
var col = ev.data.col;
|
|
if(level != "hard"){
|
|
answer.grid[row][col] = 1 - answer.grid[row][col];
|
|
}else{
|
|
answer.grid[row][col] = ((2*answer.grid[row][col] + 1)%3)/2;
|
|
}
|
|
grid.setCell(cellFiller,{row:row,col:col,entry:answer.grid[row][col]});
|
|
};
|
|
|
|
function unhighlightAllCells() {
|
|
for(var iRow = 0; iRow < nbRows; iRow++){
|
|
for(var iCol = 0; iCol < nbCol; iCol++){
|
|
if(grid.isCellHighlighted(iRow,iCol)){
|
|
grid.unhighlightCell(iRow,iCol);
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
function checkResult(noVisual) {
|
|
var error = null;
|
|
for(var iRow = 0; iRow < nbRows; iRow++){
|
|
for(var iCol = 0; iCol < nbCol; iCol++){
|
|
var cellShade = answer.cellShade[iRow][iCol];
|
|
var gridVal = answer.grid[iRow][iCol];
|
|
if(level != "hard"){
|
|
if(cellShade < 255 && gridVal == 0){
|
|
error = taskStrings.errorWhite;
|
|
}else if(cellShade == 255 && gridVal > 0){
|
|
error = taskStrings.errorBlack;
|
|
}
|
|
}else{
|
|
var whiteLimit = 2*255/3;
|
|
var blackLimit = 255/3;
|
|
if(cellShade < whiteLimit && gridVal == 0){
|
|
error = taskStrings.errorWhiteHard;
|
|
}else if(cellShade > blackLimit && gridVal == 1){
|
|
error = taskStrings.errorBlackHard;
|
|
}else if(cellShade > whiteLimit && gridVal == 0.5){
|
|
error = taskStrings.errorGreyWhite;
|
|
}else if(cellShade < blackLimit && gridVal == 0.5){
|
|
error = taskStrings.errorGreyBlack;
|
|
}
|
|
}
|
|
if(error){
|
|
if(!noVisual){
|
|
displayError(error);
|
|
grid.highlightCell(iRow,iCol);
|
|
}
|
|
return { successRate: 0, message: error };
|
|
}
|
|
|
|
}
|
|
}
|
|
if(!noVisual){
|
|
platform.validate("done");
|
|
}
|
|
return { successRate: 1, message: taskStrings.success };
|
|
};
|
|
|
|
function displayError(msg) {
|
|
$("#error").html(msg);
|
|
};
|
|
}
|
|
initWrapper(initTask, ["easy", "medium", "hard"]);
|
|
displayHelper.useFullWidth();
|