diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/test/renders/solutions/graphics-rect.json b/test/renders/solutions/graphics-rect.json
new file mode 100644
index 0000000..d51ebd2
--- /dev/null
+++ b/test/renders/solutions/graphics-rect.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ }
+}
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/test/renders/solutions/graphics-rect.json b/test/renders/solutions/graphics-rect.json
new file mode 100644
index 0000000..d51ebd2
--- /dev/null
+++ b/test/renders/solutions/graphics-rect.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/solutions/sprite-new.json b/test/renders/solutions/sprite-new.json
new file mode 100644
index 0000000..c5396a6
--- /dev/null
+++ b/test/renders/solutions/sprite-new.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ }
+}
\ No newline at end of file
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/test/renders/solutions/graphics-rect.json b/test/renders/solutions/graphics-rect.json
new file mode 100644
index 0000000..d51ebd2
--- /dev/null
+++ b/test/renders/solutions/graphics-rect.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/solutions/sprite-new.json b/test/renders/solutions/sprite-new.json
new file mode 100644
index 0000000..c5396a6
--- /dev/null
+++ b/test/renders/solutions/sprite-new.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/tests/assets/bitmap-1.png b/test/renders/tests/assets/bitmap-1.png
new file mode 100644
index 0000000..1284fb8
--- /dev/null
+++ b/test/renders/tests/assets/bitmap-1.png
Binary files differ
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/test/renders/solutions/graphics-rect.json b/test/renders/solutions/graphics-rect.json
new file mode 100644
index 0000000..d51ebd2
--- /dev/null
+++ b/test/renders/solutions/graphics-rect.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/solutions/sprite-new.json b/test/renders/solutions/sprite-new.json
new file mode 100644
index 0000000..c5396a6
--- /dev/null
+++ b/test/renders/solutions/sprite-new.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/tests/assets/bitmap-1.png b/test/renders/tests/assets/bitmap-1.png
new file mode 100644
index 0000000..1284fb8
--- /dev/null
+++ b/test/renders/tests/assets/bitmap-1.png
Binary files differ
diff --git a/test/renders/tests/graphics-rect.js b/test/renders/tests/graphics-rect.js
new file mode 100644
index 0000000..1143c96
--- /dev/null
+++ b/test/renders/tests/graphics-rect.js
@@ -0,0 +1,22 @@
+'use strict';
+
+module.exports = function ()
+{
+ const graphic = new PIXI.Graphics()
+ .beginFill(0xFFCC00, 1)
+ .drawRect(8, 8, 16, 16);
+
+ expect(graphic.width).to.equal(16);
+ expect(graphic.height).to.equal(16);
+ expect(graphic.x).to.equal(0);
+ expect(graphic.y).to.equal(0);
+
+ const bounds = graphic.getBounds();
+
+ expect(bounds.x).to.equal(8);
+ expect(bounds.y).to.equal(8);
+ expect(bounds.width).to.equal(16);
+ expect(bounds.height).to.equal(16);
+
+ this.stage.addChild(graphic);
+};
diff --git a/.eslintignore b/.eslintignore
new file mode 100644
index 0000000..ef2050a
--- /dev/null
+++ b/.eslintignore
@@ -0,0 +1 @@
+scripts/renders/node_modules/*
\ No newline at end of file
diff --git a/package.json b/package.json
index 94140a4..8cc41a6 100644
--- a/package.json
+++ b/package.json
@@ -25,6 +25,8 @@
"watch:lint": "watch \"eslint scripts src test || exit 0\" src",
"test": "floss --path test/index.js",
"test:debug": "npm test -- --debug",
+ "prerenders": "npm --prefix scripts/renders i scripts/renders",
+ "renders": "electron scripts/renders",
"precoverage": "rimraf coverage && npm run build -- --noExternal",
"coverage": "npm test -- -c bin/pixi.js -s -h",
"lint": "eslint scripts src test --max-warnings 0",
@@ -66,6 +68,7 @@
"floss": "^1.2.0",
"gh-pages": "^0.11.0",
"jaguarjs-jsdoc": "^1.0.1",
+ "js-md5": "^0.4.1",
"jsdoc": "^3.4.2",
"minimist": "^1.2.0",
"mkdirp": "^0.5.1",
diff --git a/scripts/renders/.eslintrc.json b/scripts/renders/.eslintrc.json
new file mode 100644
index 0000000..5a103d2
--- /dev/null
+++ b/scripts/renders/.eslintrc.json
@@ -0,0 +1,6 @@
+{
+ "globals": {
+ "document": false,
+ "PIXI": false
+ }
+}
\ No newline at end of file
diff --git a/scripts/renders/client.css b/scripts/renders/client.css
new file mode 100755
index 0000000..b544283
--- /dev/null
+++ b/scripts/renders/client.css
@@ -0,0 +1,145 @@
+body {
+ background:#ccc;
+ font-family:Arial, sans-serif;
+ margin: 0 0 0 0;
+ padding: 20px;
+ width: 100%;
+ height: 100%;
+ position: absolute;
+ overflow: hidden;
+ box-sizing: border-box;
+ transition:background-color 0.3s;
+}
+body, body * {
+ user-select:none;
+ -webkit-user-select:none;
+ cursor:default;
+}
+body.dragging {
+ background-color:#ddd;
+}
+body.dragging #solution {
+ background-color: #fff;
+}
+#frame {
+ margin:0 auto;
+ max-width:640px;
+ width:100%;
+}
+.hide {
+ display:none;
+}
+#solution {
+ border:1px solid #bbb;
+ padding:10px;
+ border-radius:6px;
+ box-sizing:border-box;
+ transition:background-color 0.3s;
+ font-size:10px;
+ font-family: monospace;
+ background:#ddd;
+ color:#666;
+ height:240px;
+ width:403px;
+ margin-bottom:20px;
+ resize:none;
+ white-space:pre;
+ line-height: 1.4;
+ overflow-x:hidden;
+ overflow-y: auto;
+ user-select:initial;
+ -webkit-user-select:initial;
+}
+#solution:focus, textarea:focus {
+ outline:0;
+ border-color:#4993EF;
+}
+#solution.copied {
+ position:relative;
+ border-color:#4993EF;
+}
+#solution.copied:before {
+ display:block;
+ position:absolute;
+ top:0;
+ right:10px;
+ padding:8px 16px;
+ color:#fff;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ font-size:8px;
+ border-bottom-right-radius: 5px;
+ border-bottom-left-radius: 5px;
+ font-family: Arial, sans-serif;
+ content: 'Copied To Clipboard';
+ background-color:#4993EF;
+}
+label {
+ display:inline-block;
+ padding:15px 0 10px;
+ font-size: 10px;
+ color: #666;
+ text-transform: uppercase;
+ letter-spacing: 0.2em;
+ border-radius: 6px;
+ background-color: rgba(0,0,0,0.1);
+ padding: 4px 8px;
+ color: #fff;
+ margin-bottom:10px;
+}
+button {
+ padding:10px 20px;
+ border: 1px solid #3267A9;
+ background-color: #4993EF;
+ color: #fff;
+ font-size:14px;
+ border-radius:6px;
+ cursor:pointer;
+ display:block;
+ width:100%;
+ transition:background 0.3s;
+ box-shadow: inset 0px 1px 0px 0px rgba(255,255,255,0.5);
+ outline:0;
+}
+button:hover{
+ background-color: #5CA5FF;
+}
+button:active {
+ background-color: #3267A9;
+ box-shadow: none;
+}
+button.disabled {
+ pointer-events:none;
+ background-color:#999;
+ border-color:#777;
+ box-shadow: none;
+ color:#ddd;
+}
+.cols {
+ display:flex;
+ margin-top:20px;
+}
+.cols > .preview {
+ width:100px;
+ margin-right:20px;
+}
+.cols > .output {
+ width:100%;
+}
+.full {
+ width:100%;
+}
+.center {
+ text-align:center;
+}
+.view {
+ image-rendering: optimizeSpeed;
+ image-rendering: -moz-crisp-edges;
+ image-rendering: -o-crisp-edges;
+ image-rendering: -webkit-crisp-edges;
+ image-rendering: crisp-edges;
+ image-rendering: -webkit-optimize-contrast;
+ image-rendering: pixelated;
+ -ms-interpolation-mode: nearest-neighbor;
+ zoom:3;
+}
\ No newline at end of file
diff --git a/scripts/renders/client.js b/scripts/renders/client.js
new file mode 100755
index 0000000..16c9565
--- /dev/null
+++ b/scripts/renders/client.js
@@ -0,0 +1,135 @@
+'use strict';
+
+require('../../bin/pixi');
+PIXI.utils.skipHello();
+
+const fs = require('fs');
+const path = require('path');
+const electron = require('electron');
+const remote = electron.remote;
+const clipboard = electron.clipboard;
+const dialog = remote.dialog;
+const Renderer = require('../../test/renders/lib/Renderer');
+const Droppable = require('./droppable');
+const $ = document.querySelector.bind(document);
+
+// Feature parity with floss: chai, sinon, sinon-chai
+const chai = require('chai');
+const sinon = require('sinon');
+const sinonChai = require('sinon-chai');
+
+global.chai = chai;
+global.sinon = sinon;
+global.should = chai.should;
+global.assert = chai.assert;
+global.expect = chai.expect;
+global.chai.use(sinonChai);
+
+// Create the renderer to convert the json into
+// a JSON solution which can be used by testing
+const renderer = new Renderer(
+ $('#view-webgl'),
+ $('#view-canvas')
+);
+
+// Clicking on the solution copies it to the clipboard
+const solution = $('#solution');
+
+// Current filename
+let currentName;
+let currentPath;
+
+// drag-n-drop
+new Droppable($('body'), (err, file) => // eslint-disable-line no-new
+{
+ if (err)
+ {
+ dialog.showErrorBox('Drop Error', err.message);
+ }
+ else if (!(/\.js$/).test(file))
+ {
+ dialog.showErrorBox('Invalid Filetype', 'The specified file must be a ".js" file.');
+ }
+ else
+ {
+ open(file);
+ }
+});
+
+// Click on the chooser button to browser for file
+$('#choose').addEventListener('click', () =>
+{
+ dialog.showOpenDialog({
+ filters: [{
+ name: 'JavaScript',
+ extensions: ['js'],
+ }],
+ },
+ (fileNames) =>
+ {
+ if (fileNames)
+ {
+ open(fileNames[0]);
+ }
+ });
+});
+
+const save = $('#save');
+
+save.addEventListener('click', () =>
+{
+ dialog.showSaveDialog({
+ title: 'Save Solution',
+ defaultPath: path.join(currentPath, `${currentName}.json`),
+ filters: [{
+ name: 'JSON',
+ extensions: ['json'],
+ }],
+ }, (fileName) =>
+ {
+ if (fileName)
+ {
+ fs.writeFileSync(fileName, solution.innerHTML);
+ }
+ });
+});
+
+let copiedId = null;
+
+solution.addEventListener('click', () =>
+{
+ if (solution.innerHTML)
+ {
+ clipboard.writeText(solution.innerHTML);
+ solution.className = 'copied';
+ if (copiedId)
+ {
+ clearTimeout(copiedId);
+ copiedId = null;
+ }
+ copiedId = setTimeout(() =>
+ {
+ solution.className = '';
+ copiedId = null;
+ }, 3000);
+ }
+});
+
+function open(file)
+{
+ save.className = 'disabled';
+ renderer.run(file, (err, result) =>
+ {
+ if (err)
+ {
+ dialog.showErrorBox('Invalid Output Format', err.message);
+ }
+ else
+ {
+ solution.innerHTML = JSON.stringify(result, null, ' ');
+ save.className = '';
+ currentName = path.basename(file, '.js');
+ currentPath = path.dirname(file);
+ }
+ });
+}
diff --git a/scripts/renders/droppable.js b/scripts/renders/droppable.js
new file mode 100644
index 0000000..b5db16b
--- /dev/null
+++ b/scripts/renders/droppable.js
@@ -0,0 +1,67 @@
+'use strict';
+
+/**
+ * Add drag-n-drop handling
+ * @class Droppable
+ */
+class Droppable
+{
+ /**
+ * @constructor
+ * @param {HTMLElement} node Element to listen on
+ * @param {Function} callback Callback when complete
+ */
+ constructor(node, callback)
+ {
+ node.addEventListener('dragenter', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = Droppable.CLASS_NAME;
+ });
+
+ node.addEventListener('dragover', (ev) =>
+ {
+ ev.preventDefault();
+ if (node.className !== Droppable.CLASS_NAME)
+ {
+ node.className = Droppable.CLASS_NAME;
+ }
+ });
+
+ node.addEventListener('dragleave', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ });
+
+ node.addEventListener('drop', (ev) =>
+ {
+ ev.preventDefault();
+ node.className = '';
+ const fileList = ev.dataTransfer.files;
+
+ if (fileList.length > 1)
+ {
+ callback(new Error('Only one file at a time.'));
+ }
+ else
+ {
+ const file = fileList[0];
+
+ callback(null, file.path);
+ }
+ });
+ }
+
+ /**
+ * The name of the class to add to the HTML node
+ * @static
+ * @property {String} CLASS_NAME
+ */
+ static get CLASS_NAME()
+ {
+ return 'dragging';
+ }
+}
+
+module.exports = Droppable;
diff --git a/scripts/renders/index.html b/scripts/renders/index.html
new file mode 100755
index 0000000..bc95d9a
--- /dev/null
+++ b/scripts/renders/index.html
@@ -0,0 +1,35 @@
+
+
+
+
+
+ Pixi Tests Tool
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
\ No newline at end of file
diff --git a/scripts/renders/main.js b/scripts/renders/main.js
new file mode 100644
index 0000000..7ee1a1c
--- /dev/null
+++ b/scripts/renders/main.js
@@ -0,0 +1,33 @@
+'use strict';
+
+const electron = require('electron');
+const app = electron.app; // Control application life
+const BrowserWindow = electron.BrowserWindow; // Native browser window
+
+// Keep a global reference of the window object
+let mainWindow = null;
+
+// Quit when all windows are closed.
+app.on('window-all-closed', () =>
+{
+ app.quit();
+});
+
+// This method will be called when Electron has finished
+// initialization and is ready to create browser windows.
+app.on('ready', () =>
+{
+ mainWindow = new BrowserWindow({
+ width: 675,
+ height: 400,
+ useContentSize: true,
+ resizable: false,
+ textAreasAreResizable: false,
+ });
+
+ mainWindow.loadURL(`file://${__dirname}/index.html`);
+ mainWindow.on('closed', () =>
+ {
+ mainWindow = null;
+ });
+});
diff --git a/scripts/renders/package.json b/scripts/renders/package.json
new file mode 100644
index 0000000..72753f3
--- /dev/null
+++ b/scripts/renders/package.json
@@ -0,0 +1,10 @@
+{
+ "name": "renders-app",
+ "private": true,
+ "main": "main.js",
+ "dependencies": {
+ "chai": "^3.5.0",
+ "sinon": "^1.17.6",
+ "sinon-chai": "^2.8.0"
+ }
+}
diff --git a/test/.eslintrc.json b/test/.eslintrc.json
index 09620cb..e03220d 100644
--- a/test/.eslintrc.json
+++ b/test/.eslintrc.json
@@ -11,6 +11,7 @@
"globals": {
"sinon": false,
"expect": false,
+ "assert": false,
"PIXI": false
},
"rules": {
diff --git a/test/index.js b/test/index.js
index 1d245bc..7424430 100755
--- a/test/index.js
+++ b/test/index.js
@@ -3,6 +3,8 @@
/* eslint-disable global-require */
require('../bin/pixi');
+PIXI.utils.skipHello(); // hide banner
+
describe('PIXI', function ()
{
it('should exist as a global object', function ()
@@ -11,4 +13,5 @@
});
require('./core');
require('./interaction');
+ require('./renders');
});
diff --git a/test/renders/index.js b/test/renders/index.js
new file mode 100644
index 0000000..cf9cbe2
--- /dev/null
+++ b/test/renders/index.js
@@ -0,0 +1,49 @@
+'use strict';
+
+const Renderer = require('./lib/Renderer');
+const path = require('path');
+
+describe('renders', function ()
+{
+ before(function ()
+ {
+ this.renderer = new Renderer();
+ this.validate = function (id, done)
+ {
+ const code = path.join(__dirname, 'tests', `${id}.js`);
+ const solution = path.join(__dirname, 'solutions', `${id}.json`);
+
+ this.renderer.compare(code, solution, (err, success) =>
+ {
+ if (err)
+ {
+ assert(false, err.message);
+ }
+ assert(success, 'Render not successful');
+ done();
+ });
+ };
+ });
+
+ beforeEach(function ()
+ {
+ this.renderer.clear();
+ });
+
+ after(function ()
+ {
+ this.renderer.destroy();
+ this.renderer = null;
+ this.validate = null;
+ });
+
+ it('should draw a rectangle', function (done)
+ {
+ this.validate('graphics-rect', done);
+ });
+
+ it('should draw a sprite', function (done)
+ {
+ this.validate('sprite-new', done);
+ });
+});
diff --git a/test/renders/lib/ImageDiff.js b/test/renders/lib/ImageDiff.js
new file mode 100644
index 0000000..6bf2b54
--- /dev/null
+++ b/test/renders/lib/ImageDiff.js
@@ -0,0 +1,77 @@
+'use strict';
+
+/**
+ * Compare images
+ * @class ImageDiff
+ */
+class ImageDiff
+{
+ /**
+ * @constructor
+ * @param {int} width Width of the canvas
+ * @param {int} height Height of the canvas
+ * @param {Number} tolerance Tolerance for difference
+ */
+ constructor(width, height, tolerance)
+ {
+ this.width = width;
+ this.height = height;
+ this.tolerance = tolerance;
+
+ const canvas = document.createElement('canvas');
+
+ canvas.width = width;
+ canvas.height = height;
+ this.context = canvas.getContext('2d', {
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * Compare two base64 images
+ * @method compare
+ * @param {string} src1 Canvas source
+ * @param {string} src2 Canvas source
+ * @return {Boolean} If images are within tolerance
+ */
+ compare(src1, src2)
+ {
+ const a = this.getImageData(src1);
+ const b = this.getImageData(src2);
+ const len = a.length;
+ const tolerance = this.tolerance;
+
+ const diff = a.filter(function (val, i)
+ {
+ return Math.abs(val - b[i]) / 255 > tolerance;
+ });
+
+ if (diff.length / len > tolerance)
+ {
+ return false;
+ }
+
+ return true;
+ }
+
+ /**
+ * Get an array of pixels
+ * @method getImageData
+ * @param {string} src Source of the image
+ * @return {Uint8ClampedArray} Data for image of pixels
+ */
+ getImageData(src)
+ {
+ const image = new Image();
+
+ image.src = src;
+ this.context.clearRect(0, 0, this.width, this.height);
+ this.context.drawImage(image, 0, 0, this.width, this.height);
+ const imageData = this.context.getImageData(0, 0, this.width, this.height);
+
+ return imageData.data;
+ }
+}
+
+module.exports = ImageDiff;
diff --git a/test/renders/lib/Renderer.js b/test/renders/lib/Renderer.js
new file mode 100644
index 0000000..d60217f
--- /dev/null
+++ b/test/renders/lib/Renderer.js
@@ -0,0 +1,302 @@
+'use strict';
+
+const md5 = require('js-md5');
+const path = require('path');
+const ImageDiff = require('./ImageDiff');
+
+/**
+ * Class to create and validate solutions.
+ * @class Renderer
+ */
+class Renderer
+{
+ /**
+ * @constructor
+ * @param {HTMLCanvasElement} [viewWebGL] Optional canvas element for webgl
+ * @param {HTMLCanvasElement} [viewContext2d] Optional canvas element for context2d
+ * @param {HTMLElement} [parentNode] container, defaults to `document.body`
+ */
+ constructor(viewWebGL, viewContext2d, parentNode)
+ {
+ viewWebGL = viewWebGL || document.createElement('canvas');
+ viewContext2d = viewContext2d || document.createElement('canvas');
+
+ /**
+ * The container node to add the canvas elements.
+ * @name parentNode
+ * @type {HTMLElement}
+ */
+ this.parentNode = parentNode || document.body;
+
+ if (!viewWebGL.parentNode)
+ {
+ this.parentNode.appendChild(viewWebGL);
+ }
+ if (!viewContext2d.parentNode)
+ {
+ this.parentNode.appendChild(viewContext2d);
+ }
+
+ /**
+ * The container for the display objects.
+ * @name stage
+ * @type {PIXI.Container}
+ */
+ this.stage = new PIXI.Container();
+
+ /**
+ * If the current browser supports WebGL.
+ * @name hasWebGL
+ * @type {Boolean}
+ */
+ this.hasWebGL = PIXI.utils.isWebGLSupported();
+
+ if (this.hasWebGL)
+ {
+ /**
+ * The WebGL PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.webgl = new PIXI.WebGLRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewWebGL,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ });
+ }
+
+ /**
+ * The Canvas PIXI renderer.
+ * @name webgl
+ * @type {PIXI.WebGLRenderer}
+ */
+ this.canvas = new PIXI.CanvasRenderer(Renderer.WIDTH, Renderer.HEIGHT, {
+ view: viewContext2d,
+ backgroundColor: 0xffffff,
+ antialias: false,
+ preserveDrawingBuffer: true,
+ roundPixels: true,
+ });
+ this.canvas.smoothProperty = null;
+ this.render();
+
+ /**
+ * Library for comparing images diffs.
+ * @name imagediff
+ * @type {ImageDiff}
+ */
+ this.imagediff = new ImageDiff(
+ Renderer.WIDTH,
+ Renderer.HEIGHT,
+ Renderer.TOLERANCE
+ );
+ }
+
+ /**
+ * Rerender the stage
+ * @method render
+ */
+ render()
+ {
+ if (this.hasWebGL)
+ {
+ this.webgl.render(this.stage);
+ }
+ this.canvas.render(this.stage);
+ }
+
+ /**
+ * Clear the stage
+ * @method clear
+ */
+ clear()
+ {
+ this.stage.children.forEach(function (child)
+ {
+ child.destroy(true);
+ });
+ this.render();
+ }
+
+ /**
+ * Run the solution renderer
+ * @method run
+ * @param {String} filePath Fully resolved path to javascript.
+ * @param {Function} callback Takes error and result as arguments.
+ */
+ run(filePath, callback)
+ {
+ delete require.cache[path.resolve(filePath)];
+ let data;
+ const code = require(filePath); // eslint-disable-line global-require
+
+ if (typeof code !== 'function' && !code.async)
+ {
+ callback(new Error('Invalid JS format, make sure file is '
+ + 'CommonJS compatible, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ this.clear();
+ const done = () =>
+ {
+ if (!this.stage.children.length)
+ {
+ callback(new Error('Stage has no children, make sure to '
+ + 'add children in your test, e.g. "module.exports = function(){};"'));
+ }
+ else
+ {
+ // Generate the result
+ const result = {
+ webgl: {},
+ canvas: {},
+ };
+
+ this.render();
+ if (this.hasWebGL)
+ {
+ data = this.webgl.view.toDataURL();
+ result.webgl.image = data;
+ result.webgl.hash = md5(data);
+ }
+ data = this.canvas.view.toDataURL();
+ result.canvas.image = data;
+ result.canvas.hash = md5(data);
+
+ callback(null, result);
+ }
+ };
+
+ // Support for asynchronous tests
+ if (code.async)
+ {
+ code.async.call(this, done);
+ }
+ else
+ {
+ // Just run the test synchronously
+ code.call(this);
+ done();
+ }
+ }
+ }
+
+ /**
+ * Compare a file with a solution.
+ * @method compare
+ * @param {String} filePath The file to load.
+ * @param {String} solutionPath The path to the solution file.
+ * @param {Array} solution.webgl Solution for webgl
+ * @param {Array} solution.canvas Solution for canvas
+ * @param {Function} callback Complete callback, takes err as an error and success boolean as args.
+ */
+ compare(filePath, solutionPath, callback)
+ {
+ this.run(filePath, (err, result) =>
+ {
+ const solution = require(solutionPath); // eslint-disable-line global-require
+
+ if (!solution.webgl || !solution.canvas)
+ {
+ callback(new Error('Invalid solution'));
+ }
+ else if (err)
+ {
+ callback(err);
+ }
+ else if (this.hasWebGL && !this.compareResult(solution.webgl, result.webgl))
+ {
+ callback(new Error('WebGL results do not match.'));
+ }
+ else if (!this.compareResult(solution.canvas, result.canvas))
+ {
+ callback(new Error('Canvas results do not match.'));
+ }
+ else
+ {
+ callback(null, true);
+ }
+ });
+ }
+
+ /**
+ * Compare two arrays of frames
+ * @method compareResult
+ * @private
+ * @param {Array} a First result to compare
+ * @param {Array} b Second result to compare
+ * @return {Boolean} If we're equal
+ */
+ compareResult(a, b)
+ {
+ if (a === b)
+ {
+ return true;
+ }
+ if (a === null || b === null)
+ {
+ return false;
+ }
+ if (a.hash !== b.hash)
+ {
+ if (!this.imagediff.compare(a.image, b.image))
+ {
+ return false;
+ }
+ }
+
+ return true;
+ }
+
+ /**
+ * Destroy and don't use after this.
+ * @method destroy
+ */
+ destroy()
+ {
+ this.clear();
+ this.stage.destroy(true);
+ this.parentNode = null;
+ this.canvas.destroy(true);
+ this.canvas = null;
+
+ if (this.hasWebGL)
+ {
+ this.webgl.destroy(true);
+ this.webgl = null;
+ }
+ }
+}
+
+/**
+ * The width of the render.
+ * @static
+ * @type {int}
+ * @name WIDTH
+ * @default 32
+ */
+Renderer.WIDTH = 32;
+
+/**
+ * The height of the render.
+ * @static
+ * @type {int}
+ * @name HEIGHT
+ * @default 32
+ */
+Renderer.HEIGHT = 32;
+
+/**
+ * The tolerance when comparing image solutions.
+ * for instance 0.01 would mean any difference greater
+ * than 1% would be consider not the same.
+ * @static
+ * @type {Number}
+ * @name TOLERANCE
+ * @default 0.01
+ */
+Renderer.TOLERANCE = 0.01;
+
+module.exports = Renderer;
diff --git a/test/renders/solutions/graphics-rect.json b/test/renders/solutions/graphics-rect.json
new file mode 100644
index 0000000..d51ebd2
--- /dev/null
+++ b/test/renders/solutions/graphics-rect.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "243969710ac251e9d5080cd55dedc517"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/solutions/sprite-new.json b/test/renders/solutions/sprite-new.json
new file mode 100644
index 0000000..c5396a6
--- /dev/null
+++ b/test/renders/solutions/sprite-new.json
@@ -0,0 +1,10 @@
+{
+ "webgl": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ },
+ "canvas": {
+ "image": "",
+ "hash": "bd5ca4fe3ec53c421044c78d86592b86"
+ }
+}
\ No newline at end of file
diff --git a/test/renders/tests/assets/bitmap-1.png b/test/renders/tests/assets/bitmap-1.png
new file mode 100644
index 0000000..1284fb8
--- /dev/null
+++ b/test/renders/tests/assets/bitmap-1.png
Binary files differ
diff --git a/test/renders/tests/graphics-rect.js b/test/renders/tests/graphics-rect.js
new file mode 100644
index 0000000..1143c96
--- /dev/null
+++ b/test/renders/tests/graphics-rect.js
@@ -0,0 +1,22 @@
+'use strict';
+
+module.exports = function ()
+{
+ const graphic = new PIXI.Graphics()
+ .beginFill(0xFFCC00, 1)
+ .drawRect(8, 8, 16, 16);
+
+ expect(graphic.width).to.equal(16);
+ expect(graphic.height).to.equal(16);
+ expect(graphic.x).to.equal(0);
+ expect(graphic.y).to.equal(0);
+
+ const bounds = graphic.getBounds();
+
+ expect(bounds.x).to.equal(8);
+ expect(bounds.y).to.equal(8);
+ expect(bounds.width).to.equal(16);
+ expect(bounds.height).to.equal(16);
+
+ this.stage.addChild(graphic);
+};
diff --git a/test/renders/tests/sprite-new.js b/test/renders/tests/sprite-new.js
new file mode 100644
index 0000000..b25ae83
--- /dev/null
+++ b/test/renders/tests/sprite-new.js
@@ -0,0 +1,30 @@
+'use strict';
+
+module.exports.async = function (done)
+{
+ const url = `file://${__dirname}/assets/bitmap-1.png`;
+ const loader = new PIXI.loaders.Loader();
+
+ loader.add('bitmap', url);
+ loader.once('complete', (loader, resources) =>
+ {
+ expect(resources.bitmap).to.be.ok;
+ expect(resources.bitmap.texture).to.be.ok;
+ expect(resources.bitmap.url).to.equal(url);
+
+ const sprite = new PIXI.Sprite(resources.bitmap.texture);
+
+ expect(sprite.width).to.equal(24);
+ expect(sprite.height).to.equal(24);
+
+ sprite.x = (32 - sprite.width) / 2;
+ sprite.y = (32 - sprite.height) / 2;
+
+ expect(sprite.x).to.equal(4);
+ expect(sprite.y).to.equal(4);
+
+ this.stage.addChild(sprite);
+ done();
+ });
+ loader.load();
+};