diff --git a/demo/build/darkroom.css b/demo/build/darkroom.css index 23cdaea..3d3ee18 100644 --- a/demo/build/darkroom.css +++ b/demo/build/darkroom.css @@ -21,8 +21,9 @@ .darkroom-toolbar { display: flex; flex-wrap: wrap; - position: initial; - top: -45px; + position: sticky; + position: -webkit-sticky; + top: 10px; left: 0; background: #444; height: auto; @@ -81,4 +82,4 @@ height: 24px; fill: #fff; } -/*# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGFya3Jvb20uY3NzIiwic291cmNlcyI6WyJkYXJrcm9vbS5zY3NzIiwiX2xheW91dC5zY3NzIiwiX3Rvb2xiYXIuc2NzcyJdLCJzb3VyY2VzQ29udGVudCI6WyJAaW1wb3J0ICdsYXlvdXQnO1xuQGltcG9ydCAndG9vbGJhcic7XG4iLCIuZGFya3Jvb20tY29udGFpbmVyIHtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuXG4uZGFya3Jvb20taW1hZ2UtY29udGFpbmVyIHtcbiAgdG9wOiAwO1xuICBsZWZ0OiAwO1xufVxuXG4uY2FudmFzLWNvbnRhaW5lciBjYW52YXN7XG4gIGhlaWdodDogYXV0byAhaW1wb3J0YW50O1xuICB3aWR0aDogYXV0byAhaW1wb3J0YW50O1xuICBtYXgtd2lkdGg6IDEwMCU7XG59XG5cbi5jYW52YXMtY29udGFpbmVyIC51cHBlci1jYW52YXMge1xuICBwb3NpdGlvbjogcmVsYXRpdmUgIWltcG9ydGFudDtcbn1cblxuLmNhbnZhcy1jb250YWluZXIge1xuICBoZWlnaHQ6IGF1dG8gIWltcG9ydGFudDtcbiAgd2lkdGg6IGF1dG8gIWltcG9ydGFudDtcbiAgbWF4LXdpZHRoOiAxMDAlO1xufVxuXG4uZGFya3Jvb20taW1hZ2UtY29udGFpbmVyIGltZyB7XG4gIC8vIGRpc3BsYXk6IG5vbmU7XG59XG4iLCIvL1xuLy8gVG9vbGJhclxuLy9cbi5kYXJrcm9vbS10b29sYmFyIHtcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC13cmFwOiB3cmFwO1xuICBwb3NpdGlvbjogaW5pdGlhbDtcbiAgdG9wOiAtNDVweDtcbiAgbGVmdDogMDtcbiAgYmFja2dyb3VuZDogIzQ0NDtcbiAgaGVpZ2h0OiBhdXRvO1xuICBtaW4td2lkdGg6IDQwcHg7XG4gIHdpZHRoOiBmaXQtY29udGVudDtcbiAgd2lkdGg6IC1tb3otZml0LWNvbnRlbnQ7XG4gIHotaW5kZXg6IDk5O1xuICBib3JkZXItcmFkaXVzOiAycHg7XG4gIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gIHBhZGRpbmc6IDAgNXB4O1xufVxuXG4vL1xuLy8gQnV0dG9uIEdyb3VwXG4vL1xuLmRhcmtyb29tLWJ1dHRvbi1ncm91cCB7XG4gIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgbWFyZ2luOiAwO1xuICBwYWRkaW5nOiAwO1xuICAvLyBib3JkZXItcmlnaHQ6IDFweCBzb2xpZCAjNzc3O1xuXG4gICY6bGFzdC1jaGlsZCB7XG4gICAgYm9yZGVyLXJpZ2h0OiBub25lO1xuICB9XG4gIGlucHV0IHtcbiAgICBoZWlnaHQ6IDMzcHg7XG4gICAgd2lkdGg6IDU3cHg7XG4gICAgcGFkZGluZy10b3A6IDBweDtcbiAgfVxufVxuLy9cbi8vIEJ1dHRvblxuLy9cbi5kYXJrcm9vbS1idXR0b24ge1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuICBiYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDtcbiAgYm9yZGVyOiBub25lO1xuICBvdXRsaW5lOiBub25lO1xuICBwYWRkaW5nOiAycHggMCAwIDA7XG4gIHdpZHRoOiA0MHB4O1xuICBoZWlnaHQ6IDQwcHg7XG5cbiAgJjpob3ZlciB7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIGJhY2tncm91bmQ6ICM1NTU7XG4gIH1cbiAgJjphY3RpdmUge1xuICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICBiYWNrZ3JvdW5kOiAjMzMzO1xuICB9XG5cbiAgJjpkaXNhYmxlZCAuZGFya3Jvb20taWNvbiB7XG4gICAgZmlsbDogIzY2NjtcbiAgfVxuICAmOmRpc2FibGVkOmhvdmVyIHtcbiAgICBjdXJzb3I6IGRlZmF1bHQ7XG4gICAgLypjdXJzb3I6IG5vdC1hbGxvd2VkOyovXG4gICAgYmFja2dyb3VuZDogdHJhbnNwYXJlbnQ7XG4gIH1cbiAgJi5kYXJrcm9vbS1idXR0b24tYWN0aXZlIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjMzNiNWU1O1xuICB9XG4gICYuZGFya3Jvb20tYnV0dG9uLWhpZGRlbiB7XG4gICAgZGlzcGxheTogbm9uZTtcbiAgfVxuICAmLmRhcmtyb29tLWJ1dHRvbi1zdWNjZXNzIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjOTljYzAwO1xuICB9XG4gICYuZGFya3Jvb20tYnV0dG9uLXdhcm5pbmcgLmRhcmtyb29tLWljb24ge1xuICAgIGZpbGw6ICNGRkJCMzM7XG4gIH1cbiAgJi5kYXJrcm9vbS1idXR0b24tZGFuZ2VyIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjRkY0NDQ0O1xuICB9XG59XG5cbi8vXG4vLyBJY29uXG4vL1xuLmRhcmtyb29tLWljb24ge1xuICB3aWR0aDogMjRweDtcbiAgaGVpZ2h0OiAyNHB4O1xuICBmaWxsOiAjZmZmO1xufVxuIl0sIm1hcHBpbmdzIjoiQUNBQSxBQUFBLG1CQUFtQixDQUFDO0VBQ2xCLFFBQVEsRUFBRSxRQUFTLEdBQ3BCOztBQUVELEFBQUEseUJBQXlCLENBQUM7RUFDeEIsR0FBRyxFQUFFLENBQUU7RUFDUCxJQUFJLEVBQUUsQ0FBRSxHQUNUOztBQUVELEFBQWtCLGlCQUFELENBQUMsTUFBTSxDQUFBO0VBQ3RCLE1BQU0sRUFBRSxlQUFnQjtFQUN4QixLQUFLLEVBQUUsZUFBZ0I7RUFDdkIsU0FBUyxFQUFFLElBQUssR0FDakI7O0FBRUQsQUFBa0IsaUJBQUQsQ0FBQyxhQUFhLENBQUM7RUFDOUIsUUFBUSxFQUFFLG1CQUFvQixHQUMvQjs7QUFFRCxBQUFBLGlCQUFpQixDQUFDO0VBQ2hCLE1BQU0sRUFBRSxlQUFnQjtFQUN4QixLQUFLLEVBQUUsZUFBZ0I7RUFDdkIsU0FBUyxFQUFFLElBQUssR0FDakI7O0FDcEJELEFBQUEsaUJBQWlCLENBQUM7RUFDaEIsT0FBTyxFQUFFLElBQUs7RUFDZCxTQUFTLEVBQUUsSUFBSztFQUNoQixRQUFRLEVBQUUsT0FBUTtFQUNsQixHQUFHLEVBQUUsS0FBTTtFQUNYLElBQUksRUFBRSxDQUFFO0VBQ1IsVUFBVSxFQUFFLElBQUs7RUFDakIsTUFBTSxFQUFFLElBQUs7RUFDYixTQUFTLEVBQUUsSUFBSztFQUNoQixLQUFLLEVBQUUsV0FBWTtFQUNuQixLQUFLLEVBQUUsZ0JBQWlCO0VBQ3hCLE9BQU8sRUFBRSxFQUFHO0VBQ1osYUFBYSxFQUFFLEdBQUk7RUFDbkIsV0FBVyxFQUFFLE1BQU87RUFDcEIsT0FBTyxFQUFFLEtBQU0sR0FDaEI7O0FBS0QsQUFBQSxzQkFBc0IsQ0FBQztFQUNyQixPQUFPLEVBQUUsWUFBYTtFQUN0QixNQUFNLEVBQUUsQ0FBRTtFQUNWLE9BQU8sRUFBRSxDQUFFLEdBV1o7RUFkRCxBQUFBLHNCQUFzQixBQU1uQixXQUFXLENBQUM7SUFDWCxZQUFZLEVBQUUsSUFBSyxHQUNwQjtFQVJILEFBU0Usc0JBVG9CLENBU3BCLEtBQUssQ0FBQztJQUNKLE1BQU0sRUFBRSxJQUFLO0lBQ2IsS0FBSyxFQUFFLElBQUs7SUFDWixXQUFXLEVBQUUsR0FBSSxHQUNsQjs7QUFLSCxBQUFBLGdCQUFnQixDQUFDO0VBQ2YsVUFBVSxFQUFFLFVBQVc7RUFDdkIsVUFBVSxFQUFFLFdBQVk7RUFDeEIsTUFBTSxFQUFFLElBQUs7RUFDYixPQUFPLEVBQUUsSUFBSztFQUNkLE9BQU8sRUFBRSxTQUFVO0VBQ25CLEtBQUssRUFBRSxJQUFLO0VBQ1osTUFBTSxFQUFFLElBQUssR0FrQ2Q7RUF6Q0QsQUFBQSxnQkFBZ0IsQUFTYixNQUFNLENBQUM7SUFDTixNQUFNLEVBQUUsT0FBUTtJQUNoQixVQUFVLEVBQUUsSUFBSyxHQUNsQjtFQVpILEFBQUEsZ0JBQWdCLEFBYWIsT0FBTyxDQUFDO0lBQ1AsTUFBTSxFQUFFLE9BQVE7SUFDaEIsVUFBVSxFQUFFLElBQUssR0FDbEI7RUFoQkgsQUFrQmEsZ0JBbEJHLEFBa0JiLFNBQVMsQ0FBQyxjQUFjLENBQUM7SUFDeEIsSUFBSSxFQUFFLElBQUssR0FDWjtFQXBCSCxBQUFBLGdCQUFnQixBQXFCYixTQUFTLEFBQUEsTUFBTSxDQUFDO0lBQ2YsTUFBTSxFQUFFLE9BQVE7SUFDaEIsd0JBQXdCO0lBQ3hCLFVBQVUsRUFBRSxXQUFZLEdBQ3pCO0VBekJILEFBMEIyQixnQkExQlgsQUEwQmIsdUJBQXVCLENBQUMsY0FBYyxDQUFDO0lBQ3RDLElBQUksRUFBRSxPQUFRLEdBQ2Y7RUE1QkgsQUFBQSxnQkFBZ0IsQUE2QmIsdUJBQXVCLENBQUM7SUFDdkIsT0FBTyxFQUFFLElBQUssR0FDZjtFQS9CSCxBQWdDNEIsZ0JBaENaLEFBZ0NiLHdCQUF3QixDQUFDLGNBQWMsQ0FBQztJQUN2QyxJQUFJLEVBQUUsT0FBUSxHQUNmO0VBbENILEFBbUM0QixnQkFuQ1osQUFtQ2Isd0JBQXdCLENBQUMsY0FBYyxDQUFDO0lBQ3ZDLElBQUksRUFBRSxPQUFRLEdBQ2Y7RUFyQ0gsQUFzQzJCLGdCQXRDWCxBQXNDYix1QkFBdUIsQ0FBQyxjQUFjLENBQUM7SUFDdEMsSUFBSSxFQUFFLE9BQVEsR0FDZjs7QUFNSCxBQUFBLGNBQWMsQ0FBQztFQUNiLEtBQUssRUFBRSxJQUFLO0VBQ1osTUFBTSxFQUFFLElBQUs7RUFDYixJQUFJLEVBQUUsSUFBSyxHQUNaIiwibmFtZXMiOltdfQ== */ +/*# sourceMappingURL=data:application/json;charset=utf8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGFya3Jvb20uY3NzIiwic291cmNlcyI6WyJkYXJrcm9vbS5zY3NzIiwiX2xheW91dC5zY3NzIiwiX3Rvb2xiYXIuc2NzcyJdLCJzb3VyY2VzQ29udGVudCI6WyJAaW1wb3J0ICdsYXlvdXQnO1xuQGltcG9ydCAndG9vbGJhcic7XG4iLCIuZGFya3Jvb20tY29udGFpbmVyIHtcbiAgcG9zaXRpb246IHJlbGF0aXZlO1xufVxuXG4uZGFya3Jvb20taW1hZ2UtY29udGFpbmVyIHtcbiAgdG9wOiAwO1xuICBsZWZ0OiAwO1xufVxuXG4uY2FudmFzLWNvbnRhaW5lciBjYW52YXN7XG4gIGhlaWdodDogYXV0byAhaW1wb3J0YW50O1xuICB3aWR0aDogYXV0byAhaW1wb3J0YW50O1xuICBtYXgtd2lkdGg6IDEwMCU7XG59XG5cbi5jYW52YXMtY29udGFpbmVyIC51cHBlci1jYW52YXMge1xuICBwb3NpdGlvbjogcmVsYXRpdmUgIWltcG9ydGFudDtcbn1cblxuLmNhbnZhcy1jb250YWluZXIge1xuICBoZWlnaHQ6IGF1dG8gIWltcG9ydGFudDtcbiAgd2lkdGg6IGF1dG8gIWltcG9ydGFudDtcbiAgbWF4LXdpZHRoOiAxMDAlO1xufVxuXG4uZGFya3Jvb20taW1hZ2UtY29udGFpbmVyIGltZyB7XG4gIC8vIGRpc3BsYXk6IG5vbmU7XG59XG4iLCIvL1xuLy8gVG9vbGJhclxuLy9cbi5kYXJrcm9vbS10b29sYmFyIHtcbiAgZGlzcGxheTogZmxleDtcbiAgZmxleC13cmFwOiB3cmFwO1xuICBwb3NpdGlvbjogc3RpY2t5O1xuICBwb3NpdGlvbjogLXdlYmtpdC1zdGlja3k7XG4gIHRvcDogMTBweDtcbiAgbGVmdDogMDtcbiAgYmFja2dyb3VuZDogIzQ0NDtcbiAgaGVpZ2h0OiBhdXRvO1xuICBtaW4td2lkdGg6IDQwcHg7XG4gIHdpZHRoOiBmaXQtY29udGVudDtcbiAgd2lkdGg6IC1tb3otZml0LWNvbnRlbnQ7XG4gIHotaW5kZXg6IDk5O1xuICBib3JkZXItcmFkaXVzOiAycHg7XG4gIHdoaXRlLXNwYWNlOiBub3dyYXA7XG4gIHBhZGRpbmc6IDAgNXB4O1xufVxuXG4vL1xuLy8gQnV0dG9uIEdyb3VwXG4vL1xuLmRhcmtyb29tLWJ1dHRvbi1ncm91cCB7XG4gIGRpc3BsYXk6IGlubGluZS1ibG9jaztcbiAgbWFyZ2luOiAwO1xuICBwYWRkaW5nOiAwO1xuICAvLyBib3JkZXItcmlnaHQ6IDFweCBzb2xpZCAjNzc3O1xuXG4gICY6bGFzdC1jaGlsZCB7XG4gICAgYm9yZGVyLXJpZ2h0OiBub25lO1xuICB9XG4gIGlucHV0IHtcbiAgICBoZWlnaHQ6IDMzcHg7XG4gICAgd2lkdGg6IDU3cHg7XG4gICAgcGFkZGluZy10b3A6IDBweDtcbiAgfVxufVxuLy9cbi8vIEJ1dHRvblxuLy9cbi5kYXJrcm9vbS1idXR0b24ge1xuICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xuICBiYWNrZ3JvdW5kOiB0cmFuc3BhcmVudDtcbiAgYm9yZGVyOiBub25lO1xuICBvdXRsaW5lOiBub25lO1xuICBwYWRkaW5nOiAycHggMCAwIDA7XG4gIHdpZHRoOiA0MHB4O1xuICBoZWlnaHQ6IDQwcHg7XG5cbiAgJjpob3ZlciB7XG4gICAgY3Vyc29yOiBwb2ludGVyO1xuICAgIGJhY2tncm91bmQ6ICM1NTU7XG4gIH1cbiAgJjphY3RpdmUge1xuICAgIGN1cnNvcjogcG9pbnRlcjtcbiAgICBiYWNrZ3JvdW5kOiAjMzMzO1xuICB9XG5cbiAgJjpkaXNhYmxlZCAuZGFya3Jvb20taWNvbiB7XG4gICAgZmlsbDogIzY2NjtcbiAgfVxuICAmOmRpc2FibGVkOmhvdmVyIHtcbiAgICBjdXJzb3I6IGRlZmF1bHQ7XG4gICAgLypjdXJzb3I6IG5vdC1hbGxvd2VkOyovXG4gICAgYmFja2dyb3VuZDogdHJhbnNwYXJlbnQ7XG4gIH1cbiAgJi5kYXJrcm9vbS1idXR0b24tYWN0aXZlIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjMzNiNWU1O1xuICB9XG4gICYuZGFya3Jvb20tYnV0dG9uLWhpZGRlbiB7XG4gICAgZGlzcGxheTogbm9uZTtcbiAgfVxuICAmLmRhcmtyb29tLWJ1dHRvbi1zdWNjZXNzIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjOTljYzAwO1xuICB9XG4gICYuZGFya3Jvb20tYnV0dG9uLXdhcm5pbmcgLmRhcmtyb29tLWljb24ge1xuICAgIGZpbGw6ICNGRkJCMzM7XG4gIH1cbiAgJi5kYXJrcm9vbS1idXR0b24tZGFuZ2VyIC5kYXJrcm9vbS1pY29uIHtcbiAgICBmaWxsOiAjRkY0NDQ0O1xuICB9XG59XG5cbi8vXG4vLyBJY29uXG4vL1xuLmRhcmtyb29tLWljb24ge1xuICB3aWR0aDogMjRweDtcbiAgaGVpZ2h0OiAyNHB4O1xuICBmaWxsOiAjZmZmO1xufVxuIl0sIm1hcHBpbmdzIjoiQUNBQSxBQUFBLG1CQUFtQixDQUFDO0VBQ2xCLFFBQVEsRUFBRSxRQUFTLEdBQ3BCOztBQUVELEFBQUEseUJBQXlCLENBQUM7RUFDeEIsR0FBRyxFQUFFLENBQUU7RUFDUCxJQUFJLEVBQUUsQ0FBRSxHQUNUOztBQUVELEFBQWtCLGlCQUFELENBQUMsTUFBTSxDQUFBO0VBQ3RCLE1BQU0sRUFBRSxlQUFnQjtFQUN4QixLQUFLLEVBQUUsZUFBZ0I7RUFDdkIsU0FBUyxFQUFFLElBQUssR0FDakI7O0FBRUQsQUFBa0IsaUJBQUQsQ0FBQyxhQUFhLENBQUM7RUFDOUIsUUFBUSxFQUFFLG1CQUFvQixHQUMvQjs7QUFFRCxBQUFBLGlCQUFpQixDQUFDO0VBQ2hCLE1BQU0sRUFBRSxlQUFnQjtFQUN4QixLQUFLLEVBQUUsZUFBZ0I7RUFDdkIsU0FBUyxFQUFFLElBQUssR0FDakI7O0FDcEJELEFBQUEsaUJBQWlCLENBQUM7RUFDaEIsT0FBTyxFQUFFLElBQUs7RUFDZCxTQUFTLEVBQUUsSUFBSztFQUNoQixRQUFRLEVBQUUsTUFBTztFQUNqQixRQUFRLEVBQUUsY0FBZTtFQUN6QixHQUFHLEVBQUUsSUFBSztFQUNWLElBQUksRUFBRSxDQUFFO0VBQ1IsVUFBVSxFQUFFLElBQUs7RUFDakIsTUFBTSxFQUFFLElBQUs7RUFDYixTQUFTLEVBQUUsSUFBSztFQUNoQixLQUFLLEVBQUUsV0FBWTtFQUNuQixLQUFLLEVBQUUsZ0JBQWlCO0VBQ3hCLE9BQU8sRUFBRSxFQUFHO0VBQ1osYUFBYSxFQUFFLEdBQUk7RUFDbkIsV0FBVyxFQUFFLE1BQU87RUFDcEIsT0FBTyxFQUFFLEtBQU0sR0FDaEI7O0FBS0QsQUFBQSxzQkFBc0IsQ0FBQztFQUNyQixPQUFPLEVBQUUsWUFBYTtFQUN0QixNQUFNLEVBQUUsQ0FBRTtFQUNWLE9BQU8sRUFBRSxDQUFFLEdBV1o7RUFkRCxBQUFBLHNCQUFzQixBQU1uQixXQUFXLENBQUM7SUFDWCxZQUFZLEVBQUUsSUFBSyxHQUNwQjtFQVJILEFBU0Usc0JBVG9CLENBU3BCLEtBQUssQ0FBQztJQUNKLE1BQU0sRUFBRSxJQUFLO0lBQ2IsS0FBSyxFQUFFLElBQUs7SUFDWixXQUFXLEVBQUUsR0FBSSxHQUNsQjs7QUFLSCxBQUFBLGdCQUFnQixDQUFDO0VBQ2YsVUFBVSxFQUFFLFVBQVc7RUFDdkIsVUFBVSxFQUFFLFdBQVk7RUFDeEIsTUFBTSxFQUFFLElBQUs7RUFDYixPQUFPLEVBQUUsSUFBSztFQUNkLE9BQU8sRUFBRSxTQUFVO0VBQ25CLEtBQUssRUFBRSxJQUFLO0VBQ1osTUFBTSxFQUFFLElBQUssR0FrQ2Q7RUF6Q0QsQUFBQSxnQkFBZ0IsQUFTYixNQUFNLENBQUM7SUFDTixNQUFNLEVBQUUsT0FBUTtJQUNoQixVQUFVLEVBQUUsSUFBSyxHQUNsQjtFQVpILEFBQUEsZ0JBQWdCLEFBYWIsT0FBTyxDQUFDO0lBQ1AsTUFBTSxFQUFFLE9BQVE7SUFDaEIsVUFBVSxFQUFFLElBQUssR0FDbEI7RUFoQkgsQUFrQmEsZ0JBbEJHLEFBa0JiLFNBQVMsQ0FBQyxjQUFjLENBQUM7SUFDeEIsSUFBSSxFQUFFLElBQUssR0FDWjtFQXBCSCxBQUFBLGdCQUFnQixBQXFCYixTQUFTLEFBQUEsTUFBTSxDQUFDO0lBQ2YsTUFBTSxFQUFFLE9BQVE7SUFDaEIsd0JBQXdCO0lBQ3hCLFVBQVUsRUFBRSxXQUFZLEdBQ3pCO0VBekJILEFBMEIyQixnQkExQlgsQUEwQmIsdUJBQXVCLENBQUMsY0FBYyxDQUFDO0lBQ3RDLElBQUksRUFBRSxPQUFRLEdBQ2Y7RUE1QkgsQUFBQSxnQkFBZ0IsQUE2QmIsdUJBQXVCLENBQUM7SUFDdkIsT0FBTyxFQUFFLElBQUssR0FDZjtFQS9CSCxBQWdDNEIsZ0JBaENaLEFBZ0NiLHdCQUF3QixDQUFDLGNBQWMsQ0FBQztJQUN2QyxJQUFJLEVBQUUsT0FBUSxHQUNmO0VBbENILEFBbUM0QixnQkFuQ1osQUFtQ2Isd0JBQXdCLENBQUMsY0FBYyxDQUFDO0lBQ3ZDLElBQUksRUFBRSxPQUFRLEdBQ2Y7RUFyQ0gsQUFzQzJCLGdCQXRDWCxBQXNDYix1QkFBdUIsQ0FBQyxjQUFjLENBQUM7SUFDdEMsSUFBSSxFQUFFLE9BQVEsR0FDZjs7QUFNSCxBQUFBLGNBQWMsQ0FBQztFQUNiLEtBQUssRUFBRSxJQUFLO0VBQ1osTUFBTSxFQUFFLElBQUs7RUFDYixJQUFJLEVBQUUsSUFBSyxHQUNaIiwibmFtZXMiOltdfQ== */ diff --git a/demo/build/darkroom.js b/demo/build/darkroom.js index f58c466..f8dacc1 100644 --- a/demo/build/darkroom.js +++ b/demo/build/darkroom.js @@ -307,6 +307,9 @@ imageElement.style.display = 'none'; mainContainerElement.appendChild(imageElement); + Stickyfill.add(toolbarElement); + Stickyfill.refreshAll(); + // Instanciate object from elements this.containerElement = mainContainerElement; this.originalImageElement = imageElement; @@ -842,7 +845,7 @@ snapshot.src = canvas.toDataURL({ left: left, top: top, - width: top, + width: width, height: height }); } @@ -2497,4 +2500,4 @@ }); }()); -//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["bootstrap.js","darkroom.js","plugin.js","transformation.js","ui.js","utils.js","darkroom.history.js","darkroom.rotate.js","darkroom.crop.js","darkroom.pencil.js","darkroom.rectangle.js","darkroom.text.js","darkroom.circle.js","zoom.js","darkroom.save.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCrXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC1KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC9rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCrSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCpMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC7IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCjNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"darkroom.js","sourcesContent":["(function() {\n    'use strict';\n\n    // Inject SVG icons into the DOM\n    var element = document.createElement('div');\n    element.id = 'darkroom-icons';\n    element.style.height = 0;\n    element.style.width = 0;\n    element.style.position = 'absolute';\n    element.style.visibility = 'hidden';\n    element.innerHTML = '<!-- inject:svg --><!-- endinject -->';\n    document.body.appendChild(element);\n\n    })();\n","(function () {\n  'use strict';\n\n  window.Darkroom = Darkroom;\n\n  // Core object of DarkroomJS.\n  // Basically it's a single object, instanciable via an element\n  // (it could be a CSS selector or a DOM element), some custom options,\n  // and a list of plugin objects (or none to use default ones).\n  function Darkroom(element, options, plugins) {\n    return this.constructor(element, options, plugins);\n  }\n\n  // Create an empty list of plugin objects, which will be filled by\n  // other plugin scripts. This is the default plugin list if none is\n  // specified in Darkroom'ss constructor.\n  Darkroom.plugins = [];\n\n  Darkroom.prototype = {\n    // Reference to the main container element\n    containerElement: null,\n\n    // Reference to the Fabric canvas object\n    canvas: null,\n\n    // Reference to the Fabric image object\n    image: null,\n\n    // Reference to the Fabric source canvas object\n    sourceCanvas: null,\n\n    // Reference to the Fabric source image object\n    sourceImage: null,\n\n    // Track of the original image element\n    originalImageElement: null,\n\n    // Stack of transformations to apply to the image source\n    transformations: [],\n\n    // Default options\n    defaults: {\n      // Canvas properties (dimension, ratio, color)\n      minWidth: null,\n      minHeight: null,\n      maxWidth: null,\n      maxHeight: null,\n      ratio: null,\n      left: 0,\n      top: 0,\n      backgroundColor: '#fff',\n\n      // Plugins options\n      plugins: {},\n\n      // Post-initialisation callback\n      initialize: function () { /* noop */ }\n    },\n\n    // List of the instancied plugins\n    plugins: {},\n\n    // This options are a merge between `defaults` and the options passed\n    // through the constructor\n    options: {},\n\n    constructor: function (element, options, plugins) {\n      this.options = Darkroom.Utils.extend(options, this.defaults);\n\n      if (typeof element === 'string')\n        element = document.querySelector(element);\n      if (null === element)\n        return;\n\n      var image = new Image();\n      image.onload = function () {\n        // Initialize the DOM/Fabric elements\n        this._initializeDOM(element);\n        this._initializeImage();\n\n        // Then initialize the plugins\n        this._initializePlugins(Darkroom.plugins);\n\n        // Public method to adjust image according to the canvas\n        this.refresh(function () {\n          // Execute a custom callback after initialization\n          this.options.initialize.bind(this).call();\n        }.bind(this));\n\n      }.bind(this)\n\n      //image.crossOrigin = 'anonymous';\n      image.src = element.src;\n    },\n\n    selfDestroy: function () {\n      var container = this.containerElement;\n      var image = new Image();\n      image.onload = function () {\n        container.parentNode.replaceChild(image, container);\n      }\n\n      image.src = this.sourceImage.toDataURL();\n    },\n\n    // Add ability to attach event listener on the core object.\n    // It uses the canvas element to process events.\n    addEventListener: function (eventName, callback) {\n      var el = this.canvas.getElement();\n      if (el.addEventListener) {\n        el.addEventListener(eventName, callback);\n      } else if (el.attachEvent) {\n        el.attachEvent('on' + eventName, callback);\n      }\n    },\n\n    dispatchEvent: function (eventName) {\n      // Use the old way of creating event to be IE compatible\n      // See https://developer.mozilla.org/en-US/docs/Web/Guide/Events/Creating_and_triggering_events\n      var event = document.createEvent('Event');\n      event.initEvent(eventName, true, true);\n\n      this.canvas.getElement().dispatchEvent(event);\n    },\n\n    // Adjust image & canvas dimension according to min/max width/height\n    // and ratio specified in the options.\n    // This method should be called after each image transformation.\n    refresh: function (next) {\n      var clone = new Image();\n      clone.onload = function () {\n        this._replaceCurrentImage(new fabric.Image(clone));\n\n        if (next) next();\n      }.bind(this);\n      clone.src = this.sourceImage.toDataURL();\n    },\n\n    _replaceCurrentImage: function (newImage) {\n      if (this.image) {\n        this.image.remove();\n      }\n\n      this.image = newImage;\n      this.image.selectable = false;\n      this.image.left = this.options.left;\n      this.image.top = this.options.top;\n\n      // Adjust width or height according to specified ratio\n      var viewport = Darkroom.Utils.computeImageViewPort(this.image);\n      var canvasWidth = viewport.width;\n      var canvasHeight = viewport.height;\n\n      // if (null !== this.options.ratio) {\n      //   var canvasRatio = +this.options.ratio;\n      //   var currentRatio = canvasWidth / canvasHeight;\n\n      //   if (currentRatio > canvasRatio) {\n      //     canvasHeight = canvasWidth / canvasRatio;\n      //   } else if (currentRatio < canvasRatio) {\n      //     canvasWidth = canvasHeight * canvasRatio;\n      //   }\n      // }\n\n      // // Then scale the image to fit into dimension limits\n      // var scaleMin = 1;\n      // var scaleMax = 1;\n      // var scaleX = 1;\n      // var scaleY = 1;\n\n      // if (null !== this.options.maxWidth && this.options.maxWidth < canvasWidth) {\n      //   scaleX =  this.options.maxWidth / canvasWidth;\n      // }\n      // if (null !== this.options.maxHeight && this.options.maxHeight < canvasHeight) {\n      //   scaleY =  this.options.maxHeight / canvasHeight;\n      // }\n      // scaleMin = Math.min(scaleX, scaleY);\n\n      // scaleX = 1;\n      // scaleY = 1;\n      // if (null !== this.options.minWidth && this.options.minWidth > canvasWidth) {\n      //   scaleX =  this.options.minWidth / canvasWidth;\n      // }\n      // if (null !== this.options.minHeight && this.options.minHeight > canvasHeight) {\n      //   scaleY =  this.options.minHeight / canvasHeight;\n      // }\n      // scaleMax = Math.max(scaleX, scaleY);\n\n      // var scale = scaleMax * scaleMin; // one should be equals to 1\n\n      // canvasWidth *= scale;\n      // canvasHeight *= scale;\n\n      // Finally place the image in the center of the canvas\n      // this.image.setScaleX(1 * scale);\n      // this.image.setScaleY(1 * scale);\n      this.canvas.add(this.image);\n      this.canvas.setWidth(canvasWidth);\n      this.canvas.setHeight(canvasHeight);\n\n      // Resize the image to fit the canvas\n      if (this.image.oCoords.br.x < this.canvas.width) {\n        this.image.left = this.image.left + Math.ceil(this.canvas.width - this.image.oCoords.br.x);\n      }\n      if (this.image.oCoords.br.y < this.canvas.height) {\n        this.image.top = this.image.top + Math.ceil(this.canvas.height - this.image.oCoords.br.y);\n      }\n      // this.canvas.centerObject(this.image);\n      this.image.setCoords();\n    },\n\n    // Apply the transformation on the current image and save it in the\n    // transformations stack (in order to reconstitute the previous states\n    // of the image).\n    applyTransformation: function (transformation) {\n      this.transformations.push(transformation);\n\n      transformation.applyTransformation(\n        this.sourceCanvas,\n        this.sourceImage,\n        this._postTransformation.bind(this)\n      );\n    },\n\n    _postTransformation: function (newImage) {\n      if (newImage)\n        this.sourceImage = newImage;\n\n      this.refresh(function () {\n        this.dispatchEvent('core:transformation');\n      }.bind(this));\n    },\n\n    // Initialize image from original element plus re-apply every\n    // transformations.\n    reinitializeImage: function () {\n      this.sourceImage.remove();\n      this._initializeImage();\n      this._popTransformation(this.transformations.slice())\n    },\n\n    _popTransformation: function (transformations) {\n      if (0 === transformations.length) {\n        this.dispatchEvent('core:reinitialized');\n        this.refresh();\n        return;\n      }\n\n      var transformation = transformations.shift();\n\n      var next = function (newImage) {\n        if (newImage) this.sourceImage = newImage;\n        this._popTransformation(transformations)\n      };\n\n      transformation.applyTransformation(\n        this.sourceCanvas,\n        this.sourceImage,\n        next.bind(this)\n      );\n    },\n\n    // Create the DOM elements and instanciate the Fabric canvas.\n    // The image element is replaced by a new `div` element.\n    // However the original image is re-injected in order to keep a trace of it.\n    _initializeDOM: function (imageElement) {\n      // Container\n      var mainContainerElement = document.createElement('div');\n      mainContainerElement.className = 'darkroom-container';\n\n      // Toolbar\n      var toolbarElement = document.createElement('div');\n      toolbarElement.className = 'darkroom-toolbar';\n      mainContainerElement.appendChild(toolbarElement);\n\n      // Viewport canvas\n      var canvasContainerElement = document.createElement('div');\n      canvasContainerElement.className = 'darkroom-image-container';\n      var canvasElement = document.createElement('canvas');\n      canvasContainerElement.appendChild(canvasElement);\n      mainContainerElement.appendChild(canvasContainerElement);\n\n      // Source canvas\n      var sourceCanvasContainerElement = document.createElement('div');\n      sourceCanvasContainerElement.className = 'darkroom-source-container';\n      sourceCanvasContainerElement.style.display = 'none';\n      var sourceCanvasElement = document.createElement('canvas');\n      sourceCanvasContainerElement.appendChild(sourceCanvasElement);\n      mainContainerElement.appendChild(sourceCanvasContainerElement);\n\n      // Original image\n      imageElement.parentNode.replaceChild(mainContainerElement, imageElement);\n      imageElement.style.display = 'none';\n      mainContainerElement.appendChild(imageElement);\n\n      // Instanciate object from elements\n      this.containerElement = mainContainerElement;\n      this.originalImageElement = imageElement;\n\n      this.toolbar = new Darkroom.UI.Toolbar(toolbarElement);\n\n      this.canvas = new fabric.Canvas(canvasElement, {\n        selection: false,\n        backgroundColor: this.options.backgroundColor\n      });\n\n      this.sourceCanvas = new fabric.Canvas(sourceCanvasElement, {\n        selection: false,\n        backgroundColor: this.options.backgroundColor\n      });\n    },\n\n    // Instanciate the Fabric image object.\n    // The image is created as a static element with no control,\n    // then it is add in the Fabric canvas object.\n    _initializeImage: function () {\n      this.sourceImage = new fabric.Image(this.originalImageElement, {\n        // Some options to make the image static\n        selectable: false,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockRotation: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false,\n      });\n\n      this.sourceCanvas.add(this.sourceImage);\n\n      // Adjust width or height according to specified ratio\n      var viewport = Darkroom.Utils.computeImageViewPort(this.sourceImage);\n      var canvasWidth = viewport.width;\n      var canvasHeight = viewport.height;\n\n      this.sourceCanvas.setWidth(canvasWidth);\n      this.sourceCanvas.setHeight(canvasHeight);\n      this.sourceCanvas.centerObject(this.sourceImage);\n      this.sourceImage.setCoords();\n    },\n\n    // Initialize every plugins.\n    // Note that plugins are instanciated in the same order than they\n    // are declared in the parameter object.\n    _initializePlugins: function (plugins) {\n      for (var name in plugins) {\n        var plugin = plugins[name];\n        var options = this.options.plugins[name];\n\n        // Setting false into the plugin options will disable the plugin\n        if (options === false)\n          continue;\n\n        // Avoid any issues with _proto_\n        if (!plugins.hasOwnProperty(name))\n          continue;\n\n        this.plugins[name] = new plugin(this, options);\n      }\n    },\n\n    clearFocus: function (instance) {\n      for (var key in this.plugins) {\n        if (key != instance) {\n          this.plugins[key].clear();\n        }\n      }\n    }\n  }\n\n})();\n","(function() {\n  'use strict';\n\n  Darkroom.Plugin = Plugin;\n\n  // Define a plugin object. This is the (abstract) parent class which\n  // has to be extended for each plugin.\n  function Plugin(darkroom, options) {\n    this.darkroom = darkroom;\n    this.options = Darkroom.Utils.extend(options, this.defaults);\n    this.initialize();\n  }\n\n  Plugin.prototype = {\n    defaults: {},\n    initialize: function() { },\n    clear: function() { }\n  }\n\n  // Inspired by Backbone.js extend capability.\n  Plugin.extend = function(protoProps) {\n    var parent = this;\n    var child;\n\n    if (protoProps && protoProps.hasOwnProperty('constructor')) {\n      child = protoProps.constructor;\n    } else {\n      child = function(){ return parent.apply(this, arguments); };\n    }\n\n    Darkroom.Utils.extend(child, parent);\n\n    var Surrogate = function(){ this.constructor = child; };\n    Surrogate.prototype = parent.prototype;\n    child.prototype = new Surrogate;\n\n    if (protoProps) Darkroom.Utils.extend(child.prototype, protoProps);\n\n    child.__super__ = parent.prototype;\n\n    return child;\n  }\n\n  })();\n","(function() {\n  'use strict';\n\n  Darkroom.Transformation = Transformation;\n\n  function Transformation(options) {\n    this.options = options;\n  }\n\n  Transformation.prototype = {\n    applyTransformation: function(image) { /* no-op */  }\n  }\n\n  // Inspired by Backbone.js extend capability.\n  Transformation.extend = function(protoProps) {\n    var parent = this;\n    var child;\n\n    if (protoProps && protoProps.hasOwnProperty('constructor')) {\n      child = protoProps.constructor;\n    } else {\n      child = function(){ return parent.apply(this, arguments); };\n    }\n\n    Darkroom.Utils.extend(child, parent);\n\n    var Surrogate = function(){ this.constructor = child; };\n    Surrogate.prototype = parent.prototype;\n    child.prototype = new Surrogate;\n\n    if (protoProps) Darkroom.Utils.extend(child.prototype, protoProps);\n\n    child.__super__ = parent.prototype;\n\n    return child;\n  }\n\n  })();\n","(function() {\n  'use strict';\n\n  Darkroom.UI = {\n    Toolbar: Toolbar,\n    ButtonGroup: ButtonGroup,\n    Button: Button,\n  };\n\n  // Toolbar object.\n  function Toolbar(element) {\n    this.element = element;\n  }\n\n  Toolbar.prototype = {\n    createButtonGroup: function(options) {\n      var buttonGroup = document.createElement('div');\n      buttonGroup.className = 'darkroom-button-group';\n      this.element.appendChild(buttonGroup);\n\n      return new ButtonGroup(buttonGroup);\n    }\n  };\n\n  // ButtonGroup object.\n  function ButtonGroup(element) {\n    this.element = element;\n  }\n\n  ButtonGroup.prototype = {\n    createButton: function(options) {\n      var defaults = {\n        image: 'help',\n        type: 'default',\n        group: 'default',\n        hide: false,\n        disabled: false\n      };\n\n      options = Darkroom.Utils.extend(options, defaults);\n\n      var buttonElement = document.createElement('button');\n      buttonElement.type = 'button';\n      buttonElement.className = 'darkroom-button darkroom-button-' + options.type;\n      buttonElement.innerHTML = '<svg class=\"darkroom-icon\"><use xlink:href=\"#' + options.image + '\" /></svg>';\n      this.element.appendChild(buttonElement);\n\n      var button = new Button(buttonElement);\n      button.hide(options.hide);\n      button.disable(options.disabled);\n\n      return button;\n    },\n\n    createInput: function(options) {\n      const defaults = {\n        image: \"help\",\n        type: \"input\",\n        group: \"default\",\n        hide: false,\n        disabled: false,\n        value: \"\"\n      };\n\n      options = Darkroom.Utils.extend(options, defaults);\n\n      const datalistElement = document.createElement(\"datalist\");\n      datalistElement.id = \"rainbow\";\n      const datalistElementOptionred = document.createElement(\"options\");\n      datalistElementOptionred.setAttribute(\"value\", \"#FF0000\");\n      datalistElementOptionred.innerHTML = \"red\";\n      datalistElement.appendChild(datalistElementOptionred);\n      const datalistElementOptionOrange = document.createElement(\"options\");\n      datalistElementOptionOrange.setAttribute(\"value\", \"#FFA500\");\n      datalistElementOptionOrange.innerHTML = \"Orange\";\n      datalistElement.appendChild(datalistElementOptionOrange);\n      const datalistElementOptionYellow = document.createElement(\"options\");\n      datalistElementOptionYellow.setAttribute(\"value\", \"#FFFF00\");\n      datalistElementOptionYellow.innerHTML = \"Yellow\";\n      datalistElement.appendChild(datalistElementOptionYellow);\n      const datalistElementOptionGreen = document.createElement(\"options\");\n      datalistElementOptionGreen.setAttribute(\"value\", \"#008000\");\n      datalistElementOptionGreen.innerHTML = \"Green\";\n      datalistElement.appendChild(datalistElementOptionGreen);\n      const datalistElementOptionBlue = document.createElement(\"options\");\n      datalistElementOptionBlue.setAttribute(\"value\", \"#0000FF\");\n      datalistElementOptionBlue.innerHTML = \"Blue\";\n      datalistElement.appendChild(datalistElementOptionBlue);\n      // this.element.appendChild(datalistElement);\n\n      const inputElement = document.createElement(\"input\");\n      inputElement.setAttribute(\"type\", \"color\");\n      inputElement.setAttribute(\"list\", \"rainbow\");\n      inputElement.value = \"#ff0000\";\n      inputElement.className = \"darkroom-button darkroom-input-\" + options.type;\n      inputElement.innerHTML = '<svg class=\"darkroom-icon\"><use xlink:href=\"#' + options.image + '\" /></svg>';\n      this.element.appendChild(inputElement);\n\n      const input = new Input(inputElement);\n      input.hide(options.hide);\n      input.disable(options.disabled);\n\n      return input;\n    }\n  };\n\n  // Button object.\n  function Button(element) {\n    this.element = element;\n  }\n\n  // Input object.\n  function Input(element) {\n    this.element = element;\n  }\n\n  Button.prototype = {\n    addEventListener: function(eventName, listener) {\n      if (this.element.addEventListener){\n        this.element.addEventListener(eventName, listener);\n      } else if (this.element.attachEvent) {\n        this.element.attachEvent('on' + eventName, listener);\n      }\n    },\n    removeEventListener: function(eventName, listener) {\n      if (this.element.removeEventListener){\n        this.element.removeEventListener(eventName, listener);\n      }\n    },\n    active: function(value) {\n      if (value)\n        this.element.classList.add('darkroom-button-active');\n      else\n        this.element.classList.remove('darkroom-button-active');\n    },\n    hide: function(value) {\n      if (value)\n        this.element.classList.add('darkroom-button-hidden');\n      else\n        this.element.classList.remove('darkroom-button-hidden');\n    },\n    disable: function(value) {\n      this.element.disabled = (value) ? true : false;\n    }\n  };\n\n  Input.prototype = {\n    addEventListener: function(eventName, listener) {\n      if (this.element.addEventListener) {\n        this.element.addEventListener(eventName, listener);\n      } else if (this.element.attachEvent) {\n        this.element.attachEvent(\"on\" + eventName, listener);\n      }\n    },\n    removeEventListener: function(eventName, listener) {\n      if (this.element.removeEventListener) {\n        this.element.removeEventListener(eventName, listener);\n      }\n    },\n    active: function(value) {\n      if (value) { this.element.classList.add(\"darkroom-input-active\"); } else { this.element.classList.remove(\"darkroom-input-active\"); }\n    },\n    hide: function(value) {\n      if (value) { this.element.classList.add(\"darkroom-button-hidden\"); } else { this.element.classList.remove(\"darkroom-button-hidden\"); }\n    },\n    disable: function(value) {\n      this.element.disabled = !!value;\n    }\n  };\n}());\n","(function() {\n  'use strict';\n\n  Darkroom.Utils = {\n    extend: extend,\n    computeImageViewPort: computeImageViewPort,\n  };\n\n\n  // Utility method to easily extend objects.\n  function extend(b, a) {\n    var prop;\n    if (b === undefined) {\n      return a;\n    }\n    for (prop in a) {\n      if (a.hasOwnProperty(prop) && b.hasOwnProperty(prop) === false) {\n        b[prop] = a[prop];\n      }\n    }\n    return b;\n  }\n\n  function computeImageViewPort(image) {\n    return {\n      height: Math.abs(image.getWidth() * (Math.sin(image.getAngle() * Math.PI/180))) + Math.abs(image.getHeight() * (Math.cos(image.getAngle() * Math.PI/180))),\n      width: Math.abs(image.getHeight() * (Math.sin(image.getAngle() * Math.PI/180))) + Math.abs(image.getWidth() * (Math.cos(image.getAngle() * Math.PI/180))),\n    }\n  }\n\n  })();\n",";(function(window, document, Darkroom, fabric) {\n  'use strict';\n\n  Darkroom.plugins['history'] = Darkroom.Plugin.extend({\n    undoTransformations: [],\n\n    initialize: function InitDarkroomHistoryPlugin() {\n      this._initButtons();\n\n      this.darkroom.addEventListener('core:transformation', this._onTranformationApplied.bind(this));\n    },\n\n    undo: function() {\n      if (this.darkroom.transformations.length === 0) {\n        return;\n      }\n\n      var lastTransformation = this.darkroom.transformations.pop();\n      this.undoTransformations.unshift(lastTransformation);\n\n      this.darkroom.reinitializeImage();\n      this._updateButtons();\n    },\n\n    redo: function() {\n      if (this.undoTransformations.length === 0) {\n        return;\n      }\n\n      var cancelTransformation = this.undoTransformations.shift();\n      this.darkroom.transformations.push(cancelTransformation);\n\n      this.darkroom.reinitializeImage();\n      this._updateButtons();\n    },\n\n    _initButtons: function() {\n      var buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.backButton = buttonGroup.createButton({\n        image: 'undo',\n        disabled: true\n      });\n\n      this.forwardButton = buttonGroup.createButton({\n        image: 'redo',\n        disabled: true\n      });\n\n      this.backButton.addEventListener('click', this.undo.bind(this));\n      this.forwardButton.addEventListener('click', this.redo.bind(this));\n\n      return this;\n    },\n\n    _updateButtons: function() {\n      this.backButton.disable((this.darkroom.transformations.length === 0))\n      this.forwardButton.disable((this.undoTransformations.length === 0))\n    },\n\n    _onTranformationApplied: function() {\n      this.undoTransformations = [];\n      this._updateButtons();\n    }\n  });\n})(window, document, Darkroom, fabric);\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Rotation = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const angle = (image.getAngle() + this.options.angle) % 360;\n      image.rotate(angle);\n\n      const height = Math.abs(image.getWidth() * (Math.sin(angle * Math.PI / 180))) + Math.abs(image.getHeight() * (Math.cos(angle * Math.PI / 180)));\n      const width = Math.abs(image.getHeight() * (Math.sin(angle * Math.PI / 180))) + Math.abs(image.getWidth() * (Math.cos(angle * Math.PI / 180)));\n\n      canvas.setWidth(width);\n      canvas.setHeight(height);\n\n      canvas.centerObject(image);\n      image.setCoords();\n      canvas.renderAll();\n\n      next();\n    }\n  });\n\n  Darkroom.plugins.rotate = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      const leftButton = buttonGroup.createButton({\n        image: \"rotate-left\"\n      });\n\n      const rightButton = buttonGroup.createButton({\n        image: \"rotate-right\"\n      });\n\n      leftButton.addEventListener(\"click\", this.rotateLeft.bind(this));\n      rightButton.addEventListener(\"click\", this.rotateRight.bind(this));\n    },\n\n    rotateLeft: function() {\n      this.rotate(-90);\n    },\n\n    rotateRight: function() {\n      this.rotate(90);\n    },\n\n    rotate: function(angle) {\n      this.darkroom.options.left = 0;\n      this.darkroom.options.top = 0;\n      this.darkroom.applyTransformation(new Rotation({\n        angle: angle\n      }));\n    }\n\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Crop = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      // Snapshot the image delimited by the crop zone\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) {\n          return;\n        }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        const width = this.width;\n        const height = this.height;\n\n        // Update canvas size\n        canvas.setWidth(width);\n        canvas.setHeight(height);\n\n        // Add image\n        image.remove();\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      const viewport = Darkroom.Utils.computeImageViewPort(image);\n      const imageWidth = viewport.width;\n      const imageHeight = viewport.height;\n\n      const left = this.options.left * imageWidth;\n      const top = this.options.top * imageHeight;\n      const width = Math.min(this.options.width * imageWidth, imageWidth - left);\n      const height = Math.min(this.options.height * imageHeight, imageHeight - top);\n\n      snapshot.src = canvas.toDataURL({\n        left: left,\n        top: top,\n        width: top,\n        height: height\n      });\n    }\n  });\n\n  const CropZone = fabric.util.createClass(fabric.Rect, {\n    _render: function(ctx) {\n      this.callSuper(\"_render\", ctx);\n\n      const dashWidth = 7;\n\n      // Set original scale\n      const flipX = this.flipX ? -1 : 1;\n      const flipY = this.flipY ? -1 : 1;\n      const scaleX = flipX / this.scaleX;\n      const scaleY = flipY / this.scaleY;\n\n      ctx.scale(scaleX, scaleY);\n\n      // Overlay rendering\n      ctx.fillStyle = \"rgba(0, 0, 0, 0.5)\";\n      this._renderOverlay(ctx);\n\n      // Set dashed borders\n      if (ctx.setLineDash !== undefined) {\n        ctx.setLineDash([dashWidth, dashWidth]);\n      } else if (ctx.mozDash !== undefined) {\n        ctx.mozDash = [dashWidth, dashWidth];\n      }\n\n      // First lines rendering with black\n      ctx.strokeStyle = \"rgba(0, 0, 0, 0.2)\";\n      this._renderBorders(ctx);\n      this._renderGrid(ctx);\n\n      // Re render lines in white\n      ctx.lineDashOffset = dashWidth;\n      ctx.strokeStyle = \"rgba(255, 255, 255, 0.4)\";\n      this._renderBorders(ctx);\n      this._renderGrid(ctx);\n\n      // Reset scale\n      ctx.scale(1 / scaleX, 1 / scaleY);\n    },\n\n    _renderOverlay: function(ctx) {\n      const canvas = ctx.canvas;\n\n      //\n      //    x0    x1        x2      x3\n      // y0 +------------------------+\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      // y1 +------+---------+-------+\n      //    |\\\\\\\\\\\\|         |\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\|    0    |\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\|         |\\\\\\\\\\\\\\|\n      // y2 +------+---------+-------+\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      // y3 +------------------------+\n      //\n\n      const x0 = Math.ceil(-this.getWidth() / 2 - this.getLeft());\n      const x1 = Math.ceil(-this.getWidth() / 2);\n      const x2 = Math.ceil(this.getWidth() / 2);\n      const x3 = Math.ceil(this.getWidth() / 2 + (canvas.width - this.getWidth() - this.getLeft()));\n\n      const y0 = Math.ceil(-this.getHeight() / 2 - this.getTop());\n      const y1 = Math.ceil(-this.getHeight() / 2);\n      const y2 = Math.ceil(this.getHeight() / 2);\n      const y3 = Math.ceil(this.getHeight() / 2 + (canvas.height - this.getHeight() - this.getTop()));\n\n      ctx.beginPath();\n\n      // Draw outer rectangle.\n      // Numbers are +/-1 so that overlay edges don't get blurry.\n      ctx.moveTo(x0 - 1, y0 - 1);\n      ctx.lineTo(x3 + 1, y0 - 1);\n      ctx.lineTo(x3 + 1, y3 + 1);\n      ctx.lineTo(x0 - 1, y3 - 1);\n      ctx.lineTo(x0 - 1, y0 - 1);\n      ctx.closePath();\n\n      // Draw inner rectangle.\n      ctx.moveTo(x1, y1);\n      ctx.lineTo(x1, y2);\n      ctx.lineTo(x2, y2);\n      ctx.lineTo(x2, y1);\n      ctx.lineTo(x1, y1);\n\n      ctx.closePath();\n      ctx.fill();\n    },\n\n    _renderBorders: function(ctx) {\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2); // upper left\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2); // upper right\n      ctx.lineTo(this.getWidth() / 2, this.getHeight() / 2); // down right\n      ctx.lineTo(-this.getWidth() / 2, this.getHeight() / 2); // down left\n      ctx.lineTo(-this.getWidth() / 2, -this.getHeight() / 2); // upper left\n      ctx.stroke();\n    },\n\n    _renderGrid: function(ctx) {\n      // Vertical lines\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2 + 1 / 3 * this.getWidth(), -this.getHeight() / 2);\n      ctx.lineTo(-this.getWidth() / 2 + 1 / 3 * this.getWidth(), this.getHeight() / 2);\n      ctx.stroke();\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2 + 2 / 3 * this.getWidth(), -this.getHeight() / 2);\n      ctx.lineTo(-this.getWidth() / 2 + 2 / 3 * this.getWidth(), this.getHeight() / 2);\n      ctx.stroke();\n      // Horizontal lines\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2 + 1 / 3 * this.getHeight());\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2 + 1 / 3 * this.getHeight());\n      ctx.stroke();\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2 + 2 / 3 * this.getHeight());\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2 + 2 / 3 * this.getHeight());\n      ctx.stroke();\n    }\n  });\n\n  Darkroom.plugins.crop = Darkroom.Plugin.extend({\n    // Init point\n    startX: null,\n    startY: null,\n\n    // Keycrop\n    isKeyCroping: false,\n    isKeyLeft: false,\n    isKeyUp: false,\n\n    defaults: {\n      // min crop dimension\n      minHeight: 1,\n      minWidth: 1,\n      // ensure crop ratio\n      ratio: null,\n      // quick crop feature (set a key code to enable it)\n      quickCropKey: false\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.cropButton = buttonGroup.createButton({\n        image: \"crop\"\n      });\n      this.okButton = buttonGroup.createButton({\n        image: \"done\",\n        type: \"success\",\n        hide: true\n      });\n      this.cancelButton = buttonGroup.createButton({\n        image: \"close\",\n        type: \"danger\",\n        hide: true\n      });\n\n      // Buttons click\n      this.cropButton.addEventListener(\"click\", this.toggleCrop.bind(this));\n      this.okButton.addEventListener(\"click\", this.cropCurrentZone.bind(this));\n      this.cancelButton.addEventListener(\"click\", this.releaseFocus.bind(this));\n\n      // Canvas events\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.onMouseUp.bind(this));\n      this.darkroom.canvas.on(\"object:moving\", this.onObjectMoving.bind(this));\n      this.darkroom.canvas.on(\"object:scaling\", this.onObjectScaling.bind(this));\n\n      fabric.util.addListener(fabric.document, \"keydown\", this.onKeyDown.bind(this));\n      fabric.util.addListener(fabric.document, \"keyup\", this.onKeyUp.bind(this));\n\n      this.darkroom.addEventListener(\"core:transformation\", this.releaseFocus.bind(this));\n    },\n\n    clear: function() {\n      if (this.hasFocus()) {\n        this.releaseFocus();\n      }\n    },\n    // Avoid crop zone to go beyond the canvas edges\n    onObjectMoving: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      const currentObject = event.target;\n      if (currentObject !== this.cropZone) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      const x = currentObject.getLeft();\n      const y = currentObject.getTop();\n      const w = currentObject.getWidth();\n      const h = currentObject.getHeight();\n      const maxX = canvas.getWidth() - w;\n      const maxY = canvas.getHeight() - h;\n\n      if (x < 0) {\n        currentObject.set(\"left\", 0);\n      }\n      if (y < 0) {\n        currentObject.set(\"top\", 0);\n      }\n      if (x > maxX) {\n        currentObject.set(\"left\", maxX);\n      }\n      if (y > maxY) {\n        currentObject.set(\"top\", maxY);\n      }\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    // Prevent crop zone from going beyond the canvas edges (like mouseMove)\n    onObjectScaling: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      let preventScaling = false;\n      const currentObject = event.target;\n      if (currentObject !== this.cropZone) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n\n      const minX = currentObject.getLeft();\n      const minY = currentObject.getTop();\n      const maxX = currentObject.getLeft() + currentObject.getWidth();\n      const maxY = currentObject.getTop() + currentObject.getHeight();\n\n      if (this.options.ratio !== null) {\n        if (minX < 0 || maxX > canvas.getWidth() || minY < 0 || maxY > canvas.getHeight()) {\n          preventScaling = true;\n        }\n      }\n\n      if (minX < 0 || maxX > canvas.getWidth() || preventScaling) {\n        const lastScaleX = this.lastScaleX || 1;\n        currentObject.setScaleX(lastScaleX);\n      }\n      if (minX < 0) {\n        currentObject.setLeft(0);\n      }\n\n      if (minY < 0 || maxY > canvas.getHeight() || preventScaling) {\n        const lastScaleY = this.lastScaleY || 1;\n        currentObject.setScaleY(lastScaleY);\n      }\n      if (minY < 0) {\n        currentObject.setTop(0);\n      }\n\n      if (currentObject.getWidth() < this.options.minWidth) {\n        currentObject.scaleToWidth(this.options.minWidth);\n      }\n      if (currentObject.getHeight() < this.options.minHeight) {\n        currentObject.scaleToHeight(this.options.minHeight);\n      }\n\n      this.lastScaleX = currentObject.getScaleX();\n      this.lastScaleY = currentObject.getScaleY();\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    // Init crop zone\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n\n      // recalculate offset, in case canvas was manipulated since last `calcOffset`\n      canvas.calcOffset();\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n      const point = new fabric.Point(x, y);\n\n      // Check if user want to scale or drag the crop zone.\n      const activeObject = canvas.getActiveObject();\n      if (activeObject === this.cropZone || this.cropZone.containsPoint(point)) {\n        return;\n      }\n\n      canvas.discardActiveObject();\n      this.cropZone.setWidth(0);\n      this.cropZone.setHeight(0);\n      this.cropZone.setScaleX(1);\n      this.cropZone.setScaleY(1);\n\n      this.startX = x;\n      this.startY = y;\n    },\n\n    // Extend crop zone\n    onMouseMove: function(event) {\n      // Quick crop feature\n      if (this.isKeyCroping) {\n        return this.onMouseMoveKeyCrop(event);\n      }\n\n      if (null === this.startX || null === this.startY) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n\n      this._renderCropZone(this.startX, this.startY, x, y);\n    },\n\n    onMouseMoveKeyCrop: function(event) {\n      const canvas = this.darkroom.canvas;\n      const zone = this.cropZone;\n\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n\n      if (!zone.left || !zone.top) {\n        zone.setTop(y);\n        zone.setLeft(x);\n      }\n\n      this.isKeyLeft = x < zone.left + zone.width / 2;\n      this.isKeyUp = y < zone.top + zone.height / 2;\n\n      this._renderCropZone(\n        Math.min(zone.left, x),\n        Math.min(zone.top, y),\n        Math.max(zone.left + zone.width, x),\n        Math.max(zone.top + zone.height, y)\n      );\n    },\n\n    // Finish crop zone\n    onMouseUp: function(event) {\n      if (null === this.startX || null === this.startY) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      this.cropZone.setCoords();\n      canvas.setActiveObject(this.cropZone);\n      canvas.calcOffset();\n\n      this.startX = null;\n      this.startY = null;\n    },\n\n    onKeyDown: function(event) {\n      if (this.options.quickCropKey === false || event.keyCode !== this.options.quickCropKey || this.isKeyCroping) {\n        return;\n      }\n\n      // Active quick crop flow\n      this.isKeyCroping = true;\n      this.darkroom.canvas.discardActiveObject();\n      this.cropZone.setWidth(0);\n      this.cropZone.setHeight(0);\n      this.cropZone.setScaleX(1);\n      this.cropZone.setScaleY(1);\n      this.cropZone.setTop(0);\n      this.cropZone.setLeft(0);\n    },\n\n    onKeyUp: function(event) {\n      if (this.options.quickCropKey === false || event.keyCode !== this.options.quickCropKey || !this.isKeyCroping) {\n        return;\n      }\n\n      // Unactive quick crop flow\n      this.isKeyCroping = false;\n      this.startX = 1;\n      this.startY = 1;\n      this.onMouseUp();\n    },\n\n    selectZone: function(x, y, width, height, forceDimension) {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n      }\n\n      if (!forceDimension) {\n        this._renderCropZone(x, y, x + width, y + height);\n      } else {\n        this.cropZone.set({\n          left: x,\n          top: y,\n          width: width,\n          height: height\n        });\n      }\n\n      const canvas = this.darkroom.canvas;\n      canvas.bringToFront(this.cropZone);\n      this.cropZone.setCoords();\n      canvas.setActiveObject(this.cropZone);\n      canvas.calcOffset();\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    toggleCrop: function() {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n      } else {\n        this.releaseFocus();\n      }\n    },\n\n    cropCurrentZone: function() {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      // Avoid croping empty zone\n      if (this.cropZone.width < 1 && this.cropZone.height < 1) {\n        return;\n      }\n\n      const image = this.darkroom.image;\n\n      // Compute crop zone dimensions\n      let top = this.cropZone.getTop() - image.getTop();\n      let left = this.cropZone.getLeft() - image.getLeft();\n      let width = this.cropZone.getWidth();\n      let height = this.cropZone.getHeight();\n\n      // Adjust dimensions to image only\n      if (top < 0) {\n        height += top;\n        top = 0;\n      }\n\n      if (left < 0) {\n        width += left;\n        left = 0;\n      }\n\n      this.darkroom.options.top = 0;\n      this.darkroom.options.left = 0;\n      this.darkroom.canvas.setZoom(1);\n      this.darkroom.plugins.zoom.updateButtons();\n      // Apply crop transformation.\n      // Make sure to use relative dimension since the crop will be applied\n      // on the source image.\n      this.darkroom.applyTransformation(new Crop({\n        top: top / image.getHeight(),\n        left: left / image.getWidth(),\n        width: width / image.getWidth(),\n        height: height / image.getHeight()\n      }));\n    },\n\n    // Test wether crop zone is set\n    hasFocus: function() {\n      return this.cropZone !== undefined;\n    },\n\n    // Create the crop zone\n    requireFocus: function() {\n      this.darkroom.clearFocus(\"crop\");\n      this.cropZone = new CropZone({\n        fill: \"transparent\",\n        hasBorders: false,\n        originX: \"left\",\n        originY: \"top\",\n        // stroke: '#444',\n        // strokeDashArray: [5, 5],\n        // borderColor: '#444',\n        cornerColor: \"#444\",\n        cornerSize: 8,\n        transparentCorners: false,\n        lockRotation: true,\n        hasRotatingPoint: false\n      });\n\n      if (this.options.ratio !== null) {\n        this.cropZone.set(\"lockUniScaling\", true);\n      }\n\n      this.darkroom.canvas.add(this.cropZone);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n\n      this.cropButton.active(true);\n      this.okButton.hide(false);\n      this.cancelButton.hide(false);\n    },\n\n    // Remove the crop zone\n    releaseFocus: function() {\n      if (undefined === this.cropZone) {\n        return;\n      }\n\n      this.cropZone.remove();\n      this.cropZone = undefined;\n\n      this.cropButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n\n      this.darkroom.canvas.defaultCursor = \"default\";\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    _renderCropZone: function(fromX, fromY, toX, toY) {\n      const canvas = this.darkroom.canvas;\n\n      const isRight = (toX > fromX);\n      let isLeft = !isRight;\n      const isDown = (toY > fromY);\n      let isUp = !isDown;\n\n      const minWidth = Math.min(+this.options.minWidth, canvas.getWidth());\n      const minHeight = Math.min(+this.options.minHeight, canvas.getHeight());\n\n      // Define corner coordinates\n      let leftX = Math.min(fromX, toX);\n      let rightX = Math.max(fromX, toX);\n      let topY = Math.min(fromY, toY);\n      let bottomY = Math.max(fromY, toY);\n\n      // Replace current point into the canvas\n      leftX = Math.max(0, leftX);\n      rightX = Math.min(canvas.getWidth(), rightX);\n      topY = Math.max(0, topY);\n      bottomY = Math.min(canvas.getHeight(), bottomY);\n\n      // Recalibrate coordinates according to given options\n      if (rightX - leftX < minWidth) {\n        if (isRight) {\n          rightX = leftX + minWidth;\n        } else {\n          leftX = rightX - minWidth;\n        }\n      }\n      if (bottomY - topY < minHeight) {\n        if (isDown) {\n          bottomY = topY + minHeight;\n        } else {\n          topY = bottomY - minHeight;\n        }\n      }\n\n      // Truncate truncate according to canvas dimensions\n      if (leftX < 0) {\n        // Translate to the left\n        rightX += Math.abs(leftX);\n        leftX = 0;\n      }\n      if (rightX > canvas.getWidth()) {\n        // Translate to the right\n        leftX -= (rightX - canvas.getWidth());\n        rightX = canvas.getWidth();\n      }\n      if (topY < 0) {\n        // Translate to the bottom\n        bottomY += Math.abs(topY);\n        topY = 0;\n      }\n      if (bottomY > canvas.getHeight()) {\n        // Translate to the right\n        topY -= (bottomY - canvas.getHeight());\n        bottomY = canvas.getHeight();\n      }\n\n      let width = rightX - leftX;\n      let height = bottomY - topY;\n      const currentRatio = width / height;\n\n      if (this.options.ratio && +this.options.ratio !== currentRatio) {\n        const ratio = +this.options.ratio;\n\n        if (this.isKeyCroping) {\n          isLeft = this.isKeyLeft;\n          isUp = this.isKeyUp;\n        }\n\n        if (currentRatio < ratio) {\n          const newWidth = height * ratio;\n          if (isLeft) {\n            leftX -= (newWidth - width);\n          }\n          width = newWidth;\n        } else if (currentRatio > ratio) {\n          const newHeight = height / (ratio * height / width);\n          if (isUp) {\n            topY -= (newHeight - height);\n          }\n          height = newHeight;\n        }\n\n        if (leftX < 0) {\n          leftX = 0;\n          // TODO\n        }\n        if (topY < 0) {\n          topY = 0;\n          // TODO\n        }\n        if (leftX + width > canvas.getWidth()) {\n          const newWidth = canvas.getWidth() - leftX;\n          height = newWidth * height / width;\n          width = newWidth;\n          if (isUp) {\n            topY = fromY - height;\n          }\n        }\n        if (topY + height > canvas.getHeight()) {\n          const newHeight = canvas.getHeight() - topY;\n          width = width * newHeight / height;\n          height = newHeight;\n          if (isLeft) {\n            leftX = fromX - width;\n          }\n        }\n      }\n\n      // Apply coordinates\n      this.cropZone.left = leftX;\n      this.cropZone.top = topY;\n      this.cropZone.width = width;\n      this.cropZone.height = height;\n\n      this.darkroom.canvas.bringToFront(this.cropZone);\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    }\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Pencil = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      // Snapshot the image delimited by the crop zone\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) {\n          return;\n        }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      canvas.add(this.options.pencilZone);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  const PencilZone = fabric.util.createClass(fabric.Object, {\n    type: \"pencilObject\",\n\n    initialize: function(options) {\n      this.options = options || {};\n      this.left = options.left || 0;\n      this.top = options.top || 0;\n      this.width = options.width;\n      this.height = options.height;\n      this.originalWidth = this.width;\n      this.originalHeight = this.height;\n      this.stroke = options.stroke;\n      this.size = options.size;\n      this.path = null;\n      this.cX = -this.getWidth() / 2;\n      this.cY = -this.getHeight() / 2;\n      this.callSuper(\"initialize\", options);\n    },\n\n    _applyScaling: function(w, h) {\n      const scaleX = this.getWidth() / w;\n      const scaleY = this.getHeight() / h;\n      this.set({\n        scaleX: this.scaleX / scaleX,\n        scaleY: this.scaleY / scaleY\n      });\n    },\n\n    _render: function(ctx) {\n      this._draw(ctx);\n    },\n\n    _draw: function(ctx) {\n      if (ctx && this.path && this.path.length > 0) {\n        ctx.lineJoin = \"round\";\n        ctx.lineCap = \"round\";\n        ctx.beginPath();\n        ctx.moveTo(\n          (this.cX + this.path[0].x),\n          (this.cY + this.path[0].y)\n        );\n\n        for (let x = 0; x < this.path.length; x += 1) {\n          const drawX = (this.cX + this.path[x].x);\n          const drawY = (this.cY + this.path[x].y);\n          ctx.lineTo(drawX, drawY);\n        }\n\n\n        ctx.strokeStyle = this.stroke;\n        ctx.lineWidth = this.size;\n        ctx.stroke();\n      }\n    },\n\n    clone: function() {\n      return new PencilZone({\n        stroke: this.stroke,\n        size: this.size,\n        width: this.width,\n        height: this.height,\n        selectable: false,\n        lockRotation: true,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false\n      });\n    }\n  });\n\n  Darkroom.plugins.pencil = Darkroom.Plugin.extend({\n    pencilMode: false,\n    pencilZone: null,\n    mouseDown: false,\n    size: 30,\n    defaults: {\n      stroke: \"black\",\n      size: 30\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.pencilButton = buttonGroup.createButton({\n        image: \"pencil\",\n        tooltip: \"Draw\"\n      });\n\n      this.path = [];\n      this.width = this.darkroom.canvas.width;\n      this.height = this.darkroom.canvas.height;\n      this.size = this.options.size;\n      this.stroke = this.options.stroke;\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.onMouseUp.bind(this));\n      this.pencilButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      if (this.hasFocus()) {\n        this.removeFocus();\n      }\n    },\n\n    hasFocus: function() {\n      return this.pencilZone !== null;\n    },\n\n    onClick: function() {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n        return;\n      }\n      this.removeFocus();\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      this.darkroom.dispatchEvent(\"pencil:begin\");\n      this.mouseDown = true;\n      this.path.push(this.getMousePosition(this.canvas, event));\n      this.renderZone();\n    },\n\n    onMouseUp: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      if (this.mouseDown) {\n        this.darkroom.dispatchEvent(\"pencil:end\");\n        this.mouseDown = false;\n        const mousePos = this.getMousePosition(this.canvas, event);\n        this.path.push(mousePos);\n        this.renderZone();\n        this.applyPencilPath();\n        this.path = [];\n      }\n    },\n\n    onMouseMove: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      if (this.mouseDown) {\n        const mousePos = this.getMousePosition(this.canvas, event);\n        this.path.push(mousePos);\n        this.renderZone();\n      }\n    },\n\n    removeFocus: function() {\n      this.darkroom.dispatchEvent(\"pencil:disabled\");\n      this.pencilButton.active(false);\n      this.pencilMode = false;\n      this.darkroom.canvas.defaultCursor = \"default\";\n      if (this.pencilZone) {\n        this.pencilZone.remove();\n        this.pencilZone = null;\n      }\n    },\n\n    requireFocus: function() {\n      this.darkroom.clearFocus(\"pencil\");\n      this.darkroom.dispatchEvent(\"pencil:enabled\");\n      this.pencilMode = true;\n      this.pencilButton.active(true);\n\n      this.pencilZone = this.newZone();\n\n      this.darkroom.canvas.add(this.pencilZone);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n    },\n\n    getMousePosition: function(canvas, event) {\n      var canvas = this.darkroom.canvas;\n      const rect = canvas.lowerCanvasEl.getBoundingClientRect();\n      const pointer = canvas.getPointer(event.e);\n      return {\n        x: pointer.x,\n        y: pointer.y,\n        cx: event.e.clientX - rect.left,\n        cy: event.e.clientY - rect.top,\n        sx: event.e.screenX,\n        sy: event.e.screenY\n      };\n    },\n\n    newZone: function() {\n      this.path = [];\n      const stroke = this.stroke;\n      const size = this.size;\n      const width = this.darkroom.canvas.getWidth();\n      const height = this.darkroom.canvas.getHeight();\n      const zone = new PencilZone({\n        stroke: stroke,\n        size: size,\n        width: width,\n        height: height,\n        selectable: false,\n        lockRotation: true,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false\n      });\n\n      return zone;\n    },\n\n    applyPencilPath: function() {\n      if (!this.hasFocus()) { return; }\n\n      // Avoid croping empty zone\n      if (this.path === null || this.path.length === 0) { return; }\n\n      // Apply crop transformation.\n      // Make sure to use relative dimension since the crop will be applied\n      // on the source image.\n      const zone = this.pencilZone.clone();\n      zone.path = this.path;\n      zone.left -= this.darkroom.options.left;\n      zone.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Pencil({\n        pencilZone: zone\n      }));\n      this.darkroom.canvas.add(this.pencilZone);\n    },\n\n    renderZone: function() {\n      if (this.path && this.path.length > 0) {\n        this.darkroom.dispatchEvent(\"pencil:update\");\n        this.pencilZone.path = this.path;\n      }\n      this.darkroom.canvas.bringToFront(this.pencilZone);\n    }\n\n  });\n}());\n","(function () {\n  let rect;\n  let isDown;\n  let origX;\n  let origY;\n  let hasFocus = true;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const RectangleTransformation = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        if (this.height < 1 || this.width < 1) { return; }\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      const newRect = this.options.rectangle;\n      const scaleX = this.options.originalWidth / canvas.getWidth();\n      const scaleY = this.options.originalHeight / canvas.getHeight();\n      newRect.set({ scaleX: 1 / scaleX, scaleY: 1 / scaleY });\n      canvas.add(newRect);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  Darkroom.plugins.rectangle = Darkroom.Plugin.extend({\n    rectangleZone: null,\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.rectangleButton = buttonGroup.createButton({\n        image: \"rectangle\",\n        tooltip: \"Draw\"\n      });\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n      this.rectangleButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      this.rectangleButton.active(false);\n      hasFocus = true;\n      isDown = false;\n    },\n\n    onClick: function() {\n      if (!hasFocus) {\n        this.rectangleButton.active(false);\n        this.darkroom.canvas.defaultCursor = \"default\";\n        hasFocus = true;\n        return;\n      }\n      this.darkroom.clearFocus(\"rectangle\");\n      this.rectangleButton.active(true);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n      hasFocus = false;\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (hasFocus) return;\n      this.darkroom.dispatchEvent(\"rectangle:begin\");\n      const canvas = this.darkroom.canvas;\n      isDown = true;\n      const pointer = canvas.getPointer(event.e);\n      origX = pointer.x;\n      origY = pointer.y;\n      rect = new fabric.Rect({\n        left: origX,\n        top: origY,\n        originX: \"left\",\n        originY: \"top\",\n        width: pointer.x - origX,\n        height: pointer.y - origY,\n        angle: 0,\n        fill: \"black\",\n        transparentCorners: false\n      });\n      canvas.add(rect);\n    },\n\n    onMouseMove: function(event) {\n      if (isDown) {\n        this.darkroom.dispatchEvent(\"rectangle:update\");\n        const canvas = this.darkroom.canvas;\n        const pointer = canvas.getPointer(event.e);\n\n        if (origX > pointer.x) {\n          rect.set({\n            left: Math.abs(pointer.x)\n          });\n        }\n        if (origY > pointer.y) {\n          rect.set({\n            top: Math.abs(pointer.y)\n          });\n        }\n\n        rect.set({\n          width: Math.abs(origX - pointer.x)\n        });\n        rect.set({\n          height: Math.abs(origY - pointer.y)\n        });\n        canvas.renderAll();\n      }\n    },\n\n    handleMouseUp: function() {\n      if (hasFocus) return;\n      const canvas = this.darkroom.canvas;\n      isDown = false;\n      this.darkroom.dispatchEvent(\"rectangle:end\");\n      rect.left -= this.darkroom.options.left;\n      rect.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new RectangleTransformation({\n        originalWidth: canvas.getWidth(),\n        originalHeight: canvas.getHeight(),\n        rectangle: rect\n      }));\n    }\n  });\n}());\n","(function () {\n  let hasFocus = false;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const Text = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) { return; }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      canvas.add(this.options.text);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n  Darkroom.plugins.text = Darkroom.Plugin.extend({\n    defaults: {\n      text: \"Sample Text...\"\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.textButton = buttonGroup.createButton({\n        image: \"text\",\n        title: \"Add Text\"\n      });\n      this.okButton = buttonGroup.createButton({\n        image: \"done\",\n        type: \"success\",\n        hide: true\n      });\n      this.cancelButton = buttonGroup.createButton({\n        image: \"close\",\n        type: \"danger\",\n        hide: true\n      });\n      this.colorPicker = buttonGroup.createInput({\n        image: \"color\",\n        title: \"Color Picker\",\n        value: \"#ff0000\",\n        hide: true\n      });\n\n      this.textButton.addEventListener(\"click\", this.addText.bind(this));\n      this.okButton.addEventListener(\"click\", this.saveText.bind(this));\n      this.colorPicker.addEventListener(\"change\", this.setColor.bind(this));\n      this.cancelButton.addEventListener(\"click\", this.releaseFocus.bind(this));\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n\n      this.darkroom.addEventListener(\"core:transformation\", this.releaseFocus.bind(this));\n    },\n\n    clear: function() {\n      this.textButton.active(false);\n      this.darkroom.dispatchEvent(\"text:end\");\n      this.textButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      if (this.newText) {\n        this.newText.remove();\n      }\n      hasFocus = false;\n    },\n\n    addText: function() {\n      if (hasFocus) {\n        this.releaseFocus();\n        return;\n      }\n      hasFocus = true;\n      this.darkroom.clearFocus(\"text\");\n\n      this.textButton.active(true);\n      this.okButton.hide(false);\n      this.cancelButton.hide(false);\n      this.colorPicker.hide(false);\n\n      const factor = this.darkroom.image.scaleX;\n      const text = new fabric.IText(this.options.text, {\n        left: 100 * factor,\n        top: 100 * factor,\n        fill: \"red\",\n        fontFamily: \"Monospace\",\n        fontSize: 25,\n        lockSkewingX: true,\n        lockSkewingY: true,\n        scaleX: factor,\n        scaleY: factor,\n        padding: 10\n      });\n\n      text.on(this.image = this.darkroom.image, {\n        scaling: function() {\n          this.factor = this.image.scaleX;\n\n          const obj = this;\n          const w = obj.width * obj.scaleX / factor;\n          const h = obj.height * obj.scaleY / factor;\n\n          obj.set({\n            width: w,\n            height: h,\n            scaleX: factor,\n            scaleY: factor\n          });\n        }\n      });\n\n      this.darkroom.canvas.add(text);\n      this.newText = text;\n    },\n\n    setColor: function(event) {\n      this.setStyle(this, \"fill\", event.target.value);\n    },\n\n    setStyle: function(object, styleName, value) {\n      const canvas = this.darkroom.canvas;\n      object.newText.fill = value;\n      canvas.renderAll();\n    },\n\n    saveText: function() {\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      this.textButton.active(false);\n      this.newText.hasBorders = false;\n      this.newText.hasControls = false;\n      this.newText.hasRotatingPoint = false;\n      this.newText.selectionColor = \"transparent\";\n      this.newText.abortCursorAnimation();\n      this.newText.left -= this.darkroom.options.left;\n      this.newText.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Text({\n        text: this.newText\n      }));\n      this.colorPicker.element.value = \"#ff0000\";\n    },\n\n    onMouseDown: function() {\n      if (!this.newText) { return; }\n      if (this.newText.active) {\n        this.isDown = true;\n      }\n    },\n\n    onMouseMove: function() {\n      if (!this.isDown) {\n        return;\n      }\n      const canvas = this.darkroom.canvas;\n      canvas.renderAll();\n    },\n\n    handleMouseUp: function() {\n      this.isDown = false;\n    },\n\n    releaseFocus: function() {\n      if (this.newText) { this.newText.remove(); }\n\n      this.textButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      this.colorPicker.element.value = \"#ff0000\";\n      hasFocus = false;\n    }\n  });\n}());\n","(function () {\n  let ellip;\n  let isDown = false;\n  let origX;\n  let origY;\n  let hasFocus = true;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const Circle = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) { return; }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next:(imgInstance);\n      };\n\n      canvas.add(this.options.circle);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  Darkroom.plugins.circle = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.circleButton = buttonGroup.createButton({\n        image: \"circle\",\n        tooltip: \"circle\"\n      });\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n      this.circleButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      this.circleButton.active(false);\n      hasFocus = true;\n      isDown = false;\n    },\n\n    onClick: function() {\n      if (!hasFocus) {\n        this.circleButton.active(false);\n        this.darkroom.canvas.defaultCursor = \"default\";\n        hasFocus = true;\n        return;\n      }\n      this.darkroom.clearFocus(\"circle\");\n      this.circleButton.active(true);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n      hasFocus = false;\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      this.isDown = true;\n      if (hasFocus) return;\n      const canvas = this.darkroom.canvas;\n      isDown = true;\n      const pointer = canvas.getPointer(event.e);\n      origX = pointer.x;\n      origY = pointer.y;\n      ellip = new fabric.Ellipse({\n        left: pointer.x,\n        top: pointer.y,\n        stroke: \"black\",\n        fill: \"transparent\",\n        strokeWidth: \"4\",\n        rx: 0,\n        ry: 0\n      });\n      canvas.add(ellip);\n    },\n\n    onMouseMove: function(event) {\n      if (!this.isDown) {\n        return;\n      }\n      if (isDown && ellip) {\n        this.darkroom.dispatchEvent(\"circle:update\");\n        const canvas = this.darkroom.canvas;\n        const pointer = canvas.getPointer(event.e);\n\n        if (origX > pointer.x) {\n          ellip.set({\n            left: Math.abs(pointer.x)\n          });\n        }\n\n        if (origY > pointer.y) {\n          ellip.set({\n            top: Math.abs(pointer.y)\n          });\n        }\n\n        ellip.set({\n          rx: Math.abs(origX - pointer.x) / 2\n        });\n        ellip.set({\n          ry: Math.abs(origY - pointer.y) / 2\n        });\n\n        ellip.setCoords();\n        canvas.renderAll();\n      }\n    },\n\n    handleMouseUp: function() {\n      if (hasFocus) { return; }\n      this.isDown = false;\n      ellip.left -= this.darkroom.options.left;\n      ellip.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Circle({\n        circle: ellip\n      }));\n    }\n  });\n}());\n","(function () {\n  const maxZoom = 2;\n  let minZoom;\n  const states = {\n    panActive: false,\n    panning: false,\n    realPan: null\n  };\n  let origX;\n  let origY;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  Darkroom.plugins.zoom = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.zoomInButton = buttonGroup.createButton({\n        image: \"zoom-in\",\n        title: \"zoom in\"\n      });\n\n      this.zoomOutButton = buttonGroup.createButton({\n        image: \"zoom-out\",\n        title: \"zoom out\",\n        disabled: true\n      });\n\n      this.pan = buttonGroup.createButton({\n        image: \"hand\",\n        disabled: true\n      });\n\n      this.zoomInButton.addEventListener(\"click\", this.zoomIn.bind(this));\n      this.zoomOutButton.addEventListener(\"click\", this.zoomOut.bind(this));\n      this.pan.addEventListener(\"click\", this.onPan.bind(this));\n      minZoom = this.darkroom.canvas.getZoom();\n    },\n\n    clear: function() {\n      states.panActive = false;\n      this.pan.active(false);\n      states.panning = false;\n      this.darkroom.canvas.defaultCursor = \"default\";\n      this.restoreZoomPan();\n    },\n\n    onPan: function() {\n      if (arguments.length > 0 && states.panActive) {\n        this.clear();\n      } else {\n        this.darkroom.clearFocus(\"zoom\");\n        const canvas = this.darkroom.canvas;\n        canvas.on(\"mouse:down\", this.panEnter.bind(this));\n        canvas.on(\"mouse:up\", this.panExit.bind(this));\n        canvas.on(\"mouse:move\", this._pan.bind(this));\n        canvas.defaultCursor = \"move\";\n        states.panActive = true;\n        this.pan.active(true);\n      }\n    },\n\n\n    panEnter: function(event) {\n      if (states.panActive) {\n        states.panning = true;\n        origX = event.e.screenX || event.e.touches[0].screenX;\n        origY = event.e.screenY || event.e.touches[0].screenY;\n      }\n    },\n\n    _pan: function(event) {\n      event.e.preventDefault();\n      const canvas = this.darkroom.canvas;\n      if (states.panning && event) {\n        if (event.e.touches) {\n          const touches = event.e.touches;\n          for (let i = 0; i < touches.length; i += 1) {\n            const delta = {\n              x: (touches[i].screenX - origX),\n              y: (touches[i].screenY - origY)\n            };\n            origX = touches[i].screenX;\n            origY = touches[i].screenY;\n            this.doPan(delta);\n          }\n        } else {\n          const delta = {\n            x: (event.e.screenX - origX),\n            y: (event.e.screenY - origY)\n          };\n          origX = event.e.screenX;\n          origY = event.e.screenY;\n          this.doPan(delta);\n        }\n        canvas.renderAll();\n      }\n    },\n\n    panExit: function() {\n      if (states.panActive) {\n        const options = this.darkroom.options;\n        const image = this.darkroom.image;\n        states.panning = false;\n        options.left = image.left;\n        options.top = image.top;\n      }\n    },\n\n    doPan: function(delta) {\n      const image = this.darkroom.image;\n      this.delta = this.constrainPan(delta);\n      image.left += delta.x;\n      image.top += delta.y;\n      image.setCoords();\n      states.realPan = {\n        x: image.oCoords.tl.x,\n        y: image.oCoords.tl.y\n      };\n    },\n\n    constrainPan: function(delta) {\n      const canvas = this.darkroom.canvas;\n      const image = this.darkroom.image;\n      const tl = image.oCoords.tl; // top left corner\n      const br = image.oCoords.br; // bottom right corner\n      if (delta.x > 0) {\n        delta.x = ((tl.x + delta.x) > 0) ? 0 : Math.floor(delta.x);\n      } else {\n        delta.x = Math.ceil(delta.x);\n        if ((Math.ceil(br.x) + delta.x) < canvas.width) {\n          delta.x = 0;\n        }\n      }\n      if (delta.y > 0) {\n        delta.y = ((tl.y + delta.y) > 0) ? 0 : Math.floor(delta.y);\n      } else {\n        delta.y = Math.ceil(delta.y);\n        if ((Math.ceil(br.y) + delta.y) < canvas.height) {\n          delta.y = 0;\n        }\n      }\n      return delta;\n    },\n\n    restoreZoomPan: function() {\n      if (states.lastScale != null) {\n        this.darkroom.image.scale(states.lastScale);\n        if (states.realPan != null) {\n          this.doPan(states.realPan);\n        }\n      }\n    },\n\n    zoomIn: function() {\n      const canvas = this.darkroom.canvas;\n      if (canvas.getZoom() <= maxZoom) {\n        canvas.setZoom(canvas.getZoom() * 1.1);\n        canvas.renderAll();\n      }\n      this.onPan();\n      this.updateButtons();\n    },\n\n    zoomOut: function() {\n      const canvas = this.darkroom.canvas;\n      const image = this.darkroom.image;\n      if (canvas.getZoom() > minZoom) {\n        canvas.setZoom(canvas.getZoom() / 1.1);\n        if (this.darkroom.image.oCoords.br.x < this.darkroom.canvas.width) {\n          const newLeft = Math.ceil(this.darkroom.canvas.width - this.darkroom.image.oCoords.br.x);\n          image.left += newLeft;\n          this.darkroom.options.left = this.darkroom.options.left + newLeft;\n        }\n        if (this.darkroom.image.oCoords.br.y < this.darkroom.canvas.height) {\n          const newTop = Math.ceil(this.darkroom.canvas.height - this.darkroom.image.oCoords.br.y);\n          image.top += newTop;\n          this.darkroom.options.top = this.darkroom.options.top + newTop;\n        }\n        image.setCoords();\n        canvas.renderAll();\n        if (canvas.getZoom() >= minZoom) {\n          this.onPan();\n        }\n        this.updateButtons();\n      }\n    },\n\n    updateButtons: function() {\n      const canvas = this.darkroom.canvas;\n      const shouldDisableMinZoom = canvas.getZoom() <= minZoom;\n      const shouldDisableMaxZooom = canvas.getZoom() >= maxZoom;\n      const shouldEnablePan = canvas.getZoom() > minZoom;\n      this.zoomOutButton.disable(shouldDisableMinZoom);\n      this.zoomInButton.disable(shouldDisableMaxZooom);\n      if (shouldDisableMinZoom) {\n        this.darkroom.options.left = 0;\n        this.darkroom.options.top = 0;\n        this.pan.active(false);\n        this.pan.disable(true);\n        this.clear();\n      }\n\n      if (shouldEnablePan) {\n        this.pan.disable(false);\n      }\n    }\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  Darkroom.plugins.save = Darkroom.Plugin.extend({\n\n    defaults: {\n      callback: function() {\n        this.darkroom.selfDestroy();\n      }\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.destroyButton = buttonGroup.createButton({\n        image: \"save\"\n      });\n\n      this.destroyButton.addEventListener(\"click\", this.options.callback.bind(this));\n    }\n  });\n}());\n"]} +//# sourceMappingURL=data:application/json;charset=utf8;base64,{"version":3,"sources":["bootstrap.js","darkroom.js","plugin.js","transformation.js","ui.js","utils.js","darkroom.history.js","darkroom.rotate.js","darkroom.crop.js","darkroom.pencil.js","darkroom.rectangle.js","darkroom.text.js","darkroom.circle.js","zoom.js","darkroom.save.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCxXA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC5CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCtCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC1KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC/BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CClEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCxDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC9rBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCrSA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CChJA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCpMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CC7IA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA,CCjNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"darkroom.js","sourcesContent":["(function() {\n    'use strict';\n\n    // Inject SVG icons into the DOM\n    var element = document.createElement('div');\n    element.id = 'darkroom-icons';\n    element.style.height = 0;\n    element.style.width = 0;\n    element.style.position = 'absolute';\n    element.style.visibility = 'hidden';\n    element.innerHTML = '<!-- inject:svg --><!-- endinject -->';\n    document.body.appendChild(element);\n\n    })();\n","(function () {\n  'use strict';\n\n  window.Darkroom = Darkroom;\n\n  // Core object of DarkroomJS.\n  // Basically it's a single object, instanciable via an element\n  // (it could be a CSS selector or a DOM element), some custom options,\n  // and a list of plugin objects (or none to use default ones).\n  function Darkroom(element, options, plugins) {\n    return this.constructor(element, options, plugins);\n  }\n\n  // Create an empty list of plugin objects, which will be filled by\n  // other plugin scripts. This is the default plugin list if none is\n  // specified in Darkroom'ss constructor.\n  Darkroom.plugins = [];\n\n  Darkroom.prototype = {\n    // Reference to the main container element\n    containerElement: null,\n\n    // Reference to the Fabric canvas object\n    canvas: null,\n\n    // Reference to the Fabric image object\n    image: null,\n\n    // Reference to the Fabric source canvas object\n    sourceCanvas: null,\n\n    // Reference to the Fabric source image object\n    sourceImage: null,\n\n    // Track of the original image element\n    originalImageElement: null,\n\n    // Stack of transformations to apply to the image source\n    transformations: [],\n\n    // Default options\n    defaults: {\n      // Canvas properties (dimension, ratio, color)\n      minWidth: null,\n      minHeight: null,\n      maxWidth: null,\n      maxHeight: null,\n      ratio: null,\n      left: 0,\n      top: 0,\n      backgroundColor: '#fff',\n\n      // Plugins options\n      plugins: {},\n\n      // Post-initialisation callback\n      initialize: function () { /* noop */ }\n    },\n\n    // List of the instancied plugins\n    plugins: {},\n\n    // This options are a merge between `defaults` and the options passed\n    // through the constructor\n    options: {},\n\n    constructor: function (element, options, plugins) {\n      this.options = Darkroom.Utils.extend(options, this.defaults);\n\n      if (typeof element === 'string')\n        element = document.querySelector(element);\n      if (null === element)\n        return;\n\n      var image = new Image();\n      image.onload = function () {\n        // Initialize the DOM/Fabric elements\n        this._initializeDOM(element);\n        this._initializeImage();\n\n        // Then initialize the plugins\n        this._initializePlugins(Darkroom.plugins);\n\n        // Public method to adjust image according to the canvas\n        this.refresh(function () {\n          // Execute a custom callback after initialization\n          this.options.initialize.bind(this).call();\n        }.bind(this));\n\n      }.bind(this)\n\n      //image.crossOrigin = 'anonymous';\n      image.src = element.src;\n    },\n\n    selfDestroy: function () {\n      var container = this.containerElement;\n      var image = new Image();\n      image.onload = function () {\n        container.parentNode.replaceChild(image, container);\n      }\n\n      image.src = this.sourceImage.toDataURL();\n    },\n\n    // Add ability to attach event listener on the core object.\n    // It uses the canvas element to process events.\n    addEventListener: function (eventName, callback) {\n      var el = this.canvas.getElement();\n      if (el.addEventListener) {\n        el.addEventListener(eventName, callback);\n      } else if (el.attachEvent) {\n        el.attachEvent('on' + eventName, callback);\n      }\n    },\n\n    dispatchEvent: function (eventName) {\n      // Use the old way of creating event to be IE compatible\n      // See https://developer.mozilla.org/en-US/docs/Web/Guide/Events/Creating_and_triggering_events\n      var event = document.createEvent('Event');\n      event.initEvent(eventName, true, true);\n\n      this.canvas.getElement().dispatchEvent(event);\n    },\n\n    // Adjust image & canvas dimension according to min/max width/height\n    // and ratio specified in the options.\n    // This method should be called after each image transformation.\n    refresh: function (next) {\n      var clone = new Image();\n      clone.onload = function () {\n        this._replaceCurrentImage(new fabric.Image(clone));\n\n        if (next) next();\n      }.bind(this);\n      clone.src = this.sourceImage.toDataURL();\n    },\n\n    _replaceCurrentImage: function (newImage) {\n      if (this.image) {\n        this.image.remove();\n      }\n\n      this.image = newImage;\n      this.image.selectable = false;\n      this.image.left = this.options.left;\n      this.image.top = this.options.top;\n\n      // Adjust width or height according to specified ratio\n      var viewport = Darkroom.Utils.computeImageViewPort(this.image);\n      var canvasWidth = viewport.width;\n      var canvasHeight = viewport.height;\n\n      // if (null !== this.options.ratio) {\n      //   var canvasRatio = +this.options.ratio;\n      //   var currentRatio = canvasWidth / canvasHeight;\n\n      //   if (currentRatio > canvasRatio) {\n      //     canvasHeight = canvasWidth / canvasRatio;\n      //   } else if (currentRatio < canvasRatio) {\n      //     canvasWidth = canvasHeight * canvasRatio;\n      //   }\n      // }\n\n      // // Then scale the image to fit into dimension limits\n      // var scaleMin = 1;\n      // var scaleMax = 1;\n      // var scaleX = 1;\n      // var scaleY = 1;\n\n      // if (null !== this.options.maxWidth && this.options.maxWidth < canvasWidth) {\n      //   scaleX =  this.options.maxWidth / canvasWidth;\n      // }\n      // if (null !== this.options.maxHeight && this.options.maxHeight < canvasHeight) {\n      //   scaleY =  this.options.maxHeight / canvasHeight;\n      // }\n      // scaleMin = Math.min(scaleX, scaleY);\n\n      // scaleX = 1;\n      // scaleY = 1;\n      // if (null !== this.options.minWidth && this.options.minWidth > canvasWidth) {\n      //   scaleX =  this.options.minWidth / canvasWidth;\n      // }\n      // if (null !== this.options.minHeight && this.options.minHeight > canvasHeight) {\n      //   scaleY =  this.options.minHeight / canvasHeight;\n      // }\n      // scaleMax = Math.max(scaleX, scaleY);\n\n      // var scale = scaleMax * scaleMin; // one should be equals to 1\n\n      // canvasWidth *= scale;\n      // canvasHeight *= scale;\n\n      // Finally place the image in the center of the canvas\n      // this.image.setScaleX(1 * scale);\n      // this.image.setScaleY(1 * scale);\n      this.canvas.add(this.image);\n      this.canvas.setWidth(canvasWidth);\n      this.canvas.setHeight(canvasHeight);\n\n      // Resize the image to fit the canvas\n      if (this.image.oCoords.br.x < this.canvas.width) {\n        this.image.left = this.image.left + Math.ceil(this.canvas.width - this.image.oCoords.br.x);\n      }\n      if (this.image.oCoords.br.y < this.canvas.height) {\n        this.image.top = this.image.top + Math.ceil(this.canvas.height - this.image.oCoords.br.y);\n      }\n      // this.canvas.centerObject(this.image);\n      this.image.setCoords();\n    },\n\n    // Apply the transformation on the current image and save it in the\n    // transformations stack (in order to reconstitute the previous states\n    // of the image).\n    applyTransformation: function (transformation) {\n      this.transformations.push(transformation);\n\n      transformation.applyTransformation(\n        this.sourceCanvas,\n        this.sourceImage,\n        this._postTransformation.bind(this)\n      );\n    },\n\n    _postTransformation: function (newImage) {\n      if (newImage)\n        this.sourceImage = newImage;\n\n      this.refresh(function () {\n        this.dispatchEvent('core:transformation');\n      }.bind(this));\n    },\n\n    // Initialize image from original element plus re-apply every\n    // transformations.\n    reinitializeImage: function () {\n      this.sourceImage.remove();\n      this._initializeImage();\n      this._popTransformation(this.transformations.slice())\n    },\n\n    _popTransformation: function (transformations) {\n      if (0 === transformations.length) {\n        this.dispatchEvent('core:reinitialized');\n        this.refresh();\n        return;\n      }\n\n      var transformation = transformations.shift();\n\n      var next = function (newImage) {\n        if (newImage) this.sourceImage = newImage;\n        this._popTransformation(transformations)\n      };\n\n      transformation.applyTransformation(\n        this.sourceCanvas,\n        this.sourceImage,\n        next.bind(this)\n      );\n    },\n\n    // Create the DOM elements and instanciate the Fabric canvas.\n    // The image element is replaced by a new `div` element.\n    // However the original image is re-injected in order to keep a trace of it.\n    _initializeDOM: function (imageElement) {\n      // Container\n      var mainContainerElement = document.createElement('div');\n      mainContainerElement.className = 'darkroom-container';\n\n      // Toolbar\n      var toolbarElement = document.createElement('div');\n      toolbarElement.className = 'darkroom-toolbar';\n      mainContainerElement.appendChild(toolbarElement);\n\n      // Viewport canvas\n      var canvasContainerElement = document.createElement('div');\n      canvasContainerElement.className = 'darkroom-image-container';\n      var canvasElement = document.createElement('canvas');\n      canvasContainerElement.appendChild(canvasElement);\n      mainContainerElement.appendChild(canvasContainerElement);\n\n      // Source canvas\n      var sourceCanvasContainerElement = document.createElement('div');\n      sourceCanvasContainerElement.className = 'darkroom-source-container';\n      sourceCanvasContainerElement.style.display = 'none';\n      var sourceCanvasElement = document.createElement('canvas');\n      sourceCanvasContainerElement.appendChild(sourceCanvasElement);\n      mainContainerElement.appendChild(sourceCanvasContainerElement);\n\n      // Original image\n      imageElement.parentNode.replaceChild(mainContainerElement, imageElement);\n      imageElement.style.display = 'none';\n      mainContainerElement.appendChild(imageElement);\n\n      Stickyfill.add(toolbarElement);\n      Stickyfill.refreshAll();\n\n      // Instanciate object from elements\n      this.containerElement = mainContainerElement;\n      this.originalImageElement = imageElement;\n\n      this.toolbar = new Darkroom.UI.Toolbar(toolbarElement);\n\n      this.canvas = new fabric.Canvas(canvasElement, {\n        selection: false,\n        backgroundColor: this.options.backgroundColor\n      });\n\n      this.sourceCanvas = new fabric.Canvas(sourceCanvasElement, {\n        selection: false,\n        backgroundColor: this.options.backgroundColor\n      });\n    },\n\n    // Instanciate the Fabric image object.\n    // The image is created as a static element with no control,\n    // then it is add in the Fabric canvas object.\n    _initializeImage: function () {\n      this.sourceImage = new fabric.Image(this.originalImageElement, {\n        // Some options to make the image static\n        selectable: false,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockRotation: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false,\n      });\n\n      this.sourceCanvas.add(this.sourceImage);\n\n      // Adjust width or height according to specified ratio\n      var viewport = Darkroom.Utils.computeImageViewPort(this.sourceImage);\n      var canvasWidth = viewport.width;\n      var canvasHeight = viewport.height;\n\n      this.sourceCanvas.setWidth(canvasWidth);\n      this.sourceCanvas.setHeight(canvasHeight);\n      this.sourceCanvas.centerObject(this.sourceImage);\n      this.sourceImage.setCoords();\n    },\n\n    // Initialize every plugins.\n    // Note that plugins are instanciated in the same order than they\n    // are declared in the parameter object.\n    _initializePlugins: function (plugins) {\n      for (var name in plugins) {\n        var plugin = plugins[name];\n        var options = this.options.plugins[name];\n\n        // Setting false into the plugin options will disable the plugin\n        if (options === false)\n          continue;\n\n        // Avoid any issues with _proto_\n        if (!plugins.hasOwnProperty(name))\n          continue;\n\n        this.plugins[name] = new plugin(this, options);\n      }\n    },\n\n    clearFocus: function (instance) {\n      for (var key in this.plugins) {\n        if (key != instance) {\n          this.plugins[key].clear();\n        }\n      }\n    }\n  }\n\n})();\n","(function() {\n  'use strict';\n\n  Darkroom.Plugin = Plugin;\n\n  // Define a plugin object. This is the (abstract) parent class which\n  // has to be extended for each plugin.\n  function Plugin(darkroom, options) {\n    this.darkroom = darkroom;\n    this.options = Darkroom.Utils.extend(options, this.defaults);\n    this.initialize();\n  }\n\n  Plugin.prototype = {\n    defaults: {},\n    initialize: function() { },\n    clear: function() { }\n  }\n\n  // Inspired by Backbone.js extend capability.\n  Plugin.extend = function(protoProps) {\n    var parent = this;\n    var child;\n\n    if (protoProps && protoProps.hasOwnProperty('constructor')) {\n      child = protoProps.constructor;\n    } else {\n      child = function(){ return parent.apply(this, arguments); };\n    }\n\n    Darkroom.Utils.extend(child, parent);\n\n    var Surrogate = function(){ this.constructor = child; };\n    Surrogate.prototype = parent.prototype;\n    child.prototype = new Surrogate;\n\n    if (protoProps) Darkroom.Utils.extend(child.prototype, protoProps);\n\n    child.__super__ = parent.prototype;\n\n    return child;\n  }\n\n  })();\n","(function() {\n  'use strict';\n\n  Darkroom.Transformation = Transformation;\n\n  function Transformation(options) {\n    this.options = options;\n  }\n\n  Transformation.prototype = {\n    applyTransformation: function(image) { /* no-op */  }\n  }\n\n  // Inspired by Backbone.js extend capability.\n  Transformation.extend = function(protoProps) {\n    var parent = this;\n    var child;\n\n    if (protoProps && protoProps.hasOwnProperty('constructor')) {\n      child = protoProps.constructor;\n    } else {\n      child = function(){ return parent.apply(this, arguments); };\n    }\n\n    Darkroom.Utils.extend(child, parent);\n\n    var Surrogate = function(){ this.constructor = child; };\n    Surrogate.prototype = parent.prototype;\n    child.prototype = new Surrogate;\n\n    if (protoProps) Darkroom.Utils.extend(child.prototype, protoProps);\n\n    child.__super__ = parent.prototype;\n\n    return child;\n  }\n\n  })();\n","(function() {\n  'use strict';\n\n  Darkroom.UI = {\n    Toolbar: Toolbar,\n    ButtonGroup: ButtonGroup,\n    Button: Button,\n  };\n\n  // Toolbar object.\n  function Toolbar(element) {\n    this.element = element;\n  }\n\n  Toolbar.prototype = {\n    createButtonGroup: function(options) {\n      var buttonGroup = document.createElement('div');\n      buttonGroup.className = 'darkroom-button-group';\n      this.element.appendChild(buttonGroup);\n\n      return new ButtonGroup(buttonGroup);\n    }\n  };\n\n  // ButtonGroup object.\n  function ButtonGroup(element) {\n    this.element = element;\n  }\n\n  ButtonGroup.prototype = {\n    createButton: function(options) {\n      var defaults = {\n        image: 'help',\n        type: 'default',\n        group: 'default',\n        hide: false,\n        disabled: false\n      };\n\n      options = Darkroom.Utils.extend(options, defaults);\n\n      var buttonElement = document.createElement('button');\n      buttonElement.type = 'button';\n      buttonElement.className = 'darkroom-button darkroom-button-' + options.type;\n      buttonElement.innerHTML = '<svg class=\"darkroom-icon\"><use xlink:href=\"#' + options.image + '\" /></svg>';\n      this.element.appendChild(buttonElement);\n\n      var button = new Button(buttonElement);\n      button.hide(options.hide);\n      button.disable(options.disabled);\n\n      return button;\n    },\n\n    createInput: function(options) {\n      const defaults = {\n        image: \"help\",\n        type: \"input\",\n        group: \"default\",\n        hide: false,\n        disabled: false,\n        value: \"\"\n      };\n\n      options = Darkroom.Utils.extend(options, defaults);\n\n      const datalistElement = document.createElement(\"datalist\");\n      datalistElement.id = \"rainbow\";\n      const datalistElementOptionred = document.createElement(\"options\");\n      datalistElementOptionred.setAttribute(\"value\", \"#FF0000\");\n      datalistElementOptionred.innerHTML = \"red\";\n      datalistElement.appendChild(datalistElementOptionred);\n      const datalistElementOptionOrange = document.createElement(\"options\");\n      datalistElementOptionOrange.setAttribute(\"value\", \"#FFA500\");\n      datalistElementOptionOrange.innerHTML = \"Orange\";\n      datalistElement.appendChild(datalistElementOptionOrange);\n      const datalistElementOptionYellow = document.createElement(\"options\");\n      datalistElementOptionYellow.setAttribute(\"value\", \"#FFFF00\");\n      datalistElementOptionYellow.innerHTML = \"Yellow\";\n      datalistElement.appendChild(datalistElementOptionYellow);\n      const datalistElementOptionGreen = document.createElement(\"options\");\n      datalistElementOptionGreen.setAttribute(\"value\", \"#008000\");\n      datalistElementOptionGreen.innerHTML = \"Green\";\n      datalistElement.appendChild(datalistElementOptionGreen);\n      const datalistElementOptionBlue = document.createElement(\"options\");\n      datalistElementOptionBlue.setAttribute(\"value\", \"#0000FF\");\n      datalistElementOptionBlue.innerHTML = \"Blue\";\n      datalistElement.appendChild(datalistElementOptionBlue);\n      // this.element.appendChild(datalistElement);\n\n      const inputElement = document.createElement(\"input\");\n      inputElement.setAttribute(\"type\", \"color\");\n      inputElement.setAttribute(\"list\", \"rainbow\");\n      inputElement.value = \"#ff0000\";\n      inputElement.className = \"darkroom-button darkroom-input-\" + options.type;\n      inputElement.innerHTML = '<svg class=\"darkroom-icon\"><use xlink:href=\"#' + options.image + '\" /></svg>';\n      this.element.appendChild(inputElement);\n\n      const input = new Input(inputElement);\n      input.hide(options.hide);\n      input.disable(options.disabled);\n\n      return input;\n    }\n  };\n\n  // Button object.\n  function Button(element) {\n    this.element = element;\n  }\n\n  // Input object.\n  function Input(element) {\n    this.element = element;\n  }\n\n  Button.prototype = {\n    addEventListener: function(eventName, listener) {\n      if (this.element.addEventListener){\n        this.element.addEventListener(eventName, listener);\n      } else if (this.element.attachEvent) {\n        this.element.attachEvent('on' + eventName, listener);\n      }\n    },\n    removeEventListener: function(eventName, listener) {\n      if (this.element.removeEventListener){\n        this.element.removeEventListener(eventName, listener);\n      }\n    },\n    active: function(value) {\n      if (value)\n        this.element.classList.add('darkroom-button-active');\n      else\n        this.element.classList.remove('darkroom-button-active');\n    },\n    hide: function(value) {\n      if (value)\n        this.element.classList.add('darkroom-button-hidden');\n      else\n        this.element.classList.remove('darkroom-button-hidden');\n    },\n    disable: function(value) {\n      this.element.disabled = (value) ? true : false;\n    }\n  };\n\n  Input.prototype = {\n    addEventListener: function(eventName, listener) {\n      if (this.element.addEventListener) {\n        this.element.addEventListener(eventName, listener);\n      } else if (this.element.attachEvent) {\n        this.element.attachEvent(\"on\" + eventName, listener);\n      }\n    },\n    removeEventListener: function(eventName, listener) {\n      if (this.element.removeEventListener) {\n        this.element.removeEventListener(eventName, listener);\n      }\n    },\n    active: function(value) {\n      if (value) { this.element.classList.add(\"darkroom-input-active\"); } else { this.element.classList.remove(\"darkroom-input-active\"); }\n    },\n    hide: function(value) {\n      if (value) { this.element.classList.add(\"darkroom-button-hidden\"); } else { this.element.classList.remove(\"darkroom-button-hidden\"); }\n    },\n    disable: function(value) {\n      this.element.disabled = !!value;\n    }\n  };\n}());\n","(function() {\n  'use strict';\n\n  Darkroom.Utils = {\n    extend: extend,\n    computeImageViewPort: computeImageViewPort,\n  };\n\n\n  // Utility method to easily extend objects.\n  function extend(b, a) {\n    var prop;\n    if (b === undefined) {\n      return a;\n    }\n    for (prop in a) {\n      if (a.hasOwnProperty(prop) && b.hasOwnProperty(prop) === false) {\n        b[prop] = a[prop];\n      }\n    }\n    return b;\n  }\n\n  function computeImageViewPort(image) {\n    return {\n      height: Math.abs(image.getWidth() * (Math.sin(image.getAngle() * Math.PI/180))) + Math.abs(image.getHeight() * (Math.cos(image.getAngle() * Math.PI/180))),\n      width: Math.abs(image.getHeight() * (Math.sin(image.getAngle() * Math.PI/180))) + Math.abs(image.getWidth() * (Math.cos(image.getAngle() * Math.PI/180))),\n    }\n  }\n\n  })();\n",";(function(window, document, Darkroom, fabric) {\n  'use strict';\n\n  Darkroom.plugins['history'] = Darkroom.Plugin.extend({\n    undoTransformations: [],\n\n    initialize: function InitDarkroomHistoryPlugin() {\n      this._initButtons();\n\n      this.darkroom.addEventListener('core:transformation', this._onTranformationApplied.bind(this));\n    },\n\n    undo: function() {\n      if (this.darkroom.transformations.length === 0) {\n        return;\n      }\n\n      var lastTransformation = this.darkroom.transformations.pop();\n      this.undoTransformations.unshift(lastTransformation);\n\n      this.darkroom.reinitializeImage();\n      this._updateButtons();\n    },\n\n    redo: function() {\n      if (this.undoTransformations.length === 0) {\n        return;\n      }\n\n      var cancelTransformation = this.undoTransformations.shift();\n      this.darkroom.transformations.push(cancelTransformation);\n\n      this.darkroom.reinitializeImage();\n      this._updateButtons();\n    },\n\n    _initButtons: function() {\n      var buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.backButton = buttonGroup.createButton({\n        image: 'undo',\n        disabled: true\n      });\n\n      this.forwardButton = buttonGroup.createButton({\n        image: 'redo',\n        disabled: true\n      });\n\n      this.backButton.addEventListener('click', this.undo.bind(this));\n      this.forwardButton.addEventListener('click', this.redo.bind(this));\n\n      return this;\n    },\n\n    _updateButtons: function() {\n      this.backButton.disable((this.darkroom.transformations.length === 0))\n      this.forwardButton.disable((this.undoTransformations.length === 0))\n    },\n\n    _onTranformationApplied: function() {\n      this.undoTransformations = [];\n      this._updateButtons();\n    }\n  });\n})(window, document, Darkroom, fabric);\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Rotation = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const angle = (image.getAngle() + this.options.angle) % 360;\n      image.rotate(angle);\n\n      const height = Math.abs(image.getWidth() * (Math.sin(angle * Math.PI / 180))) + Math.abs(image.getHeight() * (Math.cos(angle * Math.PI / 180)));\n      const width = Math.abs(image.getHeight() * (Math.sin(angle * Math.PI / 180))) + Math.abs(image.getWidth() * (Math.cos(angle * Math.PI / 180)));\n\n      canvas.setWidth(width);\n      canvas.setHeight(height);\n\n      canvas.centerObject(image);\n      image.setCoords();\n      canvas.renderAll();\n\n      next();\n    }\n  });\n\n  Darkroom.plugins.rotate = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      const leftButton = buttonGroup.createButton({\n        image: \"rotate-left\"\n      });\n\n      const rightButton = buttonGroup.createButton({\n        image: \"rotate-right\"\n      });\n\n      leftButton.addEventListener(\"click\", this.rotateLeft.bind(this));\n      rightButton.addEventListener(\"click\", this.rotateRight.bind(this));\n    },\n\n    rotateLeft: function() {\n      this.rotate(-90);\n    },\n\n    rotateRight: function() {\n      this.rotate(90);\n    },\n\n    rotate: function(angle) {\n      this.darkroom.options.left = 0;\n      this.darkroom.options.top = 0;\n      this.darkroom.applyTransformation(new Rotation({\n        angle: angle\n      }));\n    }\n\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Crop = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      // Snapshot the image delimited by the crop zone\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) {\n          return;\n        }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        const width = this.width;\n        const height = this.height;\n\n        // Update canvas size\n        canvas.setWidth(width);\n        canvas.setHeight(height);\n\n        // Add image\n        image.remove();\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      const viewport = Darkroom.Utils.computeImageViewPort(image);\n      const imageWidth = viewport.width;\n      const imageHeight = viewport.height;\n\n      const left = this.options.left * imageWidth;\n      const top = this.options.top * imageHeight;\n      const width = Math.min(this.options.width * imageWidth, imageWidth - left);\n      const height = Math.min(this.options.height * imageHeight, imageHeight - top);\n\n      snapshot.src = canvas.toDataURL({\n        left: left,\n        top: top,\n        width: width,\n        height: height\n      });\n    }\n  });\n\n  const CropZone = fabric.util.createClass(fabric.Rect, {\n    _render: function(ctx) {\n      this.callSuper(\"_render\", ctx);\n\n      const dashWidth = 7;\n\n      // Set original scale\n      const flipX = this.flipX ? -1 : 1;\n      const flipY = this.flipY ? -1 : 1;\n      const scaleX = flipX / this.scaleX;\n      const scaleY = flipY / this.scaleY;\n\n      ctx.scale(scaleX, scaleY);\n\n      // Overlay rendering\n      ctx.fillStyle = \"rgba(0, 0, 0, 0.5)\";\n      this._renderOverlay(ctx);\n\n      // Set dashed borders\n      if (ctx.setLineDash !== undefined) {\n        ctx.setLineDash([dashWidth, dashWidth]);\n      } else if (ctx.mozDash !== undefined) {\n        ctx.mozDash = [dashWidth, dashWidth];\n      }\n\n      // First lines rendering with black\n      ctx.strokeStyle = \"rgba(0, 0, 0, 0.2)\";\n      this._renderBorders(ctx);\n      this._renderGrid(ctx);\n\n      // Re render lines in white\n      ctx.lineDashOffset = dashWidth;\n      ctx.strokeStyle = \"rgba(255, 255, 255, 0.4)\";\n      this._renderBorders(ctx);\n      this._renderGrid(ctx);\n\n      // Reset scale\n      ctx.scale(1 / scaleX, 1 / scaleY);\n    },\n\n    _renderOverlay: function(ctx) {\n      const canvas = ctx.canvas;\n\n      //\n      //    x0    x1        x2      x3\n      // y0 +------------------------+\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      // y1 +------+---------+-------+\n      //    |\\\\\\\\\\\\|         |\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\|    0    |\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\|         |\\\\\\\\\\\\\\|\n      // y2 +------+---------+-------+\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      //    |\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\\|\n      // y3 +------------------------+\n      //\n\n      const x0 = Math.ceil(-this.getWidth() / 2 - this.getLeft());\n      const x1 = Math.ceil(-this.getWidth() / 2);\n      const x2 = Math.ceil(this.getWidth() / 2);\n      const x3 = Math.ceil(this.getWidth() / 2 + (canvas.width - this.getWidth() - this.getLeft()));\n\n      const y0 = Math.ceil(-this.getHeight() / 2 - this.getTop());\n      const y1 = Math.ceil(-this.getHeight() / 2);\n      const y2 = Math.ceil(this.getHeight() / 2);\n      const y3 = Math.ceil(this.getHeight() / 2 + (canvas.height - this.getHeight() - this.getTop()));\n\n      ctx.beginPath();\n\n      // Draw outer rectangle.\n      // Numbers are +/-1 so that overlay edges don't get blurry.\n      ctx.moveTo(x0 - 1, y0 - 1);\n      ctx.lineTo(x3 + 1, y0 - 1);\n      ctx.lineTo(x3 + 1, y3 + 1);\n      ctx.lineTo(x0 - 1, y3 - 1);\n      ctx.lineTo(x0 - 1, y0 - 1);\n      ctx.closePath();\n\n      // Draw inner rectangle.\n      ctx.moveTo(x1, y1);\n      ctx.lineTo(x1, y2);\n      ctx.lineTo(x2, y2);\n      ctx.lineTo(x2, y1);\n      ctx.lineTo(x1, y1);\n\n      ctx.closePath();\n      ctx.fill();\n    },\n\n    _renderBorders: function(ctx) {\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2); // upper left\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2); // upper right\n      ctx.lineTo(this.getWidth() / 2, this.getHeight() / 2); // down right\n      ctx.lineTo(-this.getWidth() / 2, this.getHeight() / 2); // down left\n      ctx.lineTo(-this.getWidth() / 2, -this.getHeight() / 2); // upper left\n      ctx.stroke();\n    },\n\n    _renderGrid: function(ctx) {\n      // Vertical lines\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2 + 1 / 3 * this.getWidth(), -this.getHeight() / 2);\n      ctx.lineTo(-this.getWidth() / 2 + 1 / 3 * this.getWidth(), this.getHeight() / 2);\n      ctx.stroke();\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2 + 2 / 3 * this.getWidth(), -this.getHeight() / 2);\n      ctx.lineTo(-this.getWidth() / 2 + 2 / 3 * this.getWidth(), this.getHeight() / 2);\n      ctx.stroke();\n      // Horizontal lines\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2 + 1 / 3 * this.getHeight());\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2 + 1 / 3 * this.getHeight());\n      ctx.stroke();\n      ctx.beginPath();\n      ctx.moveTo(-this.getWidth() / 2, -this.getHeight() / 2 + 2 / 3 * this.getHeight());\n      ctx.lineTo(this.getWidth() / 2, -this.getHeight() / 2 + 2 / 3 * this.getHeight());\n      ctx.stroke();\n    }\n  });\n\n  Darkroom.plugins.crop = Darkroom.Plugin.extend({\n    // Init point\n    startX: null,\n    startY: null,\n\n    // Keycrop\n    isKeyCroping: false,\n    isKeyLeft: false,\n    isKeyUp: false,\n\n    defaults: {\n      // min crop dimension\n      minHeight: 1,\n      minWidth: 1,\n      // ensure crop ratio\n      ratio: null,\n      // quick crop feature (set a key code to enable it)\n      quickCropKey: false\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.cropButton = buttonGroup.createButton({\n        image: \"crop\"\n      });\n      this.okButton = buttonGroup.createButton({\n        image: \"done\",\n        type: \"success\",\n        hide: true\n      });\n      this.cancelButton = buttonGroup.createButton({\n        image: \"close\",\n        type: \"danger\",\n        hide: true\n      });\n\n      // Buttons click\n      this.cropButton.addEventListener(\"click\", this.toggleCrop.bind(this));\n      this.okButton.addEventListener(\"click\", this.cropCurrentZone.bind(this));\n      this.cancelButton.addEventListener(\"click\", this.releaseFocus.bind(this));\n\n      // Canvas events\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.onMouseUp.bind(this));\n      this.darkroom.canvas.on(\"object:moving\", this.onObjectMoving.bind(this));\n      this.darkroom.canvas.on(\"object:scaling\", this.onObjectScaling.bind(this));\n\n      fabric.util.addListener(fabric.document, \"keydown\", this.onKeyDown.bind(this));\n      fabric.util.addListener(fabric.document, \"keyup\", this.onKeyUp.bind(this));\n\n      this.darkroom.addEventListener(\"core:transformation\", this.releaseFocus.bind(this));\n    },\n\n    clear: function() {\n      if (this.hasFocus()) {\n        this.releaseFocus();\n      }\n    },\n    // Avoid crop zone to go beyond the canvas edges\n    onObjectMoving: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      const currentObject = event.target;\n      if (currentObject !== this.cropZone) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      const x = currentObject.getLeft();\n      const y = currentObject.getTop();\n      const w = currentObject.getWidth();\n      const h = currentObject.getHeight();\n      const maxX = canvas.getWidth() - w;\n      const maxY = canvas.getHeight() - h;\n\n      if (x < 0) {\n        currentObject.set(\"left\", 0);\n      }\n      if (y < 0) {\n        currentObject.set(\"top\", 0);\n      }\n      if (x > maxX) {\n        currentObject.set(\"left\", maxX);\n      }\n      if (y > maxY) {\n        currentObject.set(\"top\", maxY);\n      }\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    // Prevent crop zone from going beyond the canvas edges (like mouseMove)\n    onObjectScaling: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      let preventScaling = false;\n      const currentObject = event.target;\n      if (currentObject !== this.cropZone) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n\n      const minX = currentObject.getLeft();\n      const minY = currentObject.getTop();\n      const maxX = currentObject.getLeft() + currentObject.getWidth();\n      const maxY = currentObject.getTop() + currentObject.getHeight();\n\n      if (this.options.ratio !== null) {\n        if (minX < 0 || maxX > canvas.getWidth() || minY < 0 || maxY > canvas.getHeight()) {\n          preventScaling = true;\n        }\n      }\n\n      if (minX < 0 || maxX > canvas.getWidth() || preventScaling) {\n        const lastScaleX = this.lastScaleX || 1;\n        currentObject.setScaleX(lastScaleX);\n      }\n      if (minX < 0) {\n        currentObject.setLeft(0);\n      }\n\n      if (minY < 0 || maxY > canvas.getHeight() || preventScaling) {\n        const lastScaleY = this.lastScaleY || 1;\n        currentObject.setScaleY(lastScaleY);\n      }\n      if (minY < 0) {\n        currentObject.setTop(0);\n      }\n\n      if (currentObject.getWidth() < this.options.minWidth) {\n        currentObject.scaleToWidth(this.options.minWidth);\n      }\n      if (currentObject.getHeight() < this.options.minHeight) {\n        currentObject.scaleToHeight(this.options.minHeight);\n      }\n\n      this.lastScaleX = currentObject.getScaleX();\n      this.lastScaleY = currentObject.getScaleY();\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    // Init crop zone\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n\n      // recalculate offset, in case canvas was manipulated since last `calcOffset`\n      canvas.calcOffset();\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n      const point = new fabric.Point(x, y);\n\n      // Check if user want to scale or drag the crop zone.\n      const activeObject = canvas.getActiveObject();\n      if (activeObject === this.cropZone || this.cropZone.containsPoint(point)) {\n        return;\n      }\n\n      canvas.discardActiveObject();\n      this.cropZone.setWidth(0);\n      this.cropZone.setHeight(0);\n      this.cropZone.setScaleX(1);\n      this.cropZone.setScaleY(1);\n\n      this.startX = x;\n      this.startY = y;\n    },\n\n    // Extend crop zone\n    onMouseMove: function(event) {\n      // Quick crop feature\n      if (this.isKeyCroping) {\n        return this.onMouseMoveKeyCrop(event);\n      }\n\n      if (null === this.startX || null === this.startY) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n\n      this._renderCropZone(this.startX, this.startY, x, y);\n    },\n\n    onMouseMoveKeyCrop: function(event) {\n      const canvas = this.darkroom.canvas;\n      const zone = this.cropZone;\n\n      const pointer = canvas.getPointer(event.e);\n      const x = pointer.x;\n      const y = pointer.y;\n\n      if (!zone.left || !zone.top) {\n        zone.setTop(y);\n        zone.setLeft(x);\n      }\n\n      this.isKeyLeft = x < zone.left + zone.width / 2;\n      this.isKeyUp = y < zone.top + zone.height / 2;\n\n      this._renderCropZone(\n        Math.min(zone.left, x),\n        Math.min(zone.top, y),\n        Math.max(zone.left + zone.width, x),\n        Math.max(zone.top + zone.height, y)\n      );\n    },\n\n    // Finish crop zone\n    onMouseUp: function(event) {\n      if (null === this.startX || null === this.startY) {\n        return;\n      }\n\n      const canvas = this.darkroom.canvas;\n      this.cropZone.setCoords();\n      canvas.setActiveObject(this.cropZone);\n      canvas.calcOffset();\n\n      this.startX = null;\n      this.startY = null;\n    },\n\n    onKeyDown: function(event) {\n      if (this.options.quickCropKey === false || event.keyCode !== this.options.quickCropKey || this.isKeyCroping) {\n        return;\n      }\n\n      // Active quick crop flow\n      this.isKeyCroping = true;\n      this.darkroom.canvas.discardActiveObject();\n      this.cropZone.setWidth(0);\n      this.cropZone.setHeight(0);\n      this.cropZone.setScaleX(1);\n      this.cropZone.setScaleY(1);\n      this.cropZone.setTop(0);\n      this.cropZone.setLeft(0);\n    },\n\n    onKeyUp: function(event) {\n      if (this.options.quickCropKey === false || event.keyCode !== this.options.quickCropKey || !this.isKeyCroping) {\n        return;\n      }\n\n      // Unactive quick crop flow\n      this.isKeyCroping = false;\n      this.startX = 1;\n      this.startY = 1;\n      this.onMouseUp();\n    },\n\n    selectZone: function(x, y, width, height, forceDimension) {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n      }\n\n      if (!forceDimension) {\n        this._renderCropZone(x, y, x + width, y + height);\n      } else {\n        this.cropZone.set({\n          left: x,\n          top: y,\n          width: width,\n          height: height\n        });\n      }\n\n      const canvas = this.darkroom.canvas;\n      canvas.bringToFront(this.cropZone);\n      this.cropZone.setCoords();\n      canvas.setActiveObject(this.cropZone);\n      canvas.calcOffset();\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    toggleCrop: function() {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n      } else {\n        this.releaseFocus();\n      }\n    },\n\n    cropCurrentZone: function() {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      // Avoid croping empty zone\n      if (this.cropZone.width < 1 && this.cropZone.height < 1) {\n        return;\n      }\n\n      const image = this.darkroom.image;\n\n      // Compute crop zone dimensions\n      let top = this.cropZone.getTop() - image.getTop();\n      let left = this.cropZone.getLeft() - image.getLeft();\n      let width = this.cropZone.getWidth();\n      let height = this.cropZone.getHeight();\n\n      // Adjust dimensions to image only\n      if (top < 0) {\n        height += top;\n        top = 0;\n      }\n\n      if (left < 0) {\n        width += left;\n        left = 0;\n      }\n\n      this.darkroom.options.top = 0;\n      this.darkroom.options.left = 0;\n      this.darkroom.canvas.setZoom(1);\n      this.darkroom.plugins.zoom.updateButtons();\n      // Apply crop transformation.\n      // Make sure to use relative dimension since the crop will be applied\n      // on the source image.\n      this.darkroom.applyTransformation(new Crop({\n        top: top / image.getHeight(),\n        left: left / image.getWidth(),\n        width: width / image.getWidth(),\n        height: height / image.getHeight()\n      }));\n    },\n\n    // Test wether crop zone is set\n    hasFocus: function() {\n      return this.cropZone !== undefined;\n    },\n\n    // Create the crop zone\n    requireFocus: function() {\n      this.darkroom.clearFocus(\"crop\");\n      this.cropZone = new CropZone({\n        fill: \"transparent\",\n        hasBorders: false,\n        originX: \"left\",\n        originY: \"top\",\n        // stroke: '#444',\n        // strokeDashArray: [5, 5],\n        // borderColor: '#444',\n        cornerColor: \"#444\",\n        cornerSize: 8,\n        transparentCorners: false,\n        lockRotation: true,\n        hasRotatingPoint: false\n      });\n\n      if (this.options.ratio !== null) {\n        this.cropZone.set(\"lockUniScaling\", true);\n      }\n\n      this.darkroom.canvas.add(this.cropZone);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n\n      this.cropButton.active(true);\n      this.okButton.hide(false);\n      this.cancelButton.hide(false);\n    },\n\n    // Remove the crop zone\n    releaseFocus: function() {\n      if (undefined === this.cropZone) {\n        return;\n      }\n\n      this.cropZone.remove();\n      this.cropZone = undefined;\n\n      this.cropButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n\n      this.darkroom.canvas.defaultCursor = \"default\";\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    },\n\n    _renderCropZone: function(fromX, fromY, toX, toY) {\n      const canvas = this.darkroom.canvas;\n\n      const isRight = (toX > fromX);\n      let isLeft = !isRight;\n      const isDown = (toY > fromY);\n      let isUp = !isDown;\n\n      const minWidth = Math.min(+this.options.minWidth, canvas.getWidth());\n      const minHeight = Math.min(+this.options.minHeight, canvas.getHeight());\n\n      // Define corner coordinates\n      let leftX = Math.min(fromX, toX);\n      let rightX = Math.max(fromX, toX);\n      let topY = Math.min(fromY, toY);\n      let bottomY = Math.max(fromY, toY);\n\n      // Replace current point into the canvas\n      leftX = Math.max(0, leftX);\n      rightX = Math.min(canvas.getWidth(), rightX);\n      topY = Math.max(0, topY);\n      bottomY = Math.min(canvas.getHeight(), bottomY);\n\n      // Recalibrate coordinates according to given options\n      if (rightX - leftX < minWidth) {\n        if (isRight) {\n          rightX = leftX + minWidth;\n        } else {\n          leftX = rightX - minWidth;\n        }\n      }\n      if (bottomY - topY < minHeight) {\n        if (isDown) {\n          bottomY = topY + minHeight;\n        } else {\n          topY = bottomY - minHeight;\n        }\n      }\n\n      // Truncate truncate according to canvas dimensions\n      if (leftX < 0) {\n        // Translate to the left\n        rightX += Math.abs(leftX);\n        leftX = 0;\n      }\n      if (rightX > canvas.getWidth()) {\n        // Translate to the right\n        leftX -= (rightX - canvas.getWidth());\n        rightX = canvas.getWidth();\n      }\n      if (topY < 0) {\n        // Translate to the bottom\n        bottomY += Math.abs(topY);\n        topY = 0;\n      }\n      if (bottomY > canvas.getHeight()) {\n        // Translate to the right\n        topY -= (bottomY - canvas.getHeight());\n        bottomY = canvas.getHeight();\n      }\n\n      let width = rightX - leftX;\n      let height = bottomY - topY;\n      const currentRatio = width / height;\n\n      if (this.options.ratio && +this.options.ratio !== currentRatio) {\n        const ratio = +this.options.ratio;\n\n        if (this.isKeyCroping) {\n          isLeft = this.isKeyLeft;\n          isUp = this.isKeyUp;\n        }\n\n        if (currentRatio < ratio) {\n          const newWidth = height * ratio;\n          if (isLeft) {\n            leftX -= (newWidth - width);\n          }\n          width = newWidth;\n        } else if (currentRatio > ratio) {\n          const newHeight = height / (ratio * height / width);\n          if (isUp) {\n            topY -= (newHeight - height);\n          }\n          height = newHeight;\n        }\n\n        if (leftX < 0) {\n          leftX = 0;\n          // TODO\n        }\n        if (topY < 0) {\n          topY = 0;\n          // TODO\n        }\n        if (leftX + width > canvas.getWidth()) {\n          const newWidth = canvas.getWidth() - leftX;\n          height = newWidth * height / width;\n          width = newWidth;\n          if (isUp) {\n            topY = fromY - height;\n          }\n        }\n        if (topY + height > canvas.getHeight()) {\n          const newHeight = canvas.getHeight() - topY;\n          width = width * newHeight / height;\n          height = newHeight;\n          if (isLeft) {\n            leftX = fromX - width;\n          }\n        }\n      }\n\n      // Apply coordinates\n      this.cropZone.left = leftX;\n      this.cropZone.top = topY;\n      this.cropZone.width = width;\n      this.cropZone.height = height;\n\n      this.darkroom.canvas.bringToFront(this.cropZone);\n\n      this.darkroom.dispatchEvent(\"crop:update\");\n    }\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  const Pencil = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      // Snapshot the image delimited by the crop zone\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) {\n          return;\n        }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      canvas.add(this.options.pencilZone);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  const PencilZone = fabric.util.createClass(fabric.Object, {\n    type: \"pencilObject\",\n\n    initialize: function(options) {\n      this.options = options || {};\n      this.left = options.left || 0;\n      this.top = options.top || 0;\n      this.width = options.width;\n      this.height = options.height;\n      this.originalWidth = this.width;\n      this.originalHeight = this.height;\n      this.stroke = options.stroke;\n      this.size = options.size;\n      this.path = null;\n      this.cX = -this.getWidth() / 2;\n      this.cY = -this.getHeight() / 2;\n      this.callSuper(\"initialize\", options);\n    },\n\n    _applyScaling: function(w, h) {\n      const scaleX = this.getWidth() / w;\n      const scaleY = this.getHeight() / h;\n      this.set({\n        scaleX: this.scaleX / scaleX,\n        scaleY: this.scaleY / scaleY\n      });\n    },\n\n    _render: function(ctx) {\n      this._draw(ctx);\n    },\n\n    _draw: function(ctx) {\n      if (ctx && this.path && this.path.length > 0) {\n        ctx.lineJoin = \"round\";\n        ctx.lineCap = \"round\";\n        ctx.beginPath();\n        ctx.moveTo(\n          (this.cX + this.path[0].x),\n          (this.cY + this.path[0].y)\n        );\n\n        for (let x = 0; x < this.path.length; x += 1) {\n          const drawX = (this.cX + this.path[x].x);\n          const drawY = (this.cY + this.path[x].y);\n          ctx.lineTo(drawX, drawY);\n        }\n\n\n        ctx.strokeStyle = this.stroke;\n        ctx.lineWidth = this.size;\n        ctx.stroke();\n      }\n    },\n\n    clone: function() {\n      return new PencilZone({\n        stroke: this.stroke,\n        size: this.size,\n        width: this.width,\n        height: this.height,\n        selectable: false,\n        lockRotation: true,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false\n      });\n    }\n  });\n\n  Darkroom.plugins.pencil = Darkroom.Plugin.extend({\n    pencilMode: false,\n    pencilZone: null,\n    mouseDown: false,\n    size: 30,\n    defaults: {\n      stroke: \"black\",\n      size: 30\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.pencilButton = buttonGroup.createButton({\n        image: \"pencil\",\n        tooltip: \"Draw\"\n      });\n\n      this.path = [];\n      this.width = this.darkroom.canvas.width;\n      this.height = this.darkroom.canvas.height;\n      this.size = this.options.size;\n      this.stroke = this.options.stroke;\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.onMouseUp.bind(this));\n      this.pencilButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      if (this.hasFocus()) {\n        this.removeFocus();\n      }\n    },\n\n    hasFocus: function() {\n      return this.pencilZone !== null;\n    },\n\n    onClick: function() {\n      if (!this.hasFocus()) {\n        this.requireFocus();\n        return;\n      }\n      this.removeFocus();\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      this.darkroom.dispatchEvent(\"pencil:begin\");\n      this.mouseDown = true;\n      this.path.push(this.getMousePosition(this.canvas, event));\n      this.renderZone();\n    },\n\n    onMouseUp: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      if (this.mouseDown) {\n        this.darkroom.dispatchEvent(\"pencil:end\");\n        this.mouseDown = false;\n        const mousePos = this.getMousePosition(this.canvas, event);\n        this.path.push(mousePos);\n        this.renderZone();\n        this.applyPencilPath();\n        this.path = [];\n      }\n    },\n\n    onMouseMove: function(event) {\n      if (!this.hasFocus()) {\n        return;\n      }\n\n      if (this.mouseDown) {\n        const mousePos = this.getMousePosition(this.canvas, event);\n        this.path.push(mousePos);\n        this.renderZone();\n      }\n    },\n\n    removeFocus: function() {\n      this.darkroom.dispatchEvent(\"pencil:disabled\");\n      this.pencilButton.active(false);\n      this.pencilMode = false;\n      this.darkroom.canvas.defaultCursor = \"default\";\n      if (this.pencilZone) {\n        this.pencilZone.remove();\n        this.pencilZone = null;\n      }\n    },\n\n    requireFocus: function() {\n      this.darkroom.clearFocus(\"pencil\");\n      this.darkroom.dispatchEvent(\"pencil:enabled\");\n      this.pencilMode = true;\n      this.pencilButton.active(true);\n\n      this.pencilZone = this.newZone();\n\n      this.darkroom.canvas.add(this.pencilZone);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n    },\n\n    getMousePosition: function(canvas, event) {\n      var canvas = this.darkroom.canvas;\n      const rect = canvas.lowerCanvasEl.getBoundingClientRect();\n      const pointer = canvas.getPointer(event.e);\n      return {\n        x: pointer.x,\n        y: pointer.y,\n        cx: event.e.clientX - rect.left,\n        cy: event.e.clientY - rect.top,\n        sx: event.e.screenX,\n        sy: event.e.screenY\n      };\n    },\n\n    newZone: function() {\n      this.path = [];\n      const stroke = this.stroke;\n      const size = this.size;\n      const width = this.darkroom.canvas.getWidth();\n      const height = this.darkroom.canvas.getHeight();\n      const zone = new PencilZone({\n        stroke: stroke,\n        size: size,\n        width: width,\n        height: height,\n        selectable: false,\n        lockRotation: true,\n        evented: false,\n        lockMovementX: true,\n        lockMovementY: true,\n        lockScalingX: true,\n        lockScalingY: true,\n        lockUniScaling: true,\n        hasControls: false,\n        hasBorders: false\n      });\n\n      return zone;\n    },\n\n    applyPencilPath: function() {\n      if (!this.hasFocus()) { return; }\n\n      // Avoid croping empty zone\n      if (this.path === null || this.path.length === 0) { return; }\n\n      // Apply crop transformation.\n      // Make sure to use relative dimension since the crop will be applied\n      // on the source image.\n      const zone = this.pencilZone.clone();\n      zone.path = this.path;\n      zone.left -= this.darkroom.options.left;\n      zone.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Pencil({\n        pencilZone: zone\n      }));\n      this.darkroom.canvas.add(this.pencilZone);\n    },\n\n    renderZone: function() {\n      if (this.path && this.path.length > 0) {\n        this.darkroom.dispatchEvent(\"pencil:update\");\n        this.pencilZone.path = this.path;\n      }\n      this.darkroom.canvas.bringToFront(this.pencilZone);\n    }\n\n  });\n}());\n","(function () {\n  let rect;\n  let isDown;\n  let origX;\n  let origY;\n  let hasFocus = true;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const RectangleTransformation = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        if (this.height < 1 || this.width < 1) { return; }\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      const newRect = this.options.rectangle;\n      const scaleX = this.options.originalWidth / canvas.getWidth();\n      const scaleY = this.options.originalHeight / canvas.getHeight();\n      newRect.set({ scaleX: 1 / scaleX, scaleY: 1 / scaleY });\n      canvas.add(newRect);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  Darkroom.plugins.rectangle = Darkroom.Plugin.extend({\n    rectangleZone: null,\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.rectangleButton = buttonGroup.createButton({\n        image: \"rectangle\",\n        tooltip: \"Draw\"\n      });\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n      this.rectangleButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      this.rectangleButton.active(false);\n      hasFocus = true;\n      isDown = false;\n    },\n\n    onClick: function() {\n      if (!hasFocus) {\n        this.rectangleButton.active(false);\n        this.darkroom.canvas.defaultCursor = \"default\";\n        hasFocus = true;\n        return;\n      }\n      this.darkroom.clearFocus(\"rectangle\");\n      this.rectangleButton.active(true);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n      hasFocus = false;\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      if (hasFocus) return;\n      this.darkroom.dispatchEvent(\"rectangle:begin\");\n      const canvas = this.darkroom.canvas;\n      isDown = true;\n      const pointer = canvas.getPointer(event.e);\n      origX = pointer.x;\n      origY = pointer.y;\n      rect = new fabric.Rect({\n        left: origX,\n        top: origY,\n        originX: \"left\",\n        originY: \"top\",\n        width: pointer.x - origX,\n        height: pointer.y - origY,\n        angle: 0,\n        fill: \"black\",\n        transparentCorners: false\n      });\n      canvas.add(rect);\n    },\n\n    onMouseMove: function(event) {\n      if (isDown) {\n        this.darkroom.dispatchEvent(\"rectangle:update\");\n        const canvas = this.darkroom.canvas;\n        const pointer = canvas.getPointer(event.e);\n\n        if (origX > pointer.x) {\n          rect.set({\n            left: Math.abs(pointer.x)\n          });\n        }\n        if (origY > pointer.y) {\n          rect.set({\n            top: Math.abs(pointer.y)\n          });\n        }\n\n        rect.set({\n          width: Math.abs(origX - pointer.x)\n        });\n        rect.set({\n          height: Math.abs(origY - pointer.y)\n        });\n        canvas.renderAll();\n      }\n    },\n\n    handleMouseUp: function() {\n      if (hasFocus) return;\n      const canvas = this.darkroom.canvas;\n      isDown = false;\n      this.darkroom.dispatchEvent(\"rectangle:end\");\n      rect.left -= this.darkroom.options.left;\n      rect.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new RectangleTransformation({\n        originalWidth: canvas.getWidth(),\n        originalHeight: canvas.getHeight(),\n        rectangle: rect\n      }));\n    }\n  });\n}());\n","(function () {\n  let hasFocus = false;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const Text = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) { return; }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next(imgInstance);\n      };\n\n      canvas.add(this.options.text);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n  Darkroom.plugins.text = Darkroom.Plugin.extend({\n    defaults: {\n      text: \"Sample Text...\"\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.textButton = buttonGroup.createButton({\n        image: \"text\",\n        title: \"Add Text\"\n      });\n      this.okButton = buttonGroup.createButton({\n        image: \"done\",\n        type: \"success\",\n        hide: true\n      });\n      this.cancelButton = buttonGroup.createButton({\n        image: \"close\",\n        type: \"danger\",\n        hide: true\n      });\n      this.colorPicker = buttonGroup.createInput({\n        image: \"color\",\n        title: \"Color Picker\",\n        value: \"#ff0000\",\n        hide: true\n      });\n\n      this.textButton.addEventListener(\"click\", this.addText.bind(this));\n      this.okButton.addEventListener(\"click\", this.saveText.bind(this));\n      this.colorPicker.addEventListener(\"change\", this.setColor.bind(this));\n      this.cancelButton.addEventListener(\"click\", this.releaseFocus.bind(this));\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n\n      this.darkroom.addEventListener(\"core:transformation\", this.releaseFocus.bind(this));\n    },\n\n    clear: function() {\n      this.textButton.active(false);\n      this.darkroom.dispatchEvent(\"text:end\");\n      this.textButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      if (this.newText) {\n        this.newText.remove();\n      }\n      hasFocus = false;\n    },\n\n    addText: function() {\n      if (hasFocus) {\n        this.releaseFocus();\n        return;\n      }\n      hasFocus = true;\n      this.darkroom.clearFocus(\"text\");\n\n      this.textButton.active(true);\n      this.okButton.hide(false);\n      this.cancelButton.hide(false);\n      this.colorPicker.hide(false);\n\n      const factor = this.darkroom.image.scaleX;\n      const text = new fabric.IText(this.options.text, {\n        left: 100 * factor,\n        top: 100 * factor,\n        fill: \"red\",\n        fontFamily: \"Monospace\",\n        fontSize: 25,\n        lockSkewingX: true,\n        lockSkewingY: true,\n        scaleX: factor,\n        scaleY: factor,\n        padding: 10\n      });\n\n      text.on(this.image = this.darkroom.image, {\n        scaling: function() {\n          this.factor = this.image.scaleX;\n\n          const obj = this;\n          const w = obj.width * obj.scaleX / factor;\n          const h = obj.height * obj.scaleY / factor;\n\n          obj.set({\n            width: w,\n            height: h,\n            scaleX: factor,\n            scaleY: factor\n          });\n        }\n      });\n\n      this.darkroom.canvas.add(text);\n      this.newText = text;\n    },\n\n    setColor: function(event) {\n      this.setStyle(this, \"fill\", event.target.value);\n    },\n\n    setStyle: function(object, styleName, value) {\n      const canvas = this.darkroom.canvas;\n      object.newText.fill = value;\n      canvas.renderAll();\n    },\n\n    saveText: function() {\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      this.textButton.active(false);\n      this.newText.hasBorders = false;\n      this.newText.hasControls = false;\n      this.newText.hasRotatingPoint = false;\n      this.newText.selectionColor = \"transparent\";\n      this.newText.abortCursorAnimation();\n      this.newText.left -= this.darkroom.options.left;\n      this.newText.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Text({\n        text: this.newText\n      }));\n      this.colorPicker.element.value = \"#ff0000\";\n    },\n\n    onMouseDown: function() {\n      if (!this.newText) { return; }\n      if (this.newText.active) {\n        this.isDown = true;\n      }\n    },\n\n    onMouseMove: function() {\n      if (!this.isDown) {\n        return;\n      }\n      const canvas = this.darkroom.canvas;\n      canvas.renderAll();\n    },\n\n    handleMouseUp: function() {\n      this.isDown = false;\n    },\n\n    releaseFocus: function() {\n      if (this.newText) { this.newText.remove(); }\n\n      this.textButton.active(false);\n      this.okButton.hide(true);\n      this.cancelButton.hide(true);\n      this.colorPicker.hide(true);\n      this.colorPicker.element.value = \"#ff0000\";\n      hasFocus = false;\n    }\n  });\n}());\n","(function () {\n  let ellip;\n  let isDown = false;\n  let origX;\n  let origY;\n  let hasFocus = true;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  const Circle = Darkroom.Transformation.extend({\n    applyTransformation: function(canvas, image, next) {\n      const snapshot = new Image();\n      snapshot.onload = function () {\n        // Validate image\n        if (this.height < 1 || this.width < 1) { return; }\n\n        const imgInstance = new fabric.Image(this, {\n          // options to make the image static\n          selectable: false,\n          evented: false,\n          lockMovementX: true,\n          lockMovementY: true,\n          lockRotation: true,\n          lockScalingX: true,\n          lockScalingY: true,\n          lockUniScaling: true,\n          hasControls: false,\n          hasBorders: false\n        });\n\n        // Add image\n        canvas.add(imgInstance);\n\n        next:(imgInstance);\n      };\n\n      canvas.add(this.options.circle);\n      canvas.renderAll();\n      snapshot.src = canvas.toDataURL();\n    }\n  });\n\n  Darkroom.plugins.circle = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.circleButton = buttonGroup.createButton({\n        image: \"circle\",\n        tooltip: \"circle\"\n      });\n\n      this.darkroom.canvas.on(\"mouse:down\", this.onMouseDown.bind(this));\n      this.darkroom.canvas.on(\"mouse:move\", this.onMouseMove.bind(this));\n      this.darkroom.canvas.on(\"mouse:up\", this.handleMouseUp.bind(this));\n      this.circleButton.addEventListener(\"click\", this.onClick.bind(this));\n    },\n\n    clear: function() {\n      this.circleButton.active(false);\n      hasFocus = true;\n      isDown = false;\n    },\n\n    onClick: function() {\n      if (!hasFocus) {\n        this.circleButton.active(false);\n        this.darkroom.canvas.defaultCursor = \"default\";\n        hasFocus = true;\n        return;\n      }\n      this.darkroom.clearFocus(\"circle\");\n      this.circleButton.active(true);\n      this.darkroom.canvas.defaultCursor = \"crosshair\";\n      hasFocus = false;\n    },\n\n    onMouseDown: function(event) {\n      event.e.preventDefault();\n      this.isDown = true;\n      if (hasFocus) return;\n      const canvas = this.darkroom.canvas;\n      isDown = true;\n      const pointer = canvas.getPointer(event.e);\n      origX = pointer.x;\n      origY = pointer.y;\n      ellip = new fabric.Ellipse({\n        left: pointer.x,\n        top: pointer.y,\n        stroke: \"black\",\n        fill: \"transparent\",\n        strokeWidth: \"4\",\n        rx: 0,\n        ry: 0\n      });\n      canvas.add(ellip);\n    },\n\n    onMouseMove: function(event) {\n      if (!this.isDown) {\n        return;\n      }\n      if (isDown && ellip) {\n        this.darkroom.dispatchEvent(\"circle:update\");\n        const canvas = this.darkroom.canvas;\n        const pointer = canvas.getPointer(event.e);\n\n        if (origX > pointer.x) {\n          ellip.set({\n            left: Math.abs(pointer.x)\n          });\n        }\n\n        if (origY > pointer.y) {\n          ellip.set({\n            top: Math.abs(pointer.y)\n          });\n        }\n\n        ellip.set({\n          rx: Math.abs(origX - pointer.x) / 2\n        });\n        ellip.set({\n          ry: Math.abs(origY - pointer.y) / 2\n        });\n\n        ellip.setCoords();\n        canvas.renderAll();\n      }\n    },\n\n    handleMouseUp: function() {\n      if (hasFocus) { return; }\n      this.isDown = false;\n      ellip.left -= this.darkroom.options.left;\n      ellip.top -= this.darkroom.options.top;\n      this.darkroom.applyTransformation(new Circle({\n        circle: ellip\n      }));\n    }\n  });\n}());\n","(function () {\n  const maxZoom = 2;\n  let minZoom;\n  const states = {\n    panActive: false,\n    panning: false,\n    realPan: null\n  };\n  let origX;\n  let origY;\n\n  fabric.Image.prototype.strokeWidth = 0;\n  Darkroom.plugins.zoom = Darkroom.Plugin.extend({\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.zoomInButton = buttonGroup.createButton({\n        image: \"zoom-in\",\n        title: \"zoom in\"\n      });\n\n      this.zoomOutButton = buttonGroup.createButton({\n        image: \"zoom-out\",\n        title: \"zoom out\",\n        disabled: true\n      });\n\n      this.pan = buttonGroup.createButton({\n        image: \"hand\",\n        disabled: true\n      });\n\n      this.zoomInButton.addEventListener(\"click\", this.zoomIn.bind(this));\n      this.zoomOutButton.addEventListener(\"click\", this.zoomOut.bind(this));\n      this.pan.addEventListener(\"click\", this.onPan.bind(this));\n      minZoom = this.darkroom.canvas.getZoom();\n    },\n\n    clear: function() {\n      states.panActive = false;\n      this.pan.active(false);\n      states.panning = false;\n      this.darkroom.canvas.defaultCursor = \"default\";\n      this.restoreZoomPan();\n    },\n\n    onPan: function() {\n      if (arguments.length > 0 && states.panActive) {\n        this.clear();\n      } else {\n        this.darkroom.clearFocus(\"zoom\");\n        const canvas = this.darkroom.canvas;\n        canvas.on(\"mouse:down\", this.panEnter.bind(this));\n        canvas.on(\"mouse:up\", this.panExit.bind(this));\n        canvas.on(\"mouse:move\", this._pan.bind(this));\n        canvas.defaultCursor = \"move\";\n        states.panActive = true;\n        this.pan.active(true);\n      }\n    },\n\n\n    panEnter: function(event) {\n      if (states.panActive) {\n        states.panning = true;\n        origX = event.e.screenX || event.e.touches[0].screenX;\n        origY = event.e.screenY || event.e.touches[0].screenY;\n      }\n    },\n\n    _pan: function(event) {\n      event.e.preventDefault();\n      const canvas = this.darkroom.canvas;\n      if (states.panning && event) {\n        if (event.e.touches) {\n          const touches = event.e.touches;\n          for (let i = 0; i < touches.length; i += 1) {\n            const delta = {\n              x: (touches[i].screenX - origX),\n              y: (touches[i].screenY - origY)\n            };\n            origX = touches[i].screenX;\n            origY = touches[i].screenY;\n            this.doPan(delta);\n          }\n        } else {\n          const delta = {\n            x: (event.e.screenX - origX),\n            y: (event.e.screenY - origY)\n          };\n          origX = event.e.screenX;\n          origY = event.e.screenY;\n          this.doPan(delta);\n        }\n        canvas.renderAll();\n      }\n    },\n\n    panExit: function() {\n      if (states.panActive) {\n        const options = this.darkroom.options;\n        const image = this.darkroom.image;\n        states.panning = false;\n        options.left = image.left;\n        options.top = image.top;\n      }\n    },\n\n    doPan: function(delta) {\n      const image = this.darkroom.image;\n      this.delta = this.constrainPan(delta);\n      image.left += delta.x;\n      image.top += delta.y;\n      image.setCoords();\n      states.realPan = {\n        x: image.oCoords.tl.x,\n        y: image.oCoords.tl.y\n      };\n    },\n\n    constrainPan: function(delta) {\n      const canvas = this.darkroom.canvas;\n      const image = this.darkroom.image;\n      const tl = image.oCoords.tl; // top left corner\n      const br = image.oCoords.br; // bottom right corner\n      if (delta.x > 0) {\n        delta.x = ((tl.x + delta.x) > 0) ? 0 : Math.floor(delta.x);\n      } else {\n        delta.x = Math.ceil(delta.x);\n        if ((Math.ceil(br.x) + delta.x) < canvas.width) {\n          delta.x = 0;\n        }\n      }\n      if (delta.y > 0) {\n        delta.y = ((tl.y + delta.y) > 0) ? 0 : Math.floor(delta.y);\n      } else {\n        delta.y = Math.ceil(delta.y);\n        if ((Math.ceil(br.y) + delta.y) < canvas.height) {\n          delta.y = 0;\n        }\n      }\n      return delta;\n    },\n\n    restoreZoomPan: function() {\n      if (states.lastScale != null) {\n        this.darkroom.image.scale(states.lastScale);\n        if (states.realPan != null) {\n          this.doPan(states.realPan);\n        }\n      }\n    },\n\n    zoomIn: function() {\n      const canvas = this.darkroom.canvas;\n      if (canvas.getZoom() <= maxZoom) {\n        canvas.setZoom(canvas.getZoom() * 1.1);\n        canvas.renderAll();\n      }\n      this.onPan();\n      this.updateButtons();\n    },\n\n    zoomOut: function() {\n      const canvas = this.darkroom.canvas;\n      const image = this.darkroom.image;\n      if (canvas.getZoom() > minZoom) {\n        canvas.setZoom(canvas.getZoom() / 1.1);\n        if (this.darkroom.image.oCoords.br.x < this.darkroom.canvas.width) {\n          const newLeft = Math.ceil(this.darkroom.canvas.width - this.darkroom.image.oCoords.br.x);\n          image.left += newLeft;\n          this.darkroom.options.left = this.darkroom.options.left + newLeft;\n        }\n        if (this.darkroom.image.oCoords.br.y < this.darkroom.canvas.height) {\n          const newTop = Math.ceil(this.darkroom.canvas.height - this.darkroom.image.oCoords.br.y);\n          image.top += newTop;\n          this.darkroom.options.top = this.darkroom.options.top + newTop;\n        }\n        image.setCoords();\n        canvas.renderAll();\n        if (canvas.getZoom() >= minZoom) {\n          this.onPan();\n        }\n        this.updateButtons();\n      }\n    },\n\n    updateButtons: function() {\n      const canvas = this.darkroom.canvas;\n      const shouldDisableMinZoom = canvas.getZoom() <= minZoom;\n      const shouldDisableMaxZooom = canvas.getZoom() >= maxZoom;\n      const shouldEnablePan = canvas.getZoom() > minZoom;\n      this.zoomOutButton.disable(shouldDisableMinZoom);\n      this.zoomInButton.disable(shouldDisableMaxZooom);\n      if (shouldDisableMinZoom) {\n        this.darkroom.options.left = 0;\n        this.darkroom.options.top = 0;\n        this.pan.active(false);\n        this.pan.disable(true);\n        this.clear();\n      }\n\n      if (shouldEnablePan) {\n        this.pan.disable(false);\n      }\n    }\n  });\n}());\n","(function () {\n  fabric.Image.prototype.strokeWidth = 0;\n  Darkroom.plugins.save = Darkroom.Plugin.extend({\n\n    defaults: {\n      callback: function() {\n        this.darkroom.selfDestroy();\n      }\n    },\n\n    initialize: function() {\n      const buttonGroup = this.darkroom.toolbar.createButtonGroup();\n\n      this.destroyButton = buttonGroup.createButton({\n        image: \"save\"\n      });\n\n      this.destroyButton.addEventListener(\"click\", this.options.callback.bind(this));\n    }\n  });\n}());\n"]} diff --git a/demo/index.html b/demo/index.html index 8010901..9ac4a34 100644 --- a/demo/index.html +++ b/demo/index.html @@ -132,6 +132,7 @@

Contributing

+