From a524c57014326339c8d9676b20a6a532b11ca05c Mon Sep 17 00:00:00 2001 From: Vince Date: Sat, 15 Jun 2013 18:20:25 +0200 Subject: [PATCH 01/22] Add preferences action --- css/tools.css | 6 ++++++ img/gear.png | Bin 0 -> 789 bytes index.html | 6 ++++-- 3 files changed, 10 insertions(+), 2 deletions(-) create mode 100644 img/gear.png diff --git a/css/tools.css b/css/tools.css index 69db6316..95a5a2f2 100644 --- a/css/tools.css +++ b/css/tools.css @@ -198,6 +198,12 @@ background-size: 36px 36px; } +.tool-icon.gear-icon { + background-image: url(../img/gear.png); + background-position: 6px 7px; + background-size: 32px 32px; +} + .tool-icon.upload-cloud-icon { background-image: url(../img/cloud_export.png); background-position: 4px 0px; diff --git a/img/gear.png b/img/gear.png new file mode 100644 index 0000000000000000000000000000000000000000..c09e06fa581b4656b7b9fa878e27aba95e8d8b53 GIT binary patch literal 789 zcmV+w1M2*VP)WFU8GbZ8()Nlj27Z*CxAAWdOomE>Pr0007TNkl4ZF$lId*rI!bBfDJ(`FVtE{U$T16{#`Ze;FN{64j4@^Xm4y*% zbjCucjp{=w{R2h_rIb)YZ7(NW*YDo@yZ4tY9X;eB=bqp1_w_sHe9!mZT%*z8I@%o9 z)NX*KwOhU$7-Jp*Z-J7!4uLl!(wiwT;4@GR{4c5R&a6{ebo5R3#*8sV;1f^Fnco2G41~!3aR$%1_|B7Q47h=GdeTu&`MXKuf0BdSZP1<&V zjm*Hb`|7}HM92u(0xH05{Z-UAidb_1G)3gfmtg0$(8JaT>LPL+^7nwAG3F!i6w9-E((y68KsTTOAsvO`SY1y~ChICQG3CB+W(Om}sm zUG+jGOPDU!MC2r4?J{s=N$0wVeDLM5wyK11-R$eZaiI1spoT78?ZW(d;LM^DmVhH; z%zL0S<{@KDS#{dWh{FFJGp?M4;v(=ph0Jlau8y*z
-
+
GIF
-
+
PNG
+
+
From af95a41e521c721dff7d771f6544bb0ff15770c3 Mon Sep 17 00:00:00 2001 From: Vince Date: Sat, 15 Jun 2013 18:20:36 +0200 Subject: [PATCH 02/22] Add icons resource --- resources/icons.png | Bin 0 -> 72771 bytes 1 file changed, 0 insertions(+), 0 deletions(-) create mode 100644 resources/icons.png diff --git a/resources/icons.png b/resources/icons.png new file mode 100644 index 0000000000000000000000000000000000000000..cbeb10d7edd7483008e5cf37ab4e70c31c28c591 GIT binary patch literal 72771 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 Ds-4Uf literal 0 HcmV?d00001 From 13de7cbcfb77fb5831eab4b21c8804950ab5d11d Mon Sep 17 00:00:00 2001 From: Vince Date: Sat, 15 Jun 2013 19:04:36 +0200 Subject: [PATCH 03/22] New layout: application actions sticking to the right edge --- css/style.css | 29 +++++++++++++++++++++-------- css/tools.css | 2 +- index.html | 28 +++++++++++++++------------- js/controller/ToolController.js | 4 ++-- 4 files changed, 39 insertions(+), 24 deletions(-) diff --git a/css/style.css b/css/style.css index 7d94ec42..f7d06ea4 100644 --- a/css/style.css +++ b/css/style.css @@ -12,9 +12,9 @@ body { text-align: center; font-size: 0; position: absolute; - left: 104px; + left: 100px; /* Reserve room for tools on the left edge of the screen. */ top: 10px; - right: 0; + right: 50px; /* Reserve room for actions on the right edge of the screen. */ bottom: 10px; } @@ -47,8 +47,7 @@ body { position: fixed; top: 0; bottom: 0; - left: 0; - width: 140px; + z-index: 10; } .sticky-section .wrap { @@ -56,6 +55,24 @@ body { vertical-align: middle; } +.left-sticky-section.sticky-section { + left: 0; + max-width: 100px; +} + +.left-sticky-section .tool-icon { + float: left; +} + +.right-sticky-section.sticky-section { + right: 0; + width: 50px; +} + +.right-sticky-section .tool-icon { + float: right; +} + /** * Canvases layout @@ -133,10 +150,6 @@ body { overflow: hidden; } -.application-actions { - margin-top: 35px; -} - /** * User messages */ diff --git a/css/tools.css b/css/tools.css index 95a5a2f2..d3c3d342 100644 --- a/css/tools.css +++ b/css/tools.css @@ -6,7 +6,6 @@ } .tool-icon { - float: left; cursor : pointer; width: 46px; height: 46px; @@ -219,5 +218,6 @@ font-size: 11px; text-transform: uppercase; color: #fff; + text-align: center; } diff --git a/index.html b/index.html index 505997ae..378817c6 100644 --- a/index.html +++ b/index.html @@ -17,7 +17,7 @@ -

diff --git a/js/controller/ToolController.js b/js/controller/ToolController.js index 26b008c6..cbf1d615 100644 --- a/js/controller/ToolController.js +++ b/js/controller/ToolController.js @@ -58,7 +58,7 @@ this.selectTool_(this.toolInstances[tool]); // Show tool as selected: - $('#menubar .tool-icon.selected').removeClass('selected'); + $('#tool-section .tool-icon.selected').removeClass('selected'); clickedTool.addClass('selected'); } } @@ -105,7 +105,7 @@ // Set SimplePen as default selected tool: this.selectTool_(this.toolInstances.simplePen); // Activate listener on tool panel: - $("#menubar").click($.proxy(this.onToolIconClicked_, this)); + $("#tool-section").click($.proxy(this.onToolIconClicked_, this)); // Show/hide the grid on drawing canvas: $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); From 4208e40c24ec0e770b2c9e3fc3c3a65a173c7e74 Mon Sep 17 00:00:00 2001 From: Vince Date: Sat, 15 Jun 2013 19:35:55 +0200 Subject: [PATCH 04/22] Adding drawer prototype --- css/style.css | 23 +++++++++++++++++++++-- index.html | 7 +++++-- js/piskel.js | 4 ++++ 3 files changed, 30 insertions(+), 4 deletions(-) diff --git a/css/style.css b/css/style.css index f7d06ea4..2dc6116e 100644 --- a/css/style.css +++ b/css/style.css @@ -50,7 +50,8 @@ body { z-index: 10; } -.sticky-section .wrap { +.sticky-section .wrap, +.sticky-section .drawer { display: table-cell; vertical-align: middle; } @@ -67,13 +68,31 @@ body { .right-sticky-section.sticky-section { right: 0; width: 50px; + + -webkit-transition: all 200ms ease-out; + -moz-transition: all 200ms ease-out; + -ms-transition: all 200ms ease-out; + transition: all 200ms ease-out; +} + +.right-sticky-section.expanded { + right: 300px; +} + +.drawer { + +} + +.drawer-content { + background-color: red; + height: 600px; + width: 300px; } .right-sticky-section .tool-icon { float: right; } - /** * Canvases layout */ diff --git a/index.html b/index.html index 378817c6..f19ad278 100644 --- a/index.html +++ b/index.html @@ -50,9 +50,9 @@ -
+
-
+
@@ -62,6 +62,9 @@ PNG
+
+
+
diff --git a/js/piskel.js b/js/piskel.js index 04830f4f..b130709d 100644 --- a/js/piskel.js +++ b/js/piskel.js @@ -71,6 +71,10 @@ $.namespace("pskl"); selector: '[rel=tooltip]' }); + $('#settings').click(function(evt) { + $('.right-sticky-section').toggleClass('expanded'); + }); + $('#canvas-picker').change(function(evt) { $('#canvas-picker option:selected').each(function() { console.log($(this).val()); From 765f75f255afc72183c845aece067506417a77c4 Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 09:38:31 +0200 Subject: [PATCH 05/22] Fix preview frames scrollers They were not properly updated when content was re-becoming smaller than the scroller. --- js/controller/PreviewFilmController.js | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/js/controller/PreviewFilmController.js b/js/controller/PreviewFilmController.js index 3f5a0ec3..a80f3db9 100644 --- a/js/controller/PreviewFilmController.js +++ b/js/controller/PreviewFilmController.js @@ -56,10 +56,10 @@ if (scrollBottom > treshold) { overflowBottom = true; } - var wrapper = $('#preview-list-wrapper'); - wrapper.toggleClass('top-overflow-visible', overflowTop); - wrapper.toggleClass('bottom-overflow-visible', overflowBottom); } + var wrapper = $('#preview-list-wrapper'); + wrapper.toggleClass('top-overflow-visible', overflowTop); + wrapper.toggleClass('bottom-overflow-visible', overflowBottom); }; ns.PreviewFilmController.prototype.createPreviews_ = function () { From 19b213129ae70011a6f27a5bc9862a49116a3c2b Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 10:16:26 +0200 Subject: [PATCH 06/22] Moving settings markup to drawer --- index.html | 33 ++++++++++++++++----------------- 1 file changed, 16 insertions(+), 17 deletions(-) diff --git a/index.html b/index.html index f19ad278..7fc4c808 100644 --- a/index.html +++ b/index.html @@ -31,22 +31,6 @@
- - -
- Canvas background: - -
@@ -63,7 +47,22 @@
-
+
+
+ Canvas background: + +
+ +
+ + +
+
From 955a49d2dd4ee72048b29877a565a1545430224a Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 10:17:50 +0200 Subject: [PATCH 07/22] Adding SettingsController --- index.html | 1 + js/controller/SettingsController.js | 37 +++++++++++++++++++++++++++++ js/controller/ToolController.js | 18 -------------- js/piskel.js | 9 ++----- 4 files changed, 40 insertions(+), 25 deletions(-) create mode 100644 js/controller/SettingsController.js diff --git a/index.html b/index.html index 7fc4c808..719657c6 100644 --- a/index.html +++ b/index.html @@ -142,6 +142,7 @@ + diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js new file mode 100644 index 00000000..93db72be --- /dev/null +++ b/js/controller/SettingsController.js @@ -0,0 +1,37 @@ +(function () { + var ns = $.namespace("pskl.controller"); + + ns.SettingsController = function () {}; + + /** + * Get state for the checkbox that control the display of the grid + * on the drawing canvas. + * @private + */ + ns.SettingsController.prototype.isShowGridChecked_ = function() { + var showGridCheckbox = $('#show-grid'); + var isChecked = showGridCheckbox.is(':checked'); + return isChecked; + }; + + /** + * @public + */ + ns.SettingsController.prototype.init = function() { + + // Show/hide the grid on drawing canvas: + $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); + $('#show-grid').change($.proxy(function(evt) { + var checked = this.isShowGridChecked_(); + $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [checked]); + }, this)); + + // Handle canvas background changes: + $('#canvas-picker').change(function(evt) { + $('#canvas-picker option:selected').each(function() { + console.log($(this).val()); + $('html')[0].className = $(this).val(); + }); + }); + }; +})(); \ No newline at end of file diff --git a/js/controller/ToolController.js b/js/controller/ToolController.js index cbf1d615..aa48685e 100644 --- a/js/controller/ToolController.js +++ b/js/controller/ToolController.js @@ -83,17 +83,6 @@ $('#tools-container').html(toolMarkup); }; - /** - * Get state for the checkbox that control the display of the grid - * on the drawing canvas. - * @private - */ - ns.ToolController.prototype.isShowGridChecked_ = function() { - var showGridCheckbox = $('#show-grid'); - var isChecked = showGridCheckbox.is(':checked'); - return isChecked; - }; - /** * @public */ @@ -106,12 +95,5 @@ this.selectTool_(this.toolInstances.simplePen); // Activate listener on tool panel: $("#tool-section").click($.proxy(this.onToolIconClicked_, this)); - - // Show/hide the grid on drawing canvas: - $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); - $('#show-grid').change($.proxy(function(evt) { - var checked = this.isShowGridChecked_(); - $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [checked]); - }, this)); }; })(); \ No newline at end of file diff --git a/js/piskel.js b/js/piskel.js index b130709d..6f68f1f3 100644 --- a/js/piskel.js +++ b/js/piskel.js @@ -24,6 +24,7 @@ $.namespace("pskl"); this.drawingController = new pskl.controller.DrawingController(frameSheet, $('#drawing-canvas-container')); this.animationController = new pskl.controller.AnimatedPreviewController(frameSheet, $('#preview-canvas-container')); this.previewsController = new pskl.controller.PreviewFilmController(frameSheet, $('#preview-list')); + this.settingsController = new pskl.controller.SettingsController(); // To catch the current active frame, the selection manager have to be initialized before // the 'frameSheet.setCurrentFrameIndex(0);' line below. @@ -39,6 +40,7 @@ $.namespace("pskl"); this.animationController.init(); this.previewsController.init(); + this.settingsController.init(); this.historyService = new pskl.service.HistoryService(frameSheet); this.historyService.init(); @@ -74,13 +76,6 @@ $.namespace("pskl"); $('#settings').click(function(evt) { $('.right-sticky-section').toggleClass('expanded'); }); - - $('#canvas-picker').change(function(evt) { - $('#canvas-picker option:selected').each(function() { - console.log($(this).val()); - $('html')[0].className = $(this).val(); - }); - }); }, render : function (delta) { From 3ac2c5e92240de3c3474c5986fd6f03c11938094 Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 10:18:22 +0200 Subject: [PATCH 08/22] Remove bad/deprecated TODOs --- js/piskel.js | 2 -- 1 file changed, 2 deletions(-) diff --git a/js/piskel.js b/js/piskel.js index 6f68f1f3..74114b4a 100644 --- a/js/piskel.js +++ b/js/piskel.js @@ -129,7 +129,6 @@ $.namespace("pskl"); loadFramesheetFromService : function (frameId) { var xhr = new XMLHttpRequest(); - // TODO: Change frameId to framesheetId on the backend xhr.open('GET', Constants.PISKEL_SERVICE_URL + '/get?l=' + frameId, true); xhr.responseType = 'text'; @@ -151,7 +150,6 @@ $.namespace("pskl"); // TODO(julz): Create package ? storeSheet : function (event) { - // TODO Refactor using jquery ? var xhr = new XMLHttpRequest(); var formData = new FormData(); formData.append('framesheet_content', frameSheet.serialize()); From 3d67be94a13f1f9e12558f3ef8c3947cee52a438 Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 10:36:05 +0200 Subject: [PATCH 09/22] Initial quick design for right settings drawer --- css/style.css | 55 +++++++++++++++++++++++++++++++++++++++++++-------- css/tools.css | 2 +- index.html | 14 +++++++------ 3 files changed, 56 insertions(+), 15 deletions(-) diff --git a/css/style.css b/css/style.css index 2dc6116e..0c605b63 100644 --- a/css/style.css +++ b/css/style.css @@ -47,7 +47,7 @@ body { position: fixed; top: 0; bottom: 0; - z-index: 10; + z-index: 1000; } .sticky-section .wrap, @@ -75,8 +75,8 @@ body { transition: all 200ms ease-out; } -.right-sticky-section.expanded { - right: 300px; +.right-sticky-section .tool-icon { + float: right; } .drawer { @@ -84,13 +84,52 @@ body { } .drawer-content { - background-color: red; - height: 600px; - width: 300px; + overflow: hidden; + background-color: #444; + height: 550px; + max-height: 100%; + width: 280px; + border-top-left-radius: 4px; + border-bottom-left-radius: 4px; } -.right-sticky-section .tool-icon { - float: right; +/** Righty sticky drawer expanded state. */ + +.right-sticky-section.expanded { + right: 280px; +} + +.right-sticky-section .tool-icon.has-expanded-drawer { + position: relative; + background-color: #444; +} + +.right-sticky-section .tool-icon.has-expanded-drawer:before { + content: " "; + position: absolute; + background-color: #444; + top: 0; + bottom: 0; + right: -5px; + width: 10px; + z-index: -1; +} + +.settings-title { + margin: 10px 20px; + text-transform: uppercase; + font-size: 12px; + font-weight: bold; + color: #eee; + border-bottom: 1px #aaa solid; + padding-bottom: 5px; +} + +.settings-item { + margin: 10px 20px; + color: #eee; + font-size: 13px; + cursor: pointer; } /** diff --git a/css/tools.css b/css/tools.css index d3c3d342..76165c7b 100644 --- a/css/tools.css +++ b/css/tools.css @@ -10,7 +10,7 @@ width: 46px; height: 46px; margin: 1px; - background-color: rgba(200,200,200, .1); + background-color: #3a3a3a; background-repeat: no-repeat; background-position: 12px 12px; background-size: 24px 24px; diff --git a/index.html b/index.html index 719657c6..97c50e7d 100644 --- a/index.html +++ b/index.html @@ -48,9 +48,12 @@
-
- Canvas background: - @@ -58,9 +61,8 @@
-
- - +
+
From 458f56f1ecdfe4f4ae12e7e50af22f300d585504 Mon Sep 17 00:00:00 2001 From: Vince Date: Sun, 16 Jun 2013 10:36:26 +0200 Subject: [PATCH 10/22] Move toggling code to settingscontroller --- js/controller/SettingsController.js | 8 ++++++++ js/piskel.js | 4 ---- 2 files changed, 8 insertions(+), 4 deletions(-) diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 93db72be..01281967 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -14,11 +14,19 @@ return isChecked; }; + // TODO(vincz): add get/set store + /** * @public */ ns.SettingsController.prototype.init = function() { + // Expand drawer when clicking 'Settings' tab. + $('#settings').click(function(evt) { + $('.right-sticky-section').toggleClass('expanded'); + $('#settings').toggleClass('has-expanded-drawer') + }); + // Show/hide the grid on drawing canvas: $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); $('#show-grid').change($.proxy(function(evt) { diff --git a/js/piskel.js b/js/piskel.js index 74114b4a..8d067020 100644 --- a/js/piskel.js +++ b/js/piskel.js @@ -72,10 +72,6 @@ $.namespace("pskl"); $('body').tooltip({ selector: '[rel=tooltip]' }); - - $('#settings').click(function(evt) { - $('.right-sticky-section').toggleClass('expanded'); - }); }, render : function (delta) { From 453f77cdd61eca171ad71c599c96257d3c857b1f Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 10:37:08 +0200 Subject: [PATCH 11/22] Nit: fix indentation --- js/controller/SettingsController.js | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 01281967..7752b145 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -21,11 +21,11 @@ */ ns.SettingsController.prototype.init = function() { - // Expand drawer when clicking 'Settings' tab. - $('#settings').click(function(evt) { - $('.right-sticky-section').toggleClass('expanded'); - $('#settings').toggleClass('has-expanded-drawer') - }); + // Expand drawer when clicking 'Settings' tab. + $('#settings').click(function(evt) { + $('.right-sticky-section').toggleClass('expanded'); + $('#settings').toggleClass('has-expanded-drawer'); + }); // Show/hide the grid on drawing canvas: $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); From cc3bd72b8636e5ae151d83b527c0cd1ece51e2e9 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 10:37:23 +0200 Subject: [PATCH 12/22] Small settings panel styles update --- css/style.css | 31 +++++++++++++++---------------- index.html | 31 ++++++++++++++++--------------- 2 files changed, 31 insertions(+), 31 deletions(-) diff --git a/css/style.css b/css/style.css index 0c605b63..63ff2a34 100644 --- a/css/style.css +++ b/css/style.css @@ -77,6 +77,7 @@ body { .right-sticky-section .tool-icon { float: right; + margin-right: 0; } .drawer { @@ -99,36 +100,34 @@ body { right: 280px; } +.right-sticky-section.expanded .tool-icon { + margin-right: 1px; +} + .right-sticky-section .tool-icon.has-expanded-drawer { position: relative; background-color: #444; + margin-right: 0; + padding-right: 1px; } -.right-sticky-section .tool-icon.has-expanded-drawer:before { - content: " "; - position: absolute; - background-color: #444; - top: 0; - bottom: 0; - right: -5px; - width: 10px; - z-index: -1; +.settings-section { + margin: 10px 20px; + font-size: 12px; + font-weight: bold; + color: #ccc; + text-shadow: 1px 1px #000; } .settings-title { - margin: 10px 20px; + margin-top: 20px; + margin-bottom: 10px; text-transform: uppercase; - font-size: 12px; - font-weight: bold; - color: #eee; border-bottom: 1px #aaa solid; padding-bottom: 5px; } .settings-item { - margin: 10px 20px; - color: #eee; - font-size: 13px; cursor: pointer; } diff --git a/index.html b/index.html index 97c50e7d..f6425221 100644 --- a/index.html +++ b/index.html @@ -48,21 +48,22 @@
-
- Canvas settings: -
-
- - -
- -
- +
+
+ Canvas settings: +
+
+ + +
+
+ +
From 97ddc6ecc156b9b7ae9c40c22db9f81bead26836 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 11:36:56 +0200 Subject: [PATCH 13/22] Adding simple backgorund picker (without persistence) --- css/style.css | 38 +++++++++++++++++++++++++++-- index.html | 23 +++++++++++------ js/controller/SettingsController.js | 17 +++++++++---- 3 files changed, 63 insertions(+), 15 deletions(-) diff --git a/css/style.css b/css/style.css index 63ff2a34..cc008279 100644 --- a/css/style.css +++ b/css/style.css @@ -127,8 +127,39 @@ body { padding-bottom: 5px; } -.settings-item { +.settings-item {} + +.background-picker-wrapper { + overflow: hidden; + padding: 10px 5px 20px 5px; +} + +.background-picker { cursor: pointer; + float: left; + height: 35px; + width: 35px; + background-color: transparent; + margin-right: 15px; + padding: 1px; + position: relative; +} + +.background-picker:after { + content: " "; + position: absolute; + top: -2px; + right: -2px; + bottom: -2px; + left: -2px; +} + +.background-picker:hover:after { + border: #eee 1px solid; +} + +.background-picker.selected:after { + border: gold 1px solid; } /** @@ -146,7 +177,6 @@ body { } .canvas-container .canvas-background { - background: url(../img/canvas_background/medium_canvas_background.png) repeat; position: absolute; top: 0; right: 0; @@ -154,18 +184,22 @@ body { left: 0; } +.light-picker-background, .light-canvas-background .canvas-background { background: url(../img/canvas_background/light_canvas_background.png) repeat; } +.medium-picker-background, .medium-canvas-background .canvas-background { background: url(../img/canvas_background/medium_canvas_background.png) repeat; } +.lowcont-medium-picker-background, .lowcont-medium-canvas-background .canvas-background { background: url(../img/canvas_background/lowcont_medium_canvas_background.png) repeat; } +.lowcont-dark-picker-background, .lowcont-dark-canvas-background .canvas-background { background: url(../img/canvas_background/lowcont_dark_canvas_background.png) repeat; } diff --git a/index.html b/index.html index f6425221..62fba7a2 100644 --- a/index.html +++ b/index.html @@ -15,8 +15,7 @@ - - +
    @@ -54,12 +53,20 @@
    - +
    +
    +
    +
    +
    +
    +
    +
    +
    +
    diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 7752b145..06c38e06 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -35,11 +35,18 @@ }, this)); // Handle canvas background changes: - $('#canvas-picker').change(function(evt) { - $('#canvas-picker option:selected').each(function() { - console.log($(this).val()); - $('html')[0].className = $(this).val(); - }); + $('#background-picker-wrapper').click(function(evt) { + var target = $(evt.target).closest('.background-picker'); + if (target.length) { + var backgroundClass = target.data('background'); + var body = $('body'); + body.removeClass(body.data('current-background')); + body.addClass(backgroundClass); + body.data('current-background', backgroundClass); + + $('.background-picker').removeClass('selected'); + target.addClass('selected'); + } }); }; })(); \ No newline at end of file From 403105aae4a64616e2cc362185c734dd821c0929 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 11:42:53 +0200 Subject: [PATCH 14/22] Adding TODOs related to namespace cleaning --- js/Constants.js | 1 + js/Events.js | 1 + js/piskel.js | 1 + 3 files changed, 3 insertions(+) diff --git a/js/Constants.js b/js/Constants.js index 231dfb5b..c7abf94a 100644 --- a/js/Constants.js +++ b/js/Constants.js @@ -1,3 +1,4 @@ +// TODO(grosbouddha): put under pskl namespace. var Constants = { DEFAULT_SIZE : { height : 32, diff --git a/js/Events.js b/js/Events.js index ab065356..cd018757 100644 --- a/js/Events.js +++ b/js/Events.js @@ -1,3 +1,4 @@ +// TODO(grosbouddha): put under pskl namespace. Events = { TOOL_SELECTED : "TOOL_SELECTED", diff --git a/js/piskel.js b/js/piskel.js index 8d067020..725d15c9 100644 --- a/js/piskel.js +++ b/js/piskel.js @@ -206,6 +206,7 @@ $.namespace("pskl"); } }; + // TODO(grosbouddha): Remove this window.piskel global (eventually pskl.piskel or pskl.app instead) window.piskel = piskel; piskel.init(); From 889d5c0d53c2c0b8a9e08bacefa2c8800e6fcd1d Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 19:54:43 +0200 Subject: [PATCH 15/22] Add basic UserSettings persistence static utility. Based on localStorage for now. --- index.html | 1 + js/controller/SettingsController.js | 4 ++ js/utils/UserSettings.js | 68 +++++++++++++++++++++++++++++ 3 files changed, 73 insertions(+) create mode 100644 js/utils/UserSettings.js diff --git a/index.html b/index.html index 62fba7a2..f68c91c4 100644 --- a/index.html +++ b/index.html @@ -130,6 +130,7 @@ + diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 06c38e06..4eb1f4be 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -21,6 +21,9 @@ */ ns.SettingsController.prototype.init = function() { + var show_grid = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID); + $('#show-grid').prop('checked', show_grid); + // Expand drawer when clicking 'Settings' tab. $('#settings').click(function(evt) { $('.right-sticky-section').toggleClass('expanded'); @@ -32,6 +35,7 @@ $('#show-grid').change($.proxy(function(evt) { var checked = this.isShowGridChecked_(); $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [checked]); + pskl.UserSettings.set(pskl.UserSettings.SHOW_GRID, checked); }, this)); // Handle canvas background changes: diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js new file mode 100644 index 00000000..5999d4e9 --- /dev/null +++ b/js/utils/UserSettings.js @@ -0,0 +1,68 @@ +(function () { + var ns = $.namespace("pskl"); + + ns.UserSettings = { + + SHOW_GRID: 'SHOW_GRID', + + KEY_TO_DEFAULT_VALUE_MAP_ : { + 'SHOW_GRID' : false + }, + + /** + * @private + */ + cache_: { + + }, + + /** + * Static method to access a user defined settings value ot its default + * value if not defined yet. + */ + get : function (key) { + this.checKeyValidity_(key); + if (key in this.cache_) { + return cache[key]; + } + return this.get_(key); + }, + + set : function (key, value) { + this.checKeyValidity_(key); + this.cache_[key] = value; + this.set_(key, value); + }, + + /** + * @private + */ + get_ : function(key) { + var value = window.localStorage[key]; + if (value === undefined) { + value = this.KEY_TO_DEFAULT_VALUE_MAP_[key]; + } + else { + var entry = JSON.parse(value); + value = entry.jsonValue; + } + return value; + }, + + /** + * @private + */ + set_ : function(key, value) { + var entry = { 'jsonValue': value }; + window.localStorage[key] = JSON.stringify(entry); + }, + + checKeyValidity_ : function(key) { + if(key in this.KEY_TO_DEFAULT_VALUE_MAP_) { + return true; + } + console.log("UserSettings key <"+ key +"> not find in supported keys.") + return false; + } + }; +})(); \ No newline at end of file From 1d4ff1d2dec8b679f34f4ce2478d6bd355bf4f66 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 20:24:27 +0200 Subject: [PATCH 16/22] Create an event based UserSettings and use it for the grid diplay state. --- js/Events.js | 8 +++++++- js/controller/DrawingController.js | 11 ++++++++++- js/controller/SettingsController.js | 19 ++----------------- js/rendering/FrameRenderer.js | 17 ++++++++++++++--- js/utils/UserSettings.js | 8 ++++---- 5 files changed, 37 insertions(+), 26 deletions(-) diff --git a/js/Events.js b/js/Events.js index cd018757..34283d0d 100644 --- a/js/Events.js +++ b/js/Events.js @@ -31,7 +31,13 @@ Events = { */ REDRAW_PREVIEWFILM: "REDRAW_PREVIEWFILM", - GRID_DISPLAY_STATE_CHANGED: "GRID_DISPLAY_STATE_CHANGED", + /** + * Fired each time a user setting change. + * The payload will be: + * 1st argument: Name of the settings + * 2nd argument: New value + */ + USER_SETTINGS_CHANGED: "USER_SETTINGS_CHANGED", /** * The framesheet was reseted and is now probably drastically different. diff --git a/js/controller/DrawingController.js b/js/controller/DrawingController.js index c1284ff1..63f9b9b7 100644 --- a/js/controller/DrawingController.js +++ b/js/controller/DrawingController.js @@ -61,7 +61,7 @@ $(window).resize($.proxy(this.startDPIUpdateTimer_, this)); $.subscribe(Events.FRAME_SIZE_CHANGED, $.proxy(this.updateDPI_, this)); - $.subscribe(Events.GRID_DISPLAY_STATE_CHANGED, $.proxy(this.forceRendering_, this)); + $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); }; ns.DrawingController.prototype.initMouseBehavior = function() { @@ -83,6 +83,15 @@ this.dpiUpdateTimer = window.setTimeout($.proxy(this.updateDPI_, this), 200); }, + /** + * @private + */ + ns.DrawingController.prototype.onUserSettingsChange_ = function (evt, settingsName, settingsValue) { + if(settingsName == pskl.UserSettings.SHOW_GRID) { + this.forceRendering_(); + } + }, + /** * @private */ diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 4eb1f4be..ae7e1db7 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -3,19 +3,6 @@ ns.SettingsController = function () {}; - /** - * Get state for the checkbox that control the display of the grid - * on the drawing canvas. - * @private - */ - ns.SettingsController.prototype.isShowGridChecked_ = function() { - var showGridCheckbox = $('#show-grid'); - var isChecked = showGridCheckbox.is(':checked'); - return isChecked; - }; - - // TODO(vincz): add get/set store - /** * @public */ @@ -30,11 +17,9 @@ $('#settings').toggleClass('has-expanded-drawer'); }); - // Show/hide the grid on drawing canvas: - $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [this.isShowGridChecked_()]); + // Handle grid display changes: $('#show-grid').change($.proxy(function(evt) { - var checked = this.isShowGridChecked_(); - $.publish(Events.GRID_DISPLAY_STATE_CHANGED, [checked]); + var checked = $('#show-grid').prop('checked'); pskl.UserSettings.set(pskl.UserSettings.SHOW_GRID, checked); }, this)); diff --git a/js/rendering/FrameRenderer.js b/js/rendering/FrameRenderer.js index e4fec384..8be75f6b 100644 --- a/js/rendering/FrameRenderer.js +++ b/js/rendering/FrameRenderer.js @@ -21,13 +21,14 @@ this.className = className; this.canvas = null; this.hasGrid = renderingOptions.hasGrid; - this.gridStrokeWidth = 0; + + this.gridStrokeWidth = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID) ? Constants.GRID_STROKE_WIDTH : 0; // Flag to know if the config was altered this.canvasConfigDirty = true; if(this.hasGrid) { - $.subscribe(Events.GRID_DISPLAY_STATE_CHANGED, $.proxy(this.showGrid, this)); + $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); } }; @@ -41,7 +42,17 @@ this.canvasConfigDirty = true; }; - ns.FrameRenderer.prototype.showGrid = function (evt, show) { + /** + * @private + */ + ns.FrameRenderer.prototype.onUserSettingsChange_ = function (evt, settingsName, settingsValue) { + + if(settingsName == pskl.UserSettings.SHOW_GRID) { + this.enableGrid(settingsValue); + } + }; + + ns.FrameRenderer.prototype.enableGrid = function (show) { this.gridStrokeWidth = 0; if(show) { diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js index 5999d4e9..fa6783e6 100644 --- a/js/utils/UserSettings.js +++ b/js/utils/UserSettings.js @@ -12,9 +12,7 @@ /** * @private */ - cache_: { - - }, + cache_: {}, /** * Static method to access a user defined settings value ot its default @@ -31,7 +29,9 @@ set : function (key, value) { this.checKeyValidity_(key); this.cache_[key] = value; - this.set_(key, value); + this.set_(key, value); + + $.publish(Events.USER_SETTINGS_CHANGED, [key, value]); }, /** From 9a3a87bd577adfde35cf20c4d344493ed846a843 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 21:03:22 +0200 Subject: [PATCH 17/22] Fix travis --- js/controller/SettingsController.js | 10 +++++----- js/utils/UserSettings.js | 9 ++++++--- 2 files changed, 11 insertions(+), 8 deletions(-) diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index ae7e1db7..8cefcdb1 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -8,7 +8,7 @@ */ ns.SettingsController.prototype.init = function() { - var show_grid = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID); + var show_grid = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID); $('#show-grid').prop('checked', show_grid); // Expand drawer when clicking 'Settings' tab. @@ -27,10 +27,10 @@ $('#background-picker-wrapper').click(function(evt) { var target = $(evt.target).closest('.background-picker'); if (target.length) { - var backgroundClass = target.data('background'); - var body = $('body'); - body.removeClass(body.data('current-background')); - body.addClass(backgroundClass); + var backgroundClass = target.data('background'); + var body = $('body'); + body.removeClass(body.data('current-background')); + body.addClass(backgroundClass); body.data('current-background', backgroundClass); $('.background-picker').removeClass('selected'); diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js index fa6783e6..8198d575 100644 --- a/js/utils/UserSettings.js +++ b/js/utils/UserSettings.js @@ -3,7 +3,7 @@ ns.UserSettings = { - SHOW_GRID: 'SHOW_GRID', + SHOW_GRID : 'SHOW_GRID', KEY_TO_DEFAULT_VALUE_MAP_ : { 'SHOW_GRID' : false @@ -12,7 +12,7 @@ /** * @private */ - cache_: {}, + cache_ : {}, /** * Static method to access a user defined settings value ot its default @@ -57,11 +57,14 @@ window.localStorage[key] = JSON.stringify(entry); }, + /** + * @private + */ checKeyValidity_ : function(key) { if(key in this.KEY_TO_DEFAULT_VALUE_MAP_) { return true; } - console.log("UserSettings key <"+ key +"> not find in supported keys.") + console.log("UserSettings key <"+ key +"> not find in supported keys."); return false; } }; From 2691b23c097fec50d04e65dfa8454e862f9aa9a1 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 21:23:21 +0200 Subject: [PATCH 18/22] Fix cache typo in UserSettings --- js/utils/UserSettings.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js index 8198d575..77274541 100644 --- a/js/utils/UserSettings.js +++ b/js/utils/UserSettings.js @@ -21,7 +21,7 @@ get : function (key) { this.checKeyValidity_(key); if (key in this.cache_) { - return cache[key]; + return this.cache_[key]; } return this.get_(key); }, From 9eac3414a32cef5cc43cfbcb1f645160dd5a6734 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 23:06:17 +0200 Subject: [PATCH 19/22] Clean grid code: semantic names and tiny refactor using UserSettings --- js/controller/DrawingController.js | 2 +- js/rendering/FrameRenderer.js | 13 +++++++------ 2 files changed, 8 insertions(+), 7 deletions(-) diff --git a/js/controller/DrawingController.js b/js/controller/DrawingController.js index 63f9b9b7..c6ee9844 100644 --- a/js/controller/DrawingController.js +++ b/js/controller/DrawingController.js @@ -19,7 +19,7 @@ // TODO(vincz): Store user prefs in a localstorage string ? var renderingOptions = { "dpi": this.calculateDPI_(), - "hasGrid" : true + "supportGridRendering" : true }; this.renderer = new pskl.rendering.FrameRenderer(this.container, renderingOptions, "drawing-canvas"); diff --git a/js/rendering/FrameRenderer.js b/js/rendering/FrameRenderer.js index 8be75f6b..17f2d1d2 100644 --- a/js/rendering/FrameRenderer.js +++ b/js/rendering/FrameRenderer.js @@ -2,27 +2,26 @@ var ns = $.namespace("pskl.rendering"); ns.FrameRenderer = function (container, renderingOptions, className) { - this.defaultRenderingOptions = { - "hasGrid" : false + 'supportGridRendering' : false }; renderingOptions = $.extend(true, {}, this.defaultRenderingOptions, renderingOptions); if(container === undefined) { - throw "Bad FrameRenderer initialization. undefined."; + throw 'Bad FrameRenderer initialization. undefined.'; } if(isNaN(renderingOptions.dpi)) { - throw "Bad FrameRenderer initialization. not well defined."; + throw 'Bad FrameRenderer initialization. not well defined.'; } this.container = container; this.dpi = renderingOptions.dpi; this.className = className; this.canvas = null; - this.hasGrid = renderingOptions.hasGrid; + this.supportGridRendering = renderingOptions.supportGridRendering; - this.gridStrokeWidth = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID) ? Constants.GRID_STROKE_WIDTH : 0; + this.enableGrid(pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID)); // Flag to know if the config was altered this.canvasConfigDirty = true; @@ -59,6 +58,8 @@ this.gridStrokeWidth = Constants.GRID_STROKE_WIDTH; } + ns.FrameRenderer.prototype.enableGrid = function (flag) { + this.gridStrokeWidth = (flag && this.supportGridRendering) ? Constants.GRID_STROKE_WIDTH : 0; this.canvasConfigDirty = true; }; From 5913b1964185cdfffe73f2260c93d750bc8eadb8 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 23:10:35 +0200 Subject: [PATCH 20/22] Plug canvas background settings on UserSettings storage Canvas background class are now manager at the FrameRenderer level instead of CSS classes at the top of the DOM. --- index.html | 10 ++++----- js/controller/SettingsController.js | 14 +++++++----- js/rendering/FrameRenderer.js | 34 +++++++++++++++++------------ js/utils/UserSettings.js | 4 +++- 4 files changed, 37 insertions(+), 25 deletions(-) diff --git a/index.html b/index.html index f68c91c4..77de32b0 100644 --- a/index.html +++ b/index.html @@ -15,7 +15,7 @@ - +
      @@ -54,16 +54,16 @@
      -
      -
      -
      -
      diff --git a/js/controller/SettingsController.js b/js/controller/SettingsController.js index 8cefcdb1..7c54903d 100644 --- a/js/controller/SettingsController.js +++ b/js/controller/SettingsController.js @@ -8,6 +8,13 @@ */ ns.SettingsController.prototype.init = function() { + // Highlight selected background picker: + var backgroundClass = pskl.UserSettings.get(pskl.UserSettings.CANVAS_BACKGROUND); + $('#background-picker-wrapper') + .find('.background-picker[data-background-class=' + backgroundClass + ']') + .addClass('selected'); + + // Initial state for grid display: var show_grid = pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID); $('#show-grid').prop('checked', show_grid); @@ -27,11 +34,8 @@ $('#background-picker-wrapper').click(function(evt) { var target = $(evt.target).closest('.background-picker'); if (target.length) { - var backgroundClass = target.data('background'); - var body = $('body'); - body.removeClass(body.data('current-background')); - body.addClass(backgroundClass); - body.data('current-background', backgroundClass); + var backgroundClass = target.data('background-class'); + pskl.UserSettings.set(pskl.UserSettings.CANVAS_BACKGROUND, backgroundClass); $('.background-picker').removeClass('selected'); target.addClass('selected'); diff --git a/js/rendering/FrameRenderer.js b/js/rendering/FrameRenderer.js index 17f2d1d2..5ec43c23 100644 --- a/js/rendering/FrameRenderer.js +++ b/js/rendering/FrameRenderer.js @@ -25,10 +25,8 @@ // Flag to know if the config was altered this.canvasConfigDirty = true; - - if(this.hasGrid) { - $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); - } + this.updateBackgroundClass_(pskl.UserSettings.get(pskl.UserSettings.CANVAS_BACKGROUND)); + $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); }; ns.FrameRenderer.prototype.init = function (frame) { @@ -44,20 +42,28 @@ /** * @private */ - ns.FrameRenderer.prototype.onUserSettingsChange_ = function (evt, settingsName, settingsValue) { + ns.FrameRenderer.prototype.onUserSettingsChange_ = function (evt, settingName, settingValue) { - if(settingsName == pskl.UserSettings.SHOW_GRID) { - this.enableGrid(settingsValue); + if(settingName == pskl.UserSettings.SHOW_GRID) { + this.enableGrid(settingValue); + } + else if (settingName == pskl.UserSettings.CANVAS_BACKGROUND) { + this.updateBackgroundClass_(settingValue); } }; - ns.FrameRenderer.prototype.enableGrid = function (show) { - - this.gridStrokeWidth = 0; - if(show) { - this.gridStrokeWidth = Constants.GRID_STROKE_WIDTH; - } - + /** + * @private + */ + ns.FrameRenderer.prototype.updateBackgroundClass_ = function (newClass) { + var currentClass = this.container.data('current-background-class'); + if (currentClass) { + this.container.removeClass(currentClass); + } + this.container.addClass(newClass); + this.container.data('current-background-class', newClass); + }; + ns.FrameRenderer.prototype.enableGrid = function (flag) { this.gridStrokeWidth = (flag && this.supportGridRendering) ? Constants.GRID_STROKE_WIDTH : 0; this.canvasConfigDirty = true; diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js index 77274541..73946d86 100644 --- a/js/utils/UserSettings.js +++ b/js/utils/UserSettings.js @@ -4,9 +4,11 @@ ns.UserSettings = { SHOW_GRID : 'SHOW_GRID', + CANVAS_BACKGROUND : 'CANVAS_BACKGROUND', KEY_TO_DEFAULT_VALUE_MAP_ : { - 'SHOW_GRID' : false + 'SHOW_GRID' : false, + 'CANVAS_BACKGROUND' : 'medium-canvas-background' }, /** From 73aae69425c86a97d69f7827b6e88e3ffff892e4 Mon Sep 17 00:00:00 2001 From: Vince Date: Mon, 17 Jun 2013 23:22:30 +0200 Subject: [PATCH 21/22] Fix Travis: convert tabs to spaces --- js/rendering/FrameRenderer.js | 278 +++++++++++++++++----------------- 1 file changed, 139 insertions(+), 139 deletions(-) diff --git a/js/rendering/FrameRenderer.js b/js/rendering/FrameRenderer.js index 5ec43c23..75223860 100644 --- a/js/rendering/FrameRenderer.js +++ b/js/rendering/FrameRenderer.js @@ -1,161 +1,161 @@ (function () { - var ns = $.namespace("pskl.rendering"); + var ns = $.namespace("pskl.rendering"); - ns.FrameRenderer = function (container, renderingOptions, className) { - this.defaultRenderingOptions = { - 'supportGridRendering' : false - }; - renderingOptions = $.extend(true, {}, this.defaultRenderingOptions, renderingOptions); + ns.FrameRenderer = function (container, renderingOptions, className) { + this.defaultRenderingOptions = { + 'supportGridRendering' : false + }; + renderingOptions = $.extend(true, {}, this.defaultRenderingOptions, renderingOptions); - if(container === undefined) { - throw 'Bad FrameRenderer initialization. undefined.'; - } - - if(isNaN(renderingOptions.dpi)) { - throw 'Bad FrameRenderer initialization. not well defined.'; - } + if(container === undefined) { + throw 'Bad FrameRenderer initialization. undefined.'; + } + + if(isNaN(renderingOptions.dpi)) { + throw 'Bad FrameRenderer initialization. not well defined.'; + } - this.container = container; - this.dpi = renderingOptions.dpi; - this.className = className; - this.canvas = null; - this.supportGridRendering = renderingOptions.supportGridRendering; + this.container = container; + this.dpi = renderingOptions.dpi; + this.className = className; + this.canvas = null; + this.supportGridRendering = renderingOptions.supportGridRendering; - this.enableGrid(pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID)); + this.enableGrid(pskl.UserSettings.get(pskl.UserSettings.SHOW_GRID)); - // Flag to know if the config was altered - this.canvasConfigDirty = true; - this.updateBackgroundClass_(pskl.UserSettings.get(pskl.UserSettings.CANVAS_BACKGROUND)); - $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); - }; + // Flag to know if the config was altered + this.canvasConfigDirty = true; + this.updateBackgroundClass_(pskl.UserSettings.get(pskl.UserSettings.CANVAS_BACKGROUND)); + $.subscribe(Events.USER_SETTINGS_CHANGED, $.proxy(this.onUserSettingsChange_, this)); + }; - ns.FrameRenderer.prototype.init = function (frame) { - this.render(frame); - this.lastRenderedFrame = frame; - }; + ns.FrameRenderer.prototype.init = function (frame) { + this.render(frame); + this.lastRenderedFrame = frame; + }; - ns.FrameRenderer.prototype.updateDPI = function (newDPI) { - this.dpi = newDPI; - this.canvasConfigDirty = true; - }; + ns.FrameRenderer.prototype.updateDPI = function (newDPI) { + this.dpi = newDPI; + this.canvasConfigDirty = true; + }; - /** - * @private - */ - ns.FrameRenderer.prototype.onUserSettingsChange_ = function (evt, settingName, settingValue) { - - if(settingName == pskl.UserSettings.SHOW_GRID) { - this.enableGrid(settingValue); - } - else if (settingName == pskl.UserSettings.CANVAS_BACKGROUND) { - this.updateBackgroundClass_(settingValue); - } - }; + /** + * @private + */ + ns.FrameRenderer.prototype.onUserSettingsChange_ = function (evt, settingName, settingValue) { + + if(settingName == pskl.UserSettings.SHOW_GRID) { + this.enableGrid(settingValue); + } + else if (settingName == pskl.UserSettings.CANVAS_BACKGROUND) { + this.updateBackgroundClass_(settingValue); + } + }; - /** - * @private - */ - ns.FrameRenderer.prototype.updateBackgroundClass_ = function (newClass) { - var currentClass = this.container.data('current-background-class'); - if (currentClass) { - this.container.removeClass(currentClass); - } - this.container.addClass(newClass); + /** + * @private + */ + ns.FrameRenderer.prototype.updateBackgroundClass_ = function (newClass) { + var currentClass = this.container.data('current-background-class'); + if (currentClass) { + this.container.removeClass(currentClass); + } + this.container.addClass(newClass); this.container.data('current-background-class', newClass); - }; + }; - ns.FrameRenderer.prototype.enableGrid = function (flag) { - this.gridStrokeWidth = (flag && this.supportGridRendering) ? Constants.GRID_STROKE_WIDTH : 0; - this.canvasConfigDirty = true; - }; + ns.FrameRenderer.prototype.enableGrid = function (flag) { + this.gridStrokeWidth = (flag && this.supportGridRendering) ? Constants.GRID_STROKE_WIDTH : 0; + this.canvasConfigDirty = true; + }; - ns.FrameRenderer.prototype.render = function (frame) { - this.clear(frame); - var context = this.getCanvas_(frame).getContext('2d'); - for(var col = 0, width = frame.getWidth(); col < width; col++) { - for(var row = 0, height = frame.getHeight(); row < height; row++) { - var color = frame.getPixel(col, row); - this.renderPixel_(color, col, row, context); - } - } - this.lastRenderedFrame = frame; - }; + ns.FrameRenderer.prototype.render = function (frame) { + this.clear(frame); + var context = this.getCanvas_(frame).getContext('2d'); + for(var col = 0, width = frame.getWidth(); col < width; col++) { + for(var row = 0, height = frame.getHeight(); row < height; row++) { + var color = frame.getPixel(col, row); + this.renderPixel_(color, col, row, context); + } + } + this.lastRenderedFrame = frame; + }; - ns.FrameRenderer.prototype.renderPixel_ = function (color, col, row, context) { - if(color != Constants.TRANSPARENT_COLOR) { - context.fillStyle = color; - context.fillRect(this.getFramePos_(col), this.getFramePos_(row), this.dpi, this.dpi); - } - }; + ns.FrameRenderer.prototype.renderPixel_ = function (color, col, row, context) { + if(color != Constants.TRANSPARENT_COLOR) { + context.fillStyle = color; + context.fillRect(this.getFramePos_(col), this.getFramePos_(row), this.dpi, this.dpi); + } + }; - ns.FrameRenderer.prototype.clear = function (frame) { - var canvas = this.getCanvas_(frame); - canvas.getContext("2d").clearRect(0, 0, canvas.width, canvas.height); - }; + ns.FrameRenderer.prototype.clear = function (frame) { + var canvas = this.getCanvas_(frame); + canvas.getContext("2d").clearRect(0, 0, canvas.width, canvas.height); + }; - /** - * Transform a screen pixel-based coordinate (relative to the top-left corner of the rendered - * frame) into a sprite coordinate in column and row. - * @public - */ - ns.FrameRenderer.prototype.convertPixelCoordinatesIntoSpriteCoordinate = function(coords) { - var cellSize = this.dpi + this.gridStrokeWidth; - return { - "col" : (coords.x - coords.x % cellSize) / cellSize, - "row" : (coords.y - coords.y % cellSize) / cellSize - }; - }; + /** + * Transform a screen pixel-based coordinate (relative to the top-left corner of the rendered + * frame) into a sprite coordinate in column and row. + * @public + */ + ns.FrameRenderer.prototype.convertPixelCoordinatesIntoSpriteCoordinate = function(coords) { + var cellSize = this.dpi + this.gridStrokeWidth; + return { + "col" : (coords.x - coords.x % cellSize) / cellSize, + "row" : (coords.y - coords.y % cellSize) / cellSize + }; + }; - /** - * @private - */ - ns.FrameRenderer.prototype.getFramePos_ = function(index) { - return index * this.dpi + ((index - 1) * this.gridStrokeWidth); - }; + /** + * @private + */ + ns.FrameRenderer.prototype.getFramePos_ = function(index) { + return index * this.dpi + ((index - 1) * this.gridStrokeWidth); + }; - /** - * @private - */ - ns.FrameRenderer.prototype.drawGrid_ = function(canvas, width, height, col, row) { - var ctx = canvas.getContext("2d"); - ctx.lineWidth = Constants.GRID_STROKE_WIDTH; - ctx.strokeStyle = Constants.GRID_STROKE_COLOR; - for(var c=1; c < col; c++) { - ctx.moveTo(this.getFramePos_(c), 0); - ctx.lineTo(this.getFramePos_(c), height); - ctx.stroke(); - } - - for(var r=1; r < row; r++) { - ctx.moveTo(0, this.getFramePos_(r)); - ctx.lineTo(width, this.getFramePos_(r)); - ctx.stroke(); - } - }; + /** + * @private + */ + ns.FrameRenderer.prototype.drawGrid_ = function(canvas, width, height, col, row) { + var ctx = canvas.getContext("2d"); + ctx.lineWidth = Constants.GRID_STROKE_WIDTH; + ctx.strokeStyle = Constants.GRID_STROKE_COLOR; + for(var c=1; c < col; c++) { + ctx.moveTo(this.getFramePos_(c), 0); + ctx.lineTo(this.getFramePos_(c), height); + ctx.stroke(); + } + + for(var r=1; r < row; r++) { + ctx.moveTo(0, this.getFramePos_(r)); + ctx.lineTo(width, this.getFramePos_(r)); + ctx.stroke(); + } + }; - /** - * @private - */ - ns.FrameRenderer.prototype.getCanvas_ = function (frame) { - if(this.canvasConfigDirty) { - $(this.canvas).remove(); - - var col = frame.getWidth(), - row = frame.getHeight(); - - var pixelWidth = col * this.dpi + this.gridStrokeWidth * (col - 1); - var pixelHeight = row * this.dpi + this.gridStrokeWidth * (row - 1); - var canvas = pskl.CanvasUtils.createCanvas(pixelWidth, pixelHeight, ["canvas", this.className]); + /** + * @private + */ + ns.FrameRenderer.prototype.getCanvas_ = function (frame) { + if(this.canvasConfigDirty) { + $(this.canvas).remove(); + + var col = frame.getWidth(), + row = frame.getHeight(); + + var pixelWidth = col * this.dpi + this.gridStrokeWidth * (col - 1); + var pixelHeight = row * this.dpi + this.gridStrokeWidth * (row - 1); + var canvas = pskl.CanvasUtils.createCanvas(pixelWidth, pixelHeight, ["canvas", this.className]); - this.container.append(canvas); + this.container.append(canvas); - if(this.gridStrokeWidth > 0) { - this.drawGrid_(canvas, pixelWidth, pixelHeight, col, row); - } - - this.canvas = canvas; - this.canvasConfigDirty = false; - } - return this.canvas; - }; + if(this.gridStrokeWidth > 0) { + this.drawGrid_(canvas, pixelWidth, pixelHeight, col, row); + } + + this.canvas = canvas; + this.canvasConfigDirty = false; + } + return this.canvas; + }; })(); \ No newline at end of file From 7bbcbe18619c255dec42f092c40bb076ee20f031 Mon Sep 17 00:00:00 2001 From: Vince Date: Wed, 19 Jun 2013 01:51:53 +0200 Subject: [PATCH 22/22] Review comments --- js/rendering/FrameRenderer.js | 6 ++++- js/utils/UserSettings.js | 45 +++++++++++++++++++---------------- 2 files changed, 29 insertions(+), 22 deletions(-) diff --git a/js/rendering/FrameRenderer.js b/js/rendering/FrameRenderer.js index 75223860..99d30131 100644 --- a/js/rendering/FrameRenderer.js +++ b/js/rendering/FrameRenderer.js @@ -145,7 +145,11 @@ var pixelWidth = col * this.dpi + this.gridStrokeWidth * (col - 1); var pixelHeight = row * this.dpi + this.gridStrokeWidth * (row - 1); - var canvas = pskl.CanvasUtils.createCanvas(pixelWidth, pixelHeight, ["canvas", this.className]); + var classes = ['canvas']; + if (this.className) { + classes.push(this.className); + } + var canvas = pskl.CanvasUtils.createCanvas(pixelWidth, pixelHeight, classes); this.container.append(canvas); diff --git a/js/utils/UserSettings.js b/js/utils/UserSettings.js index 73946d86..89359bee 100644 --- a/js/utils/UserSettings.js +++ b/js/utils/UserSettings.js @@ -21,17 +21,18 @@ * value if not defined yet. */ get : function (key) { - this.checKeyValidity_(key); - if (key in this.cache_) { - return this.cache_[key]; + this.checkKeyValidity_(key); + if (!(key in this.cache_)) { + this.cache_[key] = + this.readFromLocalStorage_(key) || this.readFromDefaults_(key); } - return this.get_(key); + return this.cache_[key]; }, set : function (key, value) { - this.checKeyValidity_(key); + this.checkKeyValidity_(key); this.cache_[key] = value; - this.set_(key, value); + this.writeToLocalStorage_(key, value); $.publish(Events.USER_SETTINGS_CHANGED, [key, value]); }, @@ -39,14 +40,10 @@ /** * @private */ - get_ : function(key) { + readFromLocalStorage_ : function(key) { var value = window.localStorage[key]; - if (value === undefined) { - value = this.KEY_TO_DEFAULT_VALUE_MAP_[key]; - } - else { - var entry = JSON.parse(value); - value = entry.jsonValue; + if (typeof value != "undefined") { + value = JSON.parse(value); } return value; }, @@ -54,20 +51,26 @@ /** * @private */ - set_ : function(key, value) { - var entry = { 'jsonValue': value }; - window.localStorage[key] = JSON.stringify(entry); + writeToLocalStorage_ : function(key, value) { + // TODO(grosbouddha): Catch storage exception here. + window.localStorage[key] = JSON.stringify(value); }, /** * @private */ - checKeyValidity_ : function(key) { - if(key in this.KEY_TO_DEFAULT_VALUE_MAP_) { - return true; + readFromDefaults_ : function (key) { + return this.KEY_TO_DEFAULT_VALUE_MAP_[key]; + }, + + /** + * @private + */ + checkKeyValidity_ : function(key) { + if(!(key in this.KEY_TO_DEFAULT_VALUE_MAP_)) { + // TODO(grosbouddha): Define error catching strategy and throw exception from here. + console.log("UserSettings key <"+ key +"> not find in supported keys."); } - console.log("UserSettings key <"+ key +"> not find in supported keys."); - return false; } }; })(); \ No newline at end of file