mirror of
https://github.com/piskelapp/piskel.git
synced 2023-08-10 21:12:52 +03:00
Merge pull request #16 from grosbouddha/piskel-cleanup
Piskel.js cleanup
This commit is contained in:
commit
c6d338507e
@ -265,6 +265,7 @@ ul, li {
|
|||||||
left: 40%;
|
left: 40%;
|
||||||
background-color: #F9EDBE;
|
background-color: #F9EDBE;
|
||||||
padding: 7px 22px;
|
padding: 7px 22px;
|
||||||
|
padding-right: 42px;
|
||||||
border-top-left-radius: 7px;
|
border-top-left-radius: 7px;
|
||||||
border-top-right-radius: 7px;
|
border-top-right-radius: 7px;
|
||||||
font-family: ‘Arial Black’, Gadget, sans-serif;
|
font-family: ‘Arial Black’, Gadget, sans-serif;
|
||||||
@ -276,6 +277,20 @@ ul, li {
|
|||||||
z-index: 10000;
|
z-index: 10000;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.user-message .close {
|
||||||
|
position: absolute;
|
||||||
|
top: 3px;
|
||||||
|
right: 6px;
|
||||||
|
color: gray;
|
||||||
|
font-weight: bold;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.user-message .close:hover {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
/* Force apparition of scrollbars on leopard */
|
/* Force apparition of scrollbars on leopard */
|
||||||
::-webkit-scrollbar {
|
::-webkit-scrollbar {
|
||||||
-webkit-appearance: none;
|
-webkit-appearance: none;
|
||||||
|
@ -43,7 +43,7 @@
|
|||||||
<div class="color-tool">
|
<div class="color-tool">
|
||||||
<input id="color-picker" class="color {hash:true}" type="text" value=""/>
|
<input id="color-picker" class="color {hash:true}" type="text" value=""/>
|
||||||
|
|
||||||
<ul id="palette" class="palette" onclick="piskel.onPaletteClick(event)">
|
<ul id="palette" class="palette">
|
||||||
<span class="palette-color transparent-color" data-color="TRANSPARENT" title="Transparent"></span>
|
<span class="palette-color transparent-color" data-color="TRANSPARENT" title="Transparent"></span>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
@ -75,7 +75,9 @@
|
|||||||
<script src="js/lib/jsColor_1_4_0/jscolor.js"></script>
|
<script src="js/lib/jsColor_1_4_0/jscolor.js"></script>
|
||||||
|
|
||||||
<!-- Application libraries-->
|
<!-- Application libraries-->
|
||||||
<script src="js/frameSheetModel.js"></script>
|
<script src="js/FrameSheetModel.js"></script>
|
||||||
|
<script src="js/LocalStorageService.js"></script>
|
||||||
|
<script src="js/Notification.js"></script>
|
||||||
<script src="js/drawingtools/BaseTool.js"></script>
|
<script src="js/drawingtools/BaseTool.js"></script>
|
||||||
<script src="js/drawingtools/SimplePen.js"></script>
|
<script src="js/drawingtools/SimplePen.js"></script>
|
||||||
<script src="js/drawingtools/Eraser.js"></script>
|
<script src="js/drawingtools/Eraser.js"></script>
|
||||||
|
@ -1,3 +1,6 @@
|
|||||||
var Constants = {
|
var Constants = {
|
||||||
TRANSPARENT_COLOR : "TRANSPARENT"
|
|
||||||
|
DEFAULT_PEN_COLOR : '#000000',
|
||||||
|
TRANSPARENT_COLOR : "TRANSPARENT",
|
||||||
|
PISKEL_SERVICE_URL: 'http://2.piskel-app.appspot.com'
|
||||||
};
|
};
|
22
js/Events.js
22
js/Events.js
@ -1,6 +1,26 @@
|
|||||||
Events = {
|
Events = {
|
||||||
|
|
||||||
TOOL_SELECTED : "TOOL_SELECTED",
|
TOOL_SELECTED : "TOOL_SELECTED",
|
||||||
|
COLOR_SELECTED: "COLOR_SELECTED",
|
||||||
|
COLOR_USED: "COLOR_USED",
|
||||||
|
|
||||||
|
/**
|
||||||
|
* When this event is emitted, a request is sent to the localstorage
|
||||||
|
* Service to save the current framesheet. The storage service
|
||||||
|
* may not immediately store data (internal throttling of requests).
|
||||||
|
*/
|
||||||
|
LOCALSTORAGE_REQUEST: "LOCALSTORAGE_REQUEST",
|
||||||
|
|
||||||
CANVAS_RIGHT_CLICKED: "CANVAS_RIGHT_CLICKED",
|
CANVAS_RIGHT_CLICKED: "CANVAS_RIGHT_CLICKED",
|
||||||
CANVAS_RIGHT_CLICK_RELEASED: "CANVAS_RIGHT_CLICK_RELEASED"
|
CANVAS_RIGHT_CLICK_RELEASED: "CANVAS_RIGHT_CLICK_RELEASED",
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Event to requset a refresh of the display.
|
||||||
|
* A bit overkill but, it's just workaround in our current drawing system.
|
||||||
|
* TODO: Remove or rework when redraw system is refactored.
|
||||||
|
*/
|
||||||
|
REFRESH: "REFRESH",
|
||||||
|
|
||||||
|
SHOW_NOTIFICATION: "SHOW_NOTIFICATION",
|
||||||
|
HIDE_NOTIFICATION: "HIDE_NOTIFICATION"
|
||||||
};
|
};
|
95
js/LocalStorageService.js
Normal file
95
js/LocalStorageService.js
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
/*
|
||||||
|
* @provide pskl.LocalStrageService
|
||||||
|
*
|
||||||
|
* @require Constants
|
||||||
|
* @require Events
|
||||||
|
*/
|
||||||
|
$.namespace("pskl");
|
||||||
|
|
||||||
|
pskl.LocalStorageService = (function() {
|
||||||
|
|
||||||
|
var frameSheet_;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var localStorageThrottler_ = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var persistToLocalStorageRequest_ = function() {
|
||||||
|
// Persist to localStorage when drawing. We throttle localStorage accesses
|
||||||
|
// for high frequency drawing (eg mousemove).
|
||||||
|
if(localStorageThrottler_ != null) {
|
||||||
|
window.clearTimeout(localStorageThrottler_);
|
||||||
|
}
|
||||||
|
localStorageThrottler_ = window.setTimeout(function() {
|
||||||
|
persistToLocalStorage_();
|
||||||
|
localStorageThrottler_ = null;
|
||||||
|
}, 1000);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var persistToLocalStorage_ = function() {
|
||||||
|
console.log('[LocalStorage service]: Snapshot stored')
|
||||||
|
window.localStorage['snapShot'] = frameSheet_.serialize();
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var restoreFromLocalStorage_ = function() {
|
||||||
|
frameSheet_.deserialize(window.localStorage['snapShot']);
|
||||||
|
// Model updated, redraw everything:
|
||||||
|
$.publish(Events.REFRESH);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var cleanLocalStorage_ = function() {
|
||||||
|
console.log('[LocalStorage service]: Snapshot removed')
|
||||||
|
delete window.localStorage['snapShot'];
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
init: function(frameSheet) {
|
||||||
|
|
||||||
|
if(frameSheet == undefined) {
|
||||||
|
throw "Bad LocalStorageService initialization: <undefined frameSheet>"
|
||||||
|
}
|
||||||
|
frameSheet_ = frameSheet;
|
||||||
|
|
||||||
|
$.subscribe(Events.LOCALSTORAGE_REQUEST, persistToLocalStorageRequest_);
|
||||||
|
},
|
||||||
|
|
||||||
|
// TODO(vincz): Find a good place to put this UI rendering, a service should not render UI.
|
||||||
|
displayRestoreNotification: function() {
|
||||||
|
if(window.localStorage && window.localStorage['snapShot']) {
|
||||||
|
var reloadLink = "<a href='#' class='localstorage-restore onclick='piskel.restoreFromLocalStorage()'>reload</a>";
|
||||||
|
var discardLink = "<a href='#' class='localstorage-discard' onclick='piskel.cleanLocalStorage()'>discard</a>";
|
||||||
|
var content = "Non saved version found. " + reloadLink + " or " + discardLink;
|
||||||
|
|
||||||
|
$.publish(Events.SHOW_NOTIFICATION, [{
|
||||||
|
"content": content,
|
||||||
|
"behavior": function(rootNode) {
|
||||||
|
rootNode = $(rootNode);
|
||||||
|
rootNode.click(function(evt) {
|
||||||
|
var target = $(evt.target);
|
||||||
|
if(target.hasClass("localstorage-restore")) {
|
||||||
|
restoreFromLocalStorage_();
|
||||||
|
}
|
||||||
|
else if (target.hasClass("localstorage-discard")) {
|
||||||
|
cleanLocalStorage_();
|
||||||
|
}
|
||||||
|
$.publish(Events.HIDE_NOTIFICATION);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
})();
|
39
js/Notification.js
Normal file
39
js/Notification.js
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
/*
|
||||||
|
* @provide pskl.NotificationService
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
$.namespace("pskl");
|
||||||
|
|
||||||
|
pskl.NotificationService = (function() {
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var displayMessage_ = function (evt, messageInfo) {
|
||||||
|
var message = document.createElement('div');
|
||||||
|
message.id = "user-message";
|
||||||
|
message.className = "user-message";
|
||||||
|
message.innerHTML = messageInfo.content;
|
||||||
|
message.innerHTML = message.innerHTML + "<div title='Close message' class='close'>x</div>";
|
||||||
|
document.body.appendChild(message);
|
||||||
|
$(message).find(".close").click(removeMessage_);
|
||||||
|
if(messageInfo.behavior) messageInfo.behavior(message);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var removeMessage_ = function (evt) {
|
||||||
|
var message = $("#user-message");
|
||||||
|
if (message.length) {
|
||||||
|
message.remove();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
init: function() {
|
||||||
|
$.subscribe(Events.SHOW_NOTIFICATION, displayMessage_);
|
||||||
|
$.subscribe(Events.HIDE_NOTIFICATION, removeMessage_);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
})();
|
@ -5,9 +5,12 @@
|
|||||||
* @require Events
|
* @require Events
|
||||||
* @require pskl.drawingtools
|
* @require pskl.drawingtools
|
||||||
*/
|
*/
|
||||||
|
$.namespace("pskl");
|
||||||
|
|
||||||
pskl.ToolSelector = (function() {
|
pskl.ToolSelector = (function() {
|
||||||
|
|
||||||
|
var paletteColors = [];
|
||||||
|
|
||||||
var toolInstances = {
|
var toolInstances = {
|
||||||
"simplePen" : new pskl.drawingtools.SimplePen(),
|
"simplePen" : new pskl.drawingtools.SimplePen(),
|
||||||
"eraser" : new pskl.drawingtools.Eraser(),
|
"eraser" : new pskl.drawingtools.Eraser(),
|
||||||
@ -19,7 +22,6 @@ pskl.ToolSelector = (function() {
|
|||||||
var previousSelectedTool = toolInstances.simplePen;
|
var previousSelectedTool = toolInstances.simplePen;
|
||||||
|
|
||||||
var selectTool_ = function(tool) {
|
var selectTool_ = function(tool) {
|
||||||
|
|
||||||
var maincontainer = $("body");
|
var maincontainer = $("body");
|
||||||
var previousSelectedToolClass = maincontainer.data("selected-tool-class");
|
var previousSelectedToolClass = maincontainer.data("selected-tool-class");
|
||||||
if(previousSelectedToolClass) {
|
if(previousSelectedToolClass) {
|
||||||
@ -46,6 +48,9 @@ pskl.ToolSelector = (function() {
|
|||||||
$.publish(Events.TOOL_SELECTED, [tool]);
|
$.publish(Events.TOOL_SELECTED, [tool]);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
var onToolIconClicked_ = function(evt) {
|
var onToolIconClicked_ = function(evt) {
|
||||||
var target = $(evt.target);
|
var target = $(evt.target);
|
||||||
var clickedTool = target.closest(".tool-icon");
|
var clickedTool = target.closest(".tool-icon");
|
||||||
@ -63,8 +68,76 @@ pskl.ToolSelector = (function() {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var onPickerChange_ = function(evt) {
|
||||||
|
var inputPicker = $(evt.target);
|
||||||
|
$.publish(Events.COLOR_SELECTED, [inputPicker.val()]);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
var addColorToPalette_ = function (color) {
|
||||||
|
if (paletteColors.indexOf(color) == -1) {
|
||||||
|
var paletteEl = $("#palette");
|
||||||
|
var colorEl = document.createElement("li");
|
||||||
|
colorEl.className = "palette-color";
|
||||||
|
colorEl.setAttribute("data-color", color);
|
||||||
|
colorEl.setAttribute("title", color);
|
||||||
|
colorEl.style.background = color;
|
||||||
|
paletteEl[0].appendChild(colorEl);
|
||||||
|
paletteColors.push(color);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @private
|
||||||
|
*/
|
||||||
|
onPaletteColorClick_ = function (event) {
|
||||||
|
var selectedColor = $(event.target).data("color");
|
||||||
|
var colorPicker = $('#color-picker');
|
||||||
|
if (selectedColor == Constants.TRANSPARENT_COLOR) {
|
||||||
|
// We can set the current palette color to transparent.
|
||||||
|
// You can then combine this transparent color with an advanced
|
||||||
|
// tool for customized deletions.
|
||||||
|
// Eg: bucket + transparent: Delete a colored area
|
||||||
|
// Stroke + transparent: hollow out the equivalent of a stroke
|
||||||
|
|
||||||
|
// The colorpicker can't be set to a transparent state.
|
||||||
|
// We set its background to white and insert the
|
||||||
|
// string "TRANSPARENT" to mimic this state:
|
||||||
|
colorPicker[0].color.fromString("#fff");
|
||||||
|
colorPicker.val("TRANSPARENT");
|
||||||
|
} else {
|
||||||
|
colorPicker[0].color.fromString(selectedColor);
|
||||||
|
}
|
||||||
|
$.publish(Events.COLOR_SELECTED, [selectedColor])
|
||||||
|
};
|
||||||
|
|
||||||
return {
|
return {
|
||||||
init: function() {
|
init: function() {
|
||||||
|
|
||||||
|
// Initialize tool:
|
||||||
|
// Set SimplePen as default selected tool:
|
||||||
|
selectTool_(toolInstances.simplePen);
|
||||||
|
// Activate listener on tool panel:
|
||||||
|
$("#tools-container").click(onToolIconClicked_);
|
||||||
|
|
||||||
|
// Initialize colorpicker:
|
||||||
|
var colorPicker = $('#color-picker');
|
||||||
|
colorPicker.val(Constants.DEFAULT_PEN_COLOR);
|
||||||
|
colorPicker.change(onPickerChange_);
|
||||||
|
|
||||||
|
// Initialize palette:
|
||||||
|
$("#palette").click(onPaletteColorClick_);
|
||||||
|
$.subscribe(Events.COLOR_USED, function(evt, color) {
|
||||||
|
addColorToPalette_(color);
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
// Special right click handlers (select the eraser tool)
|
||||||
$.subscribe(Events.CANVAS_RIGHT_CLICKED, function() {
|
$.subscribe(Events.CANVAS_RIGHT_CLICKED, function() {
|
||||||
previousSelectedTool = currentSelectedTool;
|
previousSelectedTool = currentSelectedTool;
|
||||||
currentSelectedTool = toolInstances.eraser;
|
currentSelectedTool = toolInstances.eraser;
|
||||||
@ -75,12 +148,9 @@ pskl.ToolSelector = (function() {
|
|||||||
currentSelectedTool = previousSelectedTool;
|
currentSelectedTool = previousSelectedTool;
|
||||||
$.publish(Events.TOOL_SELECTED, [currentSelectedTool]);
|
$.publish(Events.TOOL_SELECTED, [currentSelectedTool]);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Set SimplePen as default selected tool:
|
|
||||||
selectTool_(toolInstances.simplePen);
|
|
||||||
|
|
||||||
// Activate listener on tool panel:
|
|
||||||
$("#tools-container").click(onToolIconClicked_);
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
})()
|
})();
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
@ -21,8 +21,8 @@
|
|||||||
context.clearRect(col * dpi, row * dpi, dpi, dpi);
|
context.clearRect(col * dpi, row * dpi, dpi, dpi);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// TODO(vincz): Remove this global access to piskel when Palette component is created.
|
// TODO(vincz): Found a better design to update the palette, it's called too frequently.
|
||||||
piskel.addColorToPalette(color);
|
$.publish(Events.COLOR_USED, [color]);
|
||||||
context.fillStyle = color;
|
context.fillStyle = color;
|
||||||
context.fillRect(col * dpi, row * dpi, dpi, dpi);
|
context.fillRect(col * dpi, row * dpi, dpi, dpi);
|
||||||
}
|
}
|
||||||
|
@ -20,7 +20,7 @@
|
|||||||
|
|
||||||
// Change model:
|
// Change model:
|
||||||
frame[col][row] = Constants.TRANSPARENT_COLOR;
|
frame[col][row] = Constants.TRANSPARENT_COLOR;
|
||||||
|
|
||||||
// Draw on canvas:
|
// Draw on canvas:
|
||||||
// TODO: Remove that when we have the centralized redraw loop
|
// TODO: Remove that when we have the centralized redraw loop
|
||||||
this.drawPixelInCanvas(col, row, canvas, Constants.TRANSPARENT_COLOR, dpi);
|
this.drawPixelInCanvas(col, row, canvas, Constants.TRANSPARENT_COLOR, dpi);
|
||||||
@ -33,4 +33,9 @@
|
|||||||
this.applyToolAt(col, row, frame, color, canvas, dpi);
|
this.applyToolAt(col, row, frame, color, canvas, dpi);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ns.Eraser.prototype.releaseToolAt = function(col, row, frame, color, canvas, dpi) {
|
||||||
|
// TODO: Create a afterRelease event hook or put that deep in the model
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
|
};
|
||||||
|
|
||||||
})();
|
})();
|
@ -21,12 +21,18 @@
|
|||||||
var targetColor = pskl.utils.normalizeColor(frame[col][row]);
|
var targetColor = pskl.utils.normalizeColor(frame[col][row]);
|
||||||
//this.recursiveFloodFill_(frame, col, row, targetColor, color);
|
//this.recursiveFloodFill_(frame, col, row, targetColor, color);
|
||||||
this.queueLinearFloodFill_(frame, col, row, targetColor, color);
|
this.queueLinearFloodFill_(frame, col, row, targetColor, color);
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
|
|
||||||
// Draw in canvas:
|
// Draw in canvas:
|
||||||
// TODO: Remove that when we have the centralized redraw loop
|
// TODO: Remove that when we have the centralized redraw loop
|
||||||
this.drawFrameInCanvas(frame, canvas, dpi);
|
this.drawFrameInCanvas(frame, canvas, dpi);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ns.PaintBucket.prototype.releaseToolAt = function(col, row, frame, color, canvas, dpi) {
|
||||||
|
// TODO: Create a afterRelease event hook or put that deep in the model
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Flood-fill (node, target-color, replacement-color):
|
* Flood-fill (node, target-color, replacement-color):
|
||||||
* 1. Set Q to the empty queue.
|
* 1. Set Q to the empty queue.
|
||||||
|
@ -81,9 +81,12 @@
|
|||||||
// TODO: Remove that when we have the centralized redraw loop
|
// TODO: Remove that when we have the centralized redraw loop
|
||||||
this.drawPixelInCanvas(strokePoints[i].col, strokePoints[i].row, canvas, color, dpi);
|
this.drawPixelInCanvas(strokePoints[i].col, strokePoints[i].row, canvas, color, dpi);
|
||||||
}
|
}
|
||||||
|
|
||||||
// For now, we are done with the stroke tool and don't need an overlay anymore:
|
// For now, we are done with the stroke tool and don't need an overlay anymore:
|
||||||
this.removeCanvasOverlays();
|
this.removeCanvasOverlays();
|
||||||
|
|
||||||
|
// TODO: Create a afterRelease event hook or put that deep in the model
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -52,4 +52,9 @@
|
|||||||
this.previousCol = col;
|
this.previousCol = col;
|
||||||
this.previousRow = row;
|
this.previousRow = row;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
ns.SimplePen.prototype.releaseToolAt = function(col, row, frame, color, canvas, dpi) {
|
||||||
|
// TODO: Create a afterRelease event hook or out that deep in the model
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
|
};
|
||||||
})();
|
})();
|
||||||
|
@ -97,6 +97,9 @@
|
|||||||
|
|
||||||
// For now, we are done with the stroke tool and don't need an overlay anymore:
|
// For now, we are done with the stroke tool and don't need an overlay anymore:
|
||||||
this.removeCanvasOverlays();
|
this.removeCanvasOverlays();
|
||||||
|
|
||||||
|
// TODO: Create a afterRelease event hook or out that deep in the model
|
||||||
|
$.publish(Events.FRAMESHEET_UPDATED);
|
||||||
};
|
};
|
||||||
|
|
||||||
})();
|
})();
|
||||||
|
@ -1,5 +1,7 @@
|
|||||||
|
|
||||||
var FrameSheetModel = (function() {
|
$.namespace("pskl");
|
||||||
|
|
||||||
|
pskl.FrameSheetModel = (function() {
|
||||||
|
|
||||||
var inst;
|
var inst;
|
||||||
var frames = [];
|
var frames = [];
|
||||||
|
@ -7,7 +7,7 @@
|
|||||||
var o = $({});
|
var o = $({});
|
||||||
|
|
||||||
$.subscribe = function() {
|
$.subscribe = function() {
|
||||||
console.log("SUBSCRIBE: " + arguments[0]);
|
//console.log("SUBSCRIBE: " + arguments[0]);
|
||||||
o.on.apply(o, arguments);
|
o.on.apply(o, arguments);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -16,7 +16,7 @@
|
|||||||
};
|
};
|
||||||
|
|
||||||
$.publish = function() {
|
$.publish = function() {
|
||||||
console.log("PUBLISH: " + arguments[0]);
|
//console.log("PUBLISH: " + arguments[0]);
|
||||||
o.trigger.apply(o, arguments);
|
o.trigger.apply(o, arguments);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
171
js/piskel.js
171
js/piskel.js
@ -5,11 +5,11 @@
|
|||||||
$.namespace("pskl");
|
$.namespace("pskl");
|
||||||
|
|
||||||
(function () {
|
(function () {
|
||||||
var frameSheet,
|
|
||||||
|
|
||||||
// Constants:
|
/**
|
||||||
DEFAULT_PEN_COLOR = '#000000',
|
* FrameSheetModel instance.
|
||||||
PISKEL_SERVICE_URL = 'http://2.piskel-app.appspot.com',
|
*/
|
||||||
|
var frameSheet,
|
||||||
|
|
||||||
// Temporary zoom implementation to easily get bigger canvases to
|
// Temporary zoom implementation to easily get bigger canvases to
|
||||||
// see how good perform critical algorithms on big canvas.
|
// see how good perform critical algorithms on big canvas.
|
||||||
@ -33,15 +33,13 @@ $.namespace("pskl");
|
|||||||
drawingAreaContainer,
|
drawingAreaContainer,
|
||||||
drawingAreaCanvas,
|
drawingAreaCanvas,
|
||||||
previewCanvas,
|
previewCanvas,
|
||||||
paletteEl,
|
|
||||||
|
|
||||||
// States:
|
// States:
|
||||||
isClicked = false,
|
isClicked = false,
|
||||||
isRightClicked = false,
|
isRightClicked = false,
|
||||||
activeFrameIndex = -1,
|
activeFrameIndex = -1,
|
||||||
animIndex = 0,
|
animIndex = 0,
|
||||||
penColor = DEFAULT_PEN_COLOR,
|
penColor = Constants.DEFAULT_PEN_COLOR,
|
||||||
paletteColors = [],
|
|
||||||
currentFrame = null;
|
currentFrame = null;
|
||||||
currentToolBehavior = null,
|
currentToolBehavior = null,
|
||||||
previousMousemoveTime = 0,
|
previousMousemoveTime = 0,
|
||||||
@ -53,25 +51,29 @@ $.namespace("pskl");
|
|||||||
} else {
|
} else {
|
||||||
return "#" + color;
|
return "#" + color;
|
||||||
}
|
}
|
||||||
},
|
};
|
||||||
|
|
||||||
// setTimeout/setInterval references:
|
|
||||||
localStorageThrottler = null
|
|
||||||
;
|
|
||||||
|
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Main application controller
|
||||||
|
*/
|
||||||
var piskel = {
|
var piskel = {
|
||||||
|
|
||||||
init : function () {
|
init : function () {
|
||||||
frameSheet = FrameSheetModel.getInstance(framePixelWidth, framePixelHeight);
|
frameSheet = pskl.FrameSheetModel.getInstance(framePixelWidth, framePixelHeight);
|
||||||
frameSheet.addEmptyFrame();
|
frameSheet.addEmptyFrame();
|
||||||
this.setActiveFrame(0);
|
this.setActiveFrame(0);
|
||||||
|
|
||||||
|
pskl.NotificationService.init();
|
||||||
|
pskl.LocalStorageService.init(frameSheet);
|
||||||
|
|
||||||
|
// TODO: Add comments
|
||||||
var frameId = this.getFrameIdFromUrl();
|
var frameId = this.getFrameIdFromUrl();
|
||||||
if (frameId) {
|
if (frameId) {
|
||||||
this.displayMessage("Loading animation with id : [" + frameId + "]");
|
$.publish(Events.SHOW_NOTIFICATION, [{"content": "Loading animation with id : [" + frameId + "]"}]);
|
||||||
this.loadFramesheetFromService(frameId);
|
this.loadFramesheetFromService(frameId);
|
||||||
} else {
|
} else {
|
||||||
this.finishInit();
|
this.finishInit();
|
||||||
|
pskl.LocalStorageService.displayRestoreNotification();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -82,15 +84,22 @@ $.namespace("pskl");
|
|||||||
currentToolBehavior = toolBehavior;
|
currentToolBehavior = toolBehavior;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.initPalette();
|
$.subscribe(Events.COLOR_SELECTED, function(evt, color) {
|
||||||
this.initDrawingArea();
|
console.log("Color selected: ", color);
|
||||||
this.initPreviewSlideshow();
|
penColor = color;
|
||||||
this.initAnimationPreview();
|
});
|
||||||
this.initColorPicker();
|
|
||||||
this.initLocalStorageBackup();
|
|
||||||
pskl.ToolSelector.init();
|
|
||||||
|
|
||||||
|
$.subscribe(Events.REFRESH, function() {
|
||||||
|
piskel.setActiveFrameAndRedraw(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
// TODO: Move this into their service or behavior files:
|
||||||
|
this.initDrawingArea();
|
||||||
|
this.initPreviewSlideshow();
|
||||||
|
this.initAnimationPreview();
|
||||||
this.startAnimation();
|
this.startAnimation();
|
||||||
|
|
||||||
|
pskl.ToolSelector.init();
|
||||||
},
|
},
|
||||||
|
|
||||||
getFrameIdFromUrl : function() {
|
getFrameIdFromUrl : function() {
|
||||||
@ -102,73 +111,23 @@ $.namespace("pskl");
|
|||||||
|
|
||||||
loadFramesheetFromService : function (frameId) {
|
loadFramesheetFromService : function (frameId) {
|
||||||
var xhr = new XMLHttpRequest();
|
var xhr = new XMLHttpRequest();
|
||||||
xhr.open('GET', PISKEL_SERVICE_URL + '/get?l=' + frameId, true);
|
xhr.open('GET', Constants.PISKEL_SERVICE_URL + '/get?l=' + frameId, true);
|
||||||
xhr.responseType = 'text';
|
xhr.responseType = 'text';
|
||||||
|
|
||||||
xhr.onload = function(e) {
|
xhr.onload = function(e) {
|
||||||
frameSheet.deserialize(this.responseText);
|
frameSheet.deserialize(this.responseText);
|
||||||
piskel.removeMessage();
|
$.publish(Events.HIDE_NOTIFICATION);
|
||||||
piskel.finishInit();
|
piskel.finishInit();
|
||||||
};
|
};
|
||||||
|
|
||||||
xhr.onerror = function () {
|
xhr.onerror = function () {
|
||||||
piskel.removeMessage();
|
$.publish(Events.HIDE_NOTIFICATION);
|
||||||
piskel.finishInit();
|
piskel.finishInit();
|
||||||
};
|
};
|
||||||
|
|
||||||
xhr.send();
|
xhr.send();
|
||||||
},
|
},
|
||||||
|
|
||||||
initLocalStorageBackup: function() {
|
|
||||||
if(window.localStorage && window.localStorage['snapShot']) {
|
|
||||||
var reloadLink = "<a href='#' onclick='piskel.restoreFromLocalStorage()'>reload</a>";
|
|
||||||
var discardLink = "<a href='#' onclick='piskel.cleanLocalStorage()'>discard</a>";
|
|
||||||
this.displayMessage("Non saved version found. " + reloadLink + " or " + discardLink);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
displayMessage : function (content) {
|
|
||||||
var message = document.createElement('div');
|
|
||||||
message.id = "user-message";
|
|
||||||
message.className = "user-message";
|
|
||||||
message.innerHTML = content;
|
|
||||||
message.onclick = this.removeMessage;
|
|
||||||
document.body.appendChild(message);
|
|
||||||
},
|
|
||||||
|
|
||||||
removeMessage : function () {
|
|
||||||
var message = $("#user-message");
|
|
||||||
if (message.length) {
|
|
||||||
message.remove();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
persistToLocalStorageRequest: function() {
|
|
||||||
// Persist to localStorage when drawing. We throttle localStorage accesses
|
|
||||||
// for high frequency drawing (eg mousemove).
|
|
||||||
if(localStorageThrottler != null) {
|
|
||||||
window.clearTimeout(localStorageThrottler);
|
|
||||||
}
|
|
||||||
localStorageThrottler = window.setTimeout(function() {
|
|
||||||
piskel.persistToLocalStorage();
|
|
||||||
localStorageThrottler = null;
|
|
||||||
}, 1000);
|
|
||||||
},
|
|
||||||
|
|
||||||
persistToLocalStorage: function() {
|
|
||||||
console.log('persited')
|
|
||||||
window.localStorage['snapShot'] = frameSheet.serialize();
|
|
||||||
},
|
|
||||||
|
|
||||||
restoreFromLocalStorage: function() {
|
|
||||||
frameSheet.deserialize(window.localStorage['snapShot']);
|
|
||||||
this.setActiveFrameAndRedraw(0);
|
|
||||||
},
|
|
||||||
|
|
||||||
cleanLocalStorage: function() {
|
|
||||||
delete window.localStorage['snapShot'];
|
|
||||||
},
|
|
||||||
|
|
||||||
setActiveFrame: function(index) {
|
setActiveFrame: function(index) {
|
||||||
activeFrameIndex = index;
|
activeFrameIndex = index;
|
||||||
currentFrame = frameSheet.getFrameByIndex(activeFrameIndex)
|
currentFrame = frameSheet.getFrameByIndex(activeFrameIndex)
|
||||||
@ -194,33 +153,6 @@ $.namespace("pskl");
|
|||||||
return activeFrameIndex;
|
return activeFrameIndex;
|
||||||
},
|
},
|
||||||
|
|
||||||
initColorPicker: function() {
|
|
||||||
this.colorPicker = $('#color-picker');
|
|
||||||
this.colorPicker.val(DEFAULT_PEN_COLOR);
|
|
||||||
this.colorPicker.change(this.onPickerChange.bind(this));
|
|
||||||
},
|
|
||||||
|
|
||||||
onPickerChange : function(evt) {
|
|
||||||
var inputPicker = $(evt.target);
|
|
||||||
penColor = _normalizeColor(inputPicker.val());
|
|
||||||
},
|
|
||||||
|
|
||||||
initPalette : function (color) {
|
|
||||||
paletteEl = $('#palette')[0];
|
|
||||||
},
|
|
||||||
|
|
||||||
addColorToPalette : function (color) {
|
|
||||||
if (color && color != Constants.TRANSPARENT_COLOR && paletteColors.indexOf(color) == -1) {
|
|
||||||
var colorEl = document.createElement("li");
|
|
||||||
colorEl.className = "palette-color";
|
|
||||||
colorEl.setAttribute("data-color", color);
|
|
||||||
colorEl.setAttribute("title", color);
|
|
||||||
colorEl.style.background = color;
|
|
||||||
paletteEl.appendChild(colorEl);
|
|
||||||
paletteColors.push(color);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
initDrawingArea : function() {
|
initDrawingArea : function() {
|
||||||
drawingAreaContainer = $('#drawing-canvas-container')[0];
|
drawingAreaContainer = $('#drawing-canvas-container')[0];
|
||||||
|
|
||||||
@ -394,8 +326,8 @@ $.namespace("pskl");
|
|||||||
penColor,
|
penColor,
|
||||||
drawingAreaCanvas,
|
drawingAreaCanvas,
|
||||||
drawingCanvasDpi);
|
drawingCanvasDpi);
|
||||||
|
|
||||||
piskel.persistToLocalStorageRequest();
|
$.publish(Events.LOCALSTORAGE_REQUEST);
|
||||||
},
|
},
|
||||||
|
|
||||||
onCanvasMousemove : function (event) {
|
onCanvasMousemove : function (event) {
|
||||||
@ -417,7 +349,7 @@ $.namespace("pskl");
|
|||||||
// TODO(vincz): Find a way to move that to the model instead of being at the interaction level.
|
// TODO(vincz): Find a way to move that to the model instead of being at the interaction level.
|
||||||
// Eg when drawing, it may make sense to have it here. However for a non drawing tool,
|
// Eg when drawing, it may make sense to have it here. However for a non drawing tool,
|
||||||
// you don't need to draw anything when mousemoving and you request useless localStorage.
|
// you don't need to draw anything when mousemoving and you request useless localStorage.
|
||||||
piskel.persistToLocalStorageRequest();
|
$.publish(Events.LOCALSTORAGE_REQUEST);
|
||||||
}
|
}
|
||||||
previousMousemoveTime = currentTime;
|
previousMousemoveTime = currentTime;
|
||||||
}
|
}
|
||||||
@ -477,28 +409,6 @@ $.namespace("pskl");
|
|||||||
return false;
|
return false;
|
||||||
},
|
},
|
||||||
|
|
||||||
onPaletteClick : function (event) {
|
|
||||||
var color = $(event.target).data("color");
|
|
||||||
var colorPicker = $('#color-picker');
|
|
||||||
if (color == "TRANSPARENT") {
|
|
||||||
// We can set the current palette color to transparent.
|
|
||||||
// You can then combine this transparent color with an advanced
|
|
||||||
// tool for customized deletions.
|
|
||||||
// Eg: bucket + transparent: Delete a colored area
|
|
||||||
// Stroke + transparent: hollow out the equivalent of a stroke
|
|
||||||
penColor = Constants.TRANSPARENT_COLOR;
|
|
||||||
|
|
||||||
// The colorpicker can't be set to a transparent state.
|
|
||||||
// We set its background to white and insert the
|
|
||||||
// string "TRANSPARENT" to mimic this state:
|
|
||||||
colorPicker[0].color.fromString("#fff");
|
|
||||||
colorPicker.val("TRANSPARENT");
|
|
||||||
} else if (null !== color) {
|
|
||||||
colorPicker[0].color.fromString(color);
|
|
||||||
penColor = color;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
getRelativeCoordinates : function (x, y) {
|
getRelativeCoordinates : function (x, y) {
|
||||||
var canvasRect = drawingAreaCanvas.getBoundingClientRect();
|
var canvasRect = drawingAreaCanvas.getBoundingClientRect();
|
||||||
return {
|
return {
|
||||||
@ -518,11 +428,12 @@ $.namespace("pskl");
|
|||||||
|
|
||||||
// TODO(julz): Create package ?
|
// TODO(julz): Create package ?
|
||||||
storeSheet : function (event) {
|
storeSheet : function (event) {
|
||||||
|
// TODO Refactor using jquery ?
|
||||||
var xhr = new XMLHttpRequest();
|
var xhr = new XMLHttpRequest();
|
||||||
var formData = new FormData();
|
var formData = new FormData();
|
||||||
formData.append('framesheet_content', frameSheet.serialize());
|
formData.append('framesheet_content', frameSheet.serialize());
|
||||||
formData.append('fps_speed', $('#preview-fps').val());
|
formData.append('fps_speed', $('#preview-fps').val());
|
||||||
xhr.open('POST', PISKEL_SERVICE_URL + "/store", true);
|
xhr.open('POST', Constants.PISKEL_SERVICE_URL + "/store", true);
|
||||||
xhr.onload = function(e) {
|
xhr.onload = function(e) {
|
||||||
if (this.status == 200) {
|
if (this.status == 200) {
|
||||||
var baseUrl = window.location.href.replace(window.location.search, "");
|
var baseUrl = window.location.href.replace(window.location.search, "");
|
||||||
@ -541,4 +452,4 @@ $.namespace("pskl");
|
|||||||
window.piskel = piskel;
|
window.piskel = piskel;
|
||||||
piskel.init();
|
piskel.init();
|
||||||
|
|
||||||
})(function(id){return document.getElementById(id)});
|
})();
|
||||||
|
Loading…
x
Reference in New Issue
Block a user