annotate test_create/test_core.js @ 2907:a1e294297380

#222 WIP
author Nicholas Jillings <n.g.r.jillings@se14.qmul.ac.uk>
date Mon, 31 Jul 2017 16:15:47 +0100
parents 22d1f6d9f0bf
children 9c26e579e64d
rev   line source
nicholas@2885 1 /* globals document, angular, window, Promise, XMLHttpRequest, Specification, XMLSerializer, Blob, DOMParser, FileReader, $*/
nicholas@2851 2 function get(url) {
nicholas@2851 3 // Return a new promise.
nicholas@2851 4 return new Promise(function (resolve, reject) {
nicholas@2851 5 // Do the usual XHR stuff
nicholas@2851 6 var req = new XMLHttpRequest();
nicholas@2851 7 req.open('GET', url);
nickjillings@1370 8
nicholas@2851 9 req.onload = function () {
nicholas@2851 10 // This is called even on 404 etc
nicholas@2851 11 // so check the status
nicholas@2851 12 if (req.status == 200) {
nicholas@2851 13 // Resolve the promise with the response text
nicholas@2851 14 resolve(req.response);
nicholas@2851 15 } else {
nicholas@2851 16 // Otherwise reject with the status text
nicholas@2851 17 // which will hopefully be a meaningful error
nicholas@2851 18 reject(Error(req.statusText));
nicholas@2851 19 }
nicholas@2851 20 };
nicholas@2851 21
nicholas@2851 22 // Handle network errors
nicholas@2851 23 req.onerror = function () {
nicholas@2851 24 reject(Error("Network Error"));
nicholas@2851 25 };
nicholas@2851 26
nicholas@2851 27 // Make the request
nicholas@2851 28 req.send();
nicholas@2851 29 });
nickjillings@1370 30 }
nickjillings@1370 31
nicholas@2851 32 var AngularInterface = angular.module("creator", []);
nickjillings@1370 33
nicholas@2851 34 var specification = new Specification();
nickjillings@1370 35
nicholas@2535 36 window.onload = function () {
nicholas@2851 37 // Get the test interface specifications
nicholas@2884 38 $(function () {
nicholas@2884 39 $('[data-toggle="popover"]').popover();
nicholas@2884 40 });
nicholas@2851 41 };
nicholas@2535 42
nicholas@2861 43 function handleFiles(event) {
nicholas@2861 44 var s = angular.element(event.currentTarget).scope();
nicholas@2861 45 s.handleFiles(event);
nicholas@2861 46 s.$apply();
nicholas@2861 47 }
nicholas@2861 48
nicholas@2851 49 AngularInterface.controller("view", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2851 50 $s.popupVisible = true;
nicholas@2864 51 $s.testSpecifications = {};
nicholas@2864 52
nicholas@2864 53 (function () {
nicholas@2864 54 new Promise(function (resolve, reject) {
nicholas@2864 55 var xml = new XMLHttpRequest();
nicholas@2864 56 xml.open("GET", "test_create/interfaces/specifications.json");
nicholas@2864 57 xml.onload = function () {
nicholas@2864 58 if (xml.status === 200) {
nicholas@2864 59 resolve(xml.responseText);
nicholas@2864 60 return;
nicholas@2864 61 }
nicholas@2864 62 reject(xml.status);
nicholas@2864 63 };
nicholas@2864 64 xml.onerror = function () {
nicholas@2864 65 reject(new Error("Network Error"));
nicholas@2864 66 };
nicholas@2864 67 xml.send();
nicholas@2864 68 }).then(JSON.parse).then(function (data) {
nicholas@2864 69 $s.testSpecifications = data;
nicholas@2864 70 $s.$apply();
nicholas@2865 71 });
nicholas@2864 72 })();
nickjillings@1370 73
nicholas@2851 74 $s.showPopup = function () {
nicholas@2851 75 $s.popupVisible = true;
nicholas@2851 76 };
nicholas@2851 77 $s.hidePopup = function () {
nicholas@2851 78 $s.popupVisible = false;
nicholas@2851 79 };
nicholas@2851 80 $s.globalSchema = undefined;
nicholas@2851 81 get("xml/test-schema.xsd").then(function (text) {
nicholas@2851 82 specification.processSchema(text);
nicholas@2851 83 $s.globalSchema = specification.getSchema();
nicholas@2851 84 });
nicholas@2851 85 $s.specification = specification;
nicholas@2886 86 $s.selectedTestPrototype = undefined;
nicholas@2886 87 $s.setTestPrototype = function (obj) {
nicholas@2886 88 $s.selectedTestPrototype = obj;
nicholas@2886 89 $w.specification.interface = obj.interface;
nicholas@2886 90 }
nicholas@2857 91
nicholas@2857 92 $s.addPage = function () {
nicholas@2857 93 $s.specification.createNewPage();
nicholas@2859 94 };
nicholas@2860 95
nicholas@2862 96 $s.removePage = function (page) {
nicholas@2862 97 var index = $s.specification.pages.findIndex(function (a) {
nicholas@2862 98 return a == page;
nicholas@2862 99 });
nicholas@2862 100 if (index === -1) {
nicholas@2862 101 throw ("Invalid Page");
nicholas@2862 102 }
nicholas@2862 103 $s.specification.pages.splice(index, 1);
nicholas@2862 104 };
nicholas@2862 105
nicholas@2860 106 $s.exportXML = function () {
nicholas@2860 107 var s = new XMLSerializer();
nicholas@2860 108 var doc = specification.encode();
nicholas@2891 109 var xmlstr = s.serializeToString(doc);
nicholas@2860 110 var bb = new Blob([s.serializeToString(doc)], {
nicholas@2860 111 type: 'application/xml'
nicholas@2860 112 });
nicholas@2860 113 var dnlk = window.URL.createObjectURL(bb);
nicholas@2891 114 var a = document.createElement("a");
nicholas@2891 115 a.href = dnlk;
nicholas@2891 116 a.download = "test.xml";
nicholas@2891 117 a.click();
nicholas@2891 118 window.URL.revokeObjectURL(dnlk);
nicholas@2861 119 };
nicholas@2887 120 $s.validated = false;
nicholas@2887 121 $s.showValidationMessages = false;
nicholas@2887 122 $s.validate = function () {
nicholas@2887 123 var s = new XMLSerializer();
nicholas@2887 124 var Module = {
nicholas@2887 125 xml: s.serializeToString(specification.encode()),
nicholas@2887 126 schema: specification.getSchemaString(),
nicholas@2887 127 arguments: ["--noout", "--schema", 'test-schema.xsd', 'document.xml']
nicholas@2887 128 };
nicholas@2887 129 var xmllint = validateXML(Module);
nicholas@2887 130 console.log(xmllint);
nicholas@2887 131 if (xmllint != 'document.xml validates\n') {
nicholas@2887 132 $s.validated = false;
nicholas@2887 133 var list = $e[0].querySelector("#validation-error-list");
nicholas@2887 134 while (list.firstChild) {
nicholas@2887 135 list.removeChild(list.firstChild);
nicholas@2887 136 }
nicholas@2887 137 var errors = xmllint.split('\n');
nicholas@2887 138 errors = errors.slice(0, errors.length - 2);
nicholas@2887 139 errors.forEach(function (str) {
nicholas@2887 140 var li = document.createElement("li");
nicholas@2887 141 li.textContent = str;
nicholas@2887 142 list.appendChild(li);
nicholas@2887 143 });
nicholas@2887 144 } else {
nicholas@2887 145 $s.validated = true;
nicholas@2887 146 }
nicholas@2887 147 $s.showValidationMessages = true;
nicholas@2887 148 }
nicholas@2887 149 $s.hideValidationMessages = function () {
nicholas@2887 150 $s.showValidationMessages = false;
nicholas@2887 151 }
nicholas@2851 152 }]);
nickjillings@1370 153
nicholas@2851 154 AngularInterface.controller("introduction", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2851 155 $s.state = 0;
nicholas@2851 156 $s.next = function () {
nicholas@2851 157 $s.state++;
nicholas@2861 158 if ($s.state > 1 || $s.file) {
nicholas@2851 159 $s.hidePopup();
nicholas@2851 160 }
nicholas@2851 161 };
nicholas@2851 162 $s.back = function () {
nicholas@2851 163 $s.state--;
nicholas@2851 164 };
nicholas@2851 165 $s.mouseover = function (name) {
nicholas@2864 166 var obj = $s.testSpecifications.interfaces.find(function (i) {
nicholas@2851 167 return i.name == name;
nicholas@2851 168 });
nicholas@2851 169 if (obj) {
nicholas@2851 170 $s.description = obj.description.en;
nicholas@2851 171 }
nicholas@2851 172 };
nicholas@2851 173 $s.initialise = function (name) {
nicholas@2885 174 var obj = $s.testSpecifications.interfaces.find(function (i) {
nicholas@2851 175 return i.name == name;
nicholas@2851 176 });
nicholas@2886 177 if (obj === undefined) {
nicholas@2886 178 throw ("Cannot find specification");
nicholas@2886 179 }
nicholas@2886 180 $s.setTestPrototype(obj);
nicholas@2851 181 };
nicholas@2851 182 // Get the test interface specifications
nicholas@2861 183 $s.file = undefined;
nicholas@2851 184 $s.description = "";
nicholas@2861 185
nicholas@2861 186 $s.handleFiles = function ($event) {
nicholas@2861 187 $s.file = $event.currentTarget.files[0];
nicholas@2861 188 var r = new FileReader();
nicholas@2861 189 r.onload = function () {
nicholas@2861 190 var p = new DOMParser();
nicholas@2861 191 specification.decode(p.parseFromString(r.result, "text/xml"));
nicholas@2861 192 $s.$apply();
nicholas@2862 193 };
nicholas@2861 194 r.readAsText($s.file);
nicholas@2861 195 };
nicholas@2851 196 }]);
nickjillings@1370 197
nicholas@2851 198 AngularInterface.controller("setup", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2851 199 function initialise() {
nicholas@2851 200 if ($s.globalSchema) {
nicholas@2851 201 $s.schema = $s.globalSchema.querySelector("[name=setup]");
nickjillings@1385 202 }
nickjillings@1370 203 }
nicholas@2851 204 $s.schema = undefined;
nicholas@2851 205 $s.attributes = [];
nickjillings@1370 206
nicholas@2851 207 $s.$watch("globalSchema", initialise);
nicholas@2853 208 $s.$watch("specification.metrics.enabled.length", function () {
nicholas@2853 209 var metricsNode = document.getElementById("metricsNode");
nicholas@2853 210 if (!$s.specification.metrics) {
nicholas@2853 211 return;
nicholas@2853 212 }
nicholas@2853 213 metricsNode.querySelectorAll("input").forEach(function (DOM) {
nicholas@2853 214 DOM.checked = false;
nicholas@2853 215 });
nicholas@2853 216 $s.specification.metrics.enabled.forEach(function (metric) {
nicholas@2853 217 var DOM = metricsNode.querySelector("[value=" + metric + "]");
nicholas@2853 218 if (DOM) {
nicholas@2853 219 DOM.checked = true;
nicholas@2853 220 }
nicholas@2853 221 });
nicholas@2855 222 });
nicholas@2853 223
nicholas@2853 224 $s.enableMetric = function ($event) {
nicholas@2853 225 var metric = $event.currentTarget.value;
nicholas@2853 226 var index = specification.metrics.enabled.findIndex(function (a) {
nicholas@2853 227 return a == metric;
nicholas@2853 228 });
nicholas@2853 229 if ($event.currentTarget.checked) {
nicholas@2853 230 if (index == -1) {
nicholas@2853 231 specification.metrics.enabled.push(metric);
nicholas@2853 232 }
nicholas@2853 233 } else {
nicholas@2853 234 if (index >= 0) {
nicholas@2853 235 specification.metrics.enabled.splice(index, 1);
nicholas@2853 236 }
nicholas@2853 237 }
nicholas@2855 238 };
nicholas@2886 239
nicholas@2886 240 $s.configure = function () {}
nicholas@2886 241
nicholas@2886 242 $s.$watch("selectedTestPrototype", $s.configure);
n@2907 243
n@2907 244 $s.placeholder = function (name) {
n@2907 245 var spec = $s.schema.querySelector("attribute[name=\"" + name + "\"]") || $w.specification.schema.querySelector("attribute[name=\"" + name + "\"]");
n@2907 246 var attr = spec.getAttribute("default");
n@2907 247 if (attr === undefined) {
n@2907 248 return "";
n@2907 249 }
n@2907 250 return attr;
n@2907 251 }
nicholas@2851 252 }]);
nicholas@2853 253
nicholas@2858 254 AngularInterface.controller("survey", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2858 255 $s.addSurveyEntry = function () {
nicholas@2858 256 $s.survey.addOption();
nicholas@2858 257 };
nicholas@2858 258 $s.removeSurveyEntry = function (entry) {
nicholas@2858 259 var index = $s.survey.options.findIndex(function (a) {
nicholas@2858 260 return a == entry;
nicholas@2858 261 });
nicholas@2858 262 if (index === -1) {
nicholas@2858 263 throw ("Invalid Entry");
nicholas@2858 264 }
nicholas@2858 265 $s.survey.options.splice(index, 1);
nicholas@2858 266 };
nicholas@2858 267 }]);
nicholas@2858 268
nicholas@2853 269 AngularInterface.controller("surveyOption", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2853 270
nicholas@2853 271 $s.removeOption = function (option) {
nicholas@2853 272 var index = $s.opt.options.findIndex(function (a) {
nicholas@2853 273 return a == option;
nicholas@2853 274 });
nicholas@2853 275 if (index === -1) {
nicholas@2853 276 throw ("Invalid option");
nicholas@2853 277 }
nicholas@2853 278 $s.opt.options.splice(index, 1);
nicholas@2853 279 };
nicholas@2853 280 $s.addOption = function () {
nicholas@2853 281 $s.opt.options.push({
nicholas@2853 282 name: "",
nicholas@2853 283 text: ""
nicholas@2853 284 });
nicholas@2855 285 };
nicholas@2857 286
nicholas@2857 287 $s.addCondition = function () {
nicholas@2857 288 $s.opt.conditions.push({
nicholas@2857 289 check: "equals",
nicholas@2857 290 value: "",
nicholas@2857 291 jumpToOnPass: undefined,
nicholas@2857 292 jumpToOnFail: undefined
nicholas@2857 293 });
nicholas@2857 294 };
nicholas@2857 295
nicholas@2857 296 $s.removeCondition = function (condition) {
nicholas@2857 297 var index = $s.opt.conditions.findIndex(function (c) {
nicholas@2857 298 return c == condition;
nicholas@2857 299 });
nicholas@2857 300 if (index === -1) {
nicholas@2857 301 throw ("Invalid Condition");
nicholas@2857 302 }
nicholas@2857 303 $s.opt.conditions.splice(index, 1);
nicholas@2857 304 };
nicholas@2855 305 }]);
nicholas@2855 306
nicholas@2855 307 AngularInterface.controller("interfaceNode", ['$scope', '$element', '$window', function ($s, $e, $w) {
nicholas@2855 308 $s.$watch("interface.options.length", function () {
nicholas@2855 309 if (!$s.interface || !$s.interface.options) {
nicholas@2855 310 return;
nicholas@2855 311 }
nicholas@2855 312 var options = $e[0].querySelector(".interfaceOptions").querySelectorAll(".attribute");
nicholas@2855 313 options.forEach(function (option) {
nicholas@2855 314 var name = option.getAttribute("name");
nicholas@2855 315 var index = $s.interface.options.findIndex(function (io) {
nicholas@2855 316 return io.name == name;
nicholas@2855 317 });
nicholas@2855 318 option.querySelector("input").checked = (index >= 0);
nicholas@2855 319 if (name == "scalerange" && index >= 0) {
nicholas@2855 320 option.querySelector("[name=min]").value = $s.interface.options[index].min;
nicholas@2855 321 option.querySelector("[name=max]").value = $s.interface.options[index].max;
nicholas@2855 322 }
nicholas@2855 323 });
nicholas@2855 324 });
nicholas@2855 325 $s.enableInterfaceOption = function ($event) {
nicholas@2855 326 var name = $event.currentTarget.parentElement.getAttribute("name");
nicholas@2855 327 var type = $event.currentTarget.parentElement.getAttribute("type");
nicholas@2855 328 var index = $s.interface.options.findIndex(function (io) {
nicholas@2855 329 return io.name == name;
nicholas@2855 330 });
nicholas@2855 331 if (index == -1 && $event.currentTarget.checked) {
nicholas@2855 332 var obj = $s.interface.options.push({
nicholas@2855 333 name: name,
nicholas@2855 334 type: type
nicholas@2855 335 });
nicholas@2855 336 if (name == "scalerange") {
nicholas@2855 337 obj.min = $event.currentTarget.parentElement.querySelector("[name=min]").value;
nicholas@2855 338 obj.max = $event.currentTarget.parentElement.querySelector("[name=max]").value;
nicholas@2855 339 }
nicholas@2855 340 } else if (index >= 0 && !$event.currentTarget.checked) {
nicholas@2855 341 $s.interface.options.splice(index, 1);
nicholas@2855 342 }
nicholas@2857 343 };
nicholas@2886 344 $s.scales = [];
nicholas@2857 345 $s.removeScale = function (scale) {
nicholas@2857 346 var index = $s.interface.scales.findIndex(function (s) {
nicholas@2857 347 return s == scale;
nicholas@2857 348 });
nicholas@2857 349 if (index >= 0) {
nicholas@2857 350 $s.interface.scales.splice(index, 1);
nicholas@2857 351 }
nicholas@2857 352 };
nicholas@2857 353 $s.addScale = function () {
nicholas@2857 354 $s.interface.scales.push({
nicholas@2857 355 position: undefined,
nicholas@2857 356 text: undefined
nicholas@2857 357 });
nicholas@2857 358 };
nicholas@2864 359 $s.clearScales = function () {
nicholas@2864 360 $s.interface.scales = [];
nicholas@2864 361 };
nicholas@2876 362 $s.useScales = function (scale) {
nicholas@2864 363 $s.clearScales();
nicholas@2876 364 scale.scales.forEach(function (s) {
nicholas@2864 365 $s.interface.scales.push(s);
nicholas@2864 366 });
nicholas@2886 367 $s.selectedScale = scale.name;
nicholas@2864 368 };
nicholas@2886 369 $s.selectedScale = undefined;
nicholas@2886 370
nicholas@2886 371 $s.configure = function () {
nicholas@2886 372 if ($s.selectedTestPrototype === undefined) {
nicholas@2886 373 return;
nicholas@2886 374 }
nicholas@2886 375 if ($s.selectedTestPrototype.checks && $s.selectedTestPrototype.checks.length >= 1) {
nicholas@2886 376 $s.selectedTestPrototype.checks.forEach(function (entry) {
nicholas@2886 377 var dom = $e[0].querySelector("[name=\"" + entry.name + "\"] input");
nicholas@2886 378 if (entry.support == "none") {
nicholas@2886 379 dom.checked = false;
nicholas@2886 380 dom.disabled = true;
nicholas@2886 381 }
nicholas@2886 382 });
nicholas@2886 383 }
nicholas@2886 384 if ($s.selectedTestPrototype.show && $s.selectedTestPrototype.show.length >= 1) {
nicholas@2886 385 $s.selectedTestPrototype.show.forEach(function (entry) {
nicholas@2886 386 var dom = $e[0].querySelector("[name=\"" + entry.name + "\"] input");
nicholas@2886 387 if (entry.support == "none") {
nicholas@2886 388 dom.checked = false;
nicholas@2886 389 dom.disabled = true;
nicholas@2886 390 }
nicholas@2886 391 });
nicholas@2886 392 }
nicholas@2886 393 if ($s.interface !== specification.interfaces) {
nicholas@2886 394 // Page specific interface nodes
nicholas@2886 395 if ($s.selectedTestPrototype.hasScales !== undefined && ($s.selectedTestPrototype.hasScales == "false" || $s.selectedTestPrototype.hasScales == false)) {
nicholas@2886 396 var elem = $e[0].querySelector("[name=\"scale-selection\"]")
nicholas@2886 397 elem.style.visibility = "hidden";
nicholas@2886 398 elem.style.height = "0px";
nicholas@2886 399 }
nicholas@2886 400 if ($s.selectedTestPrototype.scales && $s.selectedTestPrototype.show.length >= 1) {
nicholas@2886 401 $s.scales = [];
nicholas@2886 402 $s.selectedTestPrototype.scales.forEach(function (scalename) {
nicholas@2886 403 var obj = $s.testSpecifications.scales.find(function (a) {
nicholas@2886 404 return a.name == scalename;
nicholas@2886 405 });
nicholas@2886 406 $s.scales.push(obj);
nicholas@2886 407 });
nicholas@2886 408 if ($s.selectedTestPrototype.scales.includes($s.selectedScale) == false) {
nicholas@2886 409 $s.clearScales();
nicholas@2886 410 }
nicholas@2886 411 if ($s.scales.length == 1) {
nicholas@2886 412 $s.clearScales();
nicholas@2886 413 $s.useScales($s.scales[0]);
nicholas@2886 414 }
nicholas@2886 415 } else {
nicholas@2886 416 $s.scales = $s.testSpecifications.scales;
nicholas@2886 417 }
nicholas@2886 418 }
nicholas@2886 419 };
nicholas@2886 420
nicholas@2886 421 $s.$watch("selectedTestPrototype", $s.configure);
nicholas@2886 422 $s.configure();
nicholas@2853 423 }]);
nicholas@2859 424 AngularInterface.controller("page", ['$scope', '$element', '$window', function ($s, $e, $w) {
n@2907 425 $s.schema = $w.specification.schema.querySelector("element[name=\"page\"]");
n@2907 426 $s.page.label = "default";
nicholas@2859 427 $s.addInterface = function () {
nicholas@2859 428 $s.page.addInterface();
nicholas@2859 429 };
nicholas@2859 430 $s.removeInterface = function (node) {
nicholas@2859 431 var index = $s.page.interfaces.findIndex(function (a) {
nicholas@2859 432 return a == node;
nicholas@2859 433 });
nicholas@2859 434 if (index === -1) {
nicholas@2859 435 throw ("Invalid node");
nicholas@2859 436 }
nicholas@2859 437 $s.page.interfaces.splice(index, 1);
nicholas@2859 438 };
nicholas@2859 439
nicholas@2859 440 $s.addCommentQuestion = function () {
nicholas@2859 441 $s.page.addCommentQuestion();
nicholas@2859 442 };
nicholas@2859 443 $s.removeCommentQuestion = function (node) {
nicholas@2859 444 var index = $s.page.commentQuestions.findIndex(function (a) {
nicholas@2859 445 return a == node;
nicholas@2859 446 });
nicholas@2859 447 if (index === -1) {
nicholas@2859 448 throw ("Invalid node");
nicholas@2859 449 }
nicholas@2859 450 $s.page.commentQuestions.splice(index, 1);
nicholas@2859 451 };
nicholas@2859 452 $s.addAudioElement = function () {
nicholas@2859 453 $s.page.addAudioElement();
nicholas@2859 454 };
nicholas@2859 455 $s.removeAudioElement = function (element) {
nicholas@2859 456 var index = $s.page.audioElements.findIndex(function (a) {
nicholas@2859 457 return a == element;
nicholas@2859 458 });
nicholas@2859 459 if (index === -1) {
nicholas@2859 460 throw ("Invalid node");
nicholas@2859 461 }
nicholas@2859 462 $s.page.audioElements.splice(index, 1);
nicholas@2859 463 };
n@2907 464
n@2907 465 $s.placeholder = function (name) {
n@2907 466 var spec = $s.schema.querySelector("attribute[name=\"" + name + "\"]") || $w.specification.schema.querySelector("attribute[name=\"" + name + "\"]");
n@2907 467 var attr = spec.getAttribute("default");
n@2907 468 if (attr === undefined) {
n@2907 469 return "";
n@2907 470 }
n@2907 471 return attr;
n@2907 472 }
nicholas@2859 473 }]);