diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js index 1f2fb2cb0..2aba5de27 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.js @@ -4,17 +4,29 @@ var LayoutEditor; var Clipboard = function () { var self = this; - this.clipboardData = {}; - this.setData = function(contentType, data, realClipBoard) { - self.clipboardData[contentType] = data; - }; - this.getData = function (contentType, realClipBoard) { - return self.clipboardData[contentType]; - }; + this._clipboardData = {}; + this._isDisabled = false; + this._wasInvoked = false; - this.disable = function() { - this.disabled = true; + this.setData = function(contentType, data) { + self._clipboardData[contentType] = data; + self._wasInvoked = true; }; + this.getData = function (contentType) { + return self._clipboardData[contentType]; + self._wasInvoked = true; + }; + this.disable = function() { + self._isDisabled = true; + self._wasInvoked = false; + self._clipboardData = {}; + }; + this.isDisabled = function () { + return self._isDisabled; + } + this.wasInvoked = function () { + return self._wasInvoked; + } } LayoutEditor.Clipboard = new Clipboard(); @@ -26,7 +38,9 @@ var LayoutEditor; return { setData: LayoutEditor.Clipboard.setData, getData: LayoutEditor.Clipboard.getData, - disable: LayoutEditor.Clipboard.disable + disable: LayoutEditor.Clipboard.disable, + isDisabled: LayoutEditor.Clipboard.isDisabled, + wasInvoked: LayoutEditor.Clipboard.wasInvoked }; } ]); @@ -48,12 +62,11 @@ angular var resetFocus = false; var element = $scope.element; - if (element.editor.isDragging || element.editor.inlineEditingIsActive) return; - // If the "real" clipboard works, then the pseudo-clipboard will have been disabled. - if (!clipboard.disabled) { + // If native clipboard support exists, the pseudo-clipboard will have been disabled. + if (!clipboard.isDisabled()) { var focusedElement = element.editor.focusedElement; if (!!focusedElement) { // Pseudo clipboard handling for browsers not allowing real clipboard operations. @@ -468,34 +481,46 @@ angular }; $(document).on("cut copy paste", function (e) { - // The real clipboard is supported, so disable the peudo clipboard. - clipboard.disable(); - var focusedElement = $scope.element.focusedElement; - if (!!focusedElement) { - $scope.$apply(function () { - switch (e.type) { - case "copy": - focusedElement.copy(e.originalEvent.clipboardData); - break; - case "cut": - focusedElement.cut(e.originalEvent.clipboardData); - break; - case "paste": - focusedElement.paste(e.originalEvent.clipboardData); - break; - } - }); - - // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost. - window.setTimeout(function () { - $scope.$apply(function () { - if (!!$scope.element.focusedElement) - $scope.element.focusedElement.setIsFocused(); - }); - }, 100); - + // If the pseudo clipboard was already invoked (which happens on the first clipboard + // operation after page load even if native clipboard support exists) then sit this + // one operation out, but make sure whatever is on the pseudo clipboard gets migrated + // to the native clipboard for subsequent operations. + if (clipboard.wasInvoked()) { + e.originalEvent.clipboardData.setData("text/plain", clipboard.getData("text/plain")); + e.originalEvent.clipboardData.setData("text/json", clipboard.getData("text/json")); e.preventDefault(); } + else { + var focusedElement = $scope.element.focusedElement; + if (!!focusedElement) { + $scope.$apply(function () { + switch (e.type) { + case "copy": + focusedElement.copy(e.originalEvent.clipboardData); + break; + case "cut": + focusedElement.cut(e.originalEvent.clipboardData); + break; + case "paste": + focusedElement.paste(e.originalEvent.clipboardData); + break; + } + }); + + // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost. + window.setTimeout(function () { + $scope.$apply(function () { + if (!!$scope.element.focusedElement) + $scope.element.focusedElement.setIsFocused(); + }); + }, 100); + + e.preventDefault(); + } + } + + // Native clipboard support obviously exists, so disable the peudo clipboard from now on. + clipboard.disable(); }); } ], @@ -1011,4 +1036,4 @@ angular }; } ]); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Module.js","Clipboard.js","ScopeConfigurator.js","Editor.js","Canvas.js","Child.js","Column.js","Content.js","Html.js","Grid.js","Row.js","Popup.js","Toolbox.js","ToolboxGroup.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/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;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChUA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/KA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"LayoutEditor.js","sourcesContent":["angular.module(\"LayoutEditor\", [\"ngSanitize\", \"ngResource\", \"ui.sortable\"]);","var LayoutEditor;\n(function(LayoutEditor) {\n\n    var Clipboard = function () {\n        var self = this;\n        this.clipboardData = {};\n        this.setData = function(contentType, data, realClipBoard) {\n            self.clipboardData[contentType] = data;\n        };\n        this.getData = function (contentType, realClipBoard) {\n            return self.clipboardData[contentType];\n        };\n\n        this.disable = function() {\n            this.disabled = true;\n        };\n    }\n\n    LayoutEditor.Clipboard = new Clipboard();\n\n    angular\n        .module(\"LayoutEditor\")\n        .factory(\"clipboard\", [\n            function() {\n                return {\n                    setData: LayoutEditor.Clipboard.setData,\n                    getData: LayoutEditor.Clipboard.getData,\n                    disable: LayoutEditor.Clipboard.disable\n                };\n            }\n        ]);\n})(LayoutEditor || (LayoutEditor = {}));","angular\n    .module(\"LayoutEditor\")\n    .factory(\"scopeConfigurator\", [\"$timeout\", \"clipboard\",\n        function ($timeout, clipboard) {\n            return {\n\n                configureForElement: function ($scope, $element) {\n                \n                    $element.find(\".layout-panel\").click(function (e) {\n                        e.stopPropagation();\n                    });\n\n                    $element.parent().keydown(function (e) {\n                        var handled = false;\n                        var resetFocus = false;\n                        var element = $scope.element;\n                    \n\n                        if (element.editor.isDragging || element.editor.inlineEditingIsActive)\n                            return;\n\n                        // If the \"real\" clipboard works, then the pseudo-clipboard will have been disabled.\n                        if (!clipboard.disabled) {\n                            var focusedElement = element.editor.focusedElement;\n                            if (!!focusedElement) {\n                                // Pseudo clipboard handling for browsers not allowing real clipboard operations.\n                                if (e.ctrlKey) {\n                                    switch (e.which) {\n                                    case 67: // C\n                                        focusedElement.copy(clipboard);\n                                        break;\n                                    case 88: // X\n                                        focusedElement.cut(clipboard);\n                                        break;\n                                    case 86: // V\n                                        focusedElement.paste(clipboard);\n                                        break;\n                                    }\n                                }\n                            }\n                        }\n\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 46) { // Del\n                            $scope.delete(element);\n                            handled = true;\n                        } else if (!e.ctrlKey && !e.shiftKey && !e.altKey && (e.which == 32 || e.which == 27)) { // Space or Esc\n                            $element.find(\".layout-panel-action-properties\").first().click();\n                            handled = true;\n                        }\n\n                        if (element.type == \"Content\") { // This is a content element.\n                            if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 13) { // Enter\n                                $element.find(\".layout-panel-action-edit\").first().click();\n                                handled = true;\n                            }\n                        }\n\n                        if (!!element.children) { // This is a container.\n                            if (!e.ctrlKey && !e.shiftKey && e.altKey && e.which == 40) { // Alt+Down\n                                if (element.children.length > 0)\n                                    element.children[0].setIsFocused();\n                                handled = true;\n                            }\n\n                            if (element.type == \"Column\") { // This is a column.\n                                var connectAdjacent = !e.ctrlKey;\n                                if (e.which == 37) { // Left\n                                    if (e.altKey)\n                                        element.expandLeft(connectAdjacent);\n                                    if (e.shiftKey)\n                                        element.contractRight(connectAdjacent);\n                                    handled = true;\n                                } else if (e.which == 39) { // Right\n                                    if (e.altKey)\n                                        element.contractLeft(connectAdjacent);\n                                    if (e.shiftKey)\n                                        element.expandRight(connectAdjacent);\n                                    handled = true;\n                                }\n                            }\n                        }\n\n                        if (!!element.parent) { // This is a child.\n                            if (e.altKey && e.which == 38) { // Alt+Up\n                                element.parent.setIsFocused();\n                                handled = true;\n                            }\n\n                            if (element.parent.type == \"Row\") { // Parent is a horizontal container.\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Left\n                                    element.parent.moveFocusPrevChild(element);\n                                    handled = true;\n                                }\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Right\n                                    element.parent.moveFocusNextChild(element);\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Ctrl+Left\n                                    element.moveUp();\n                                    resetFocus = true;\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Ctrl+Right\n                                    element.moveDown();\n                                    handled = true;\n                                }\n                            }\n                            else { // Parent is a vertical container.\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Up\n                                    element.parent.moveFocusPrevChild(element);\n                                    handled = true;\n                                }\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Down\n                                    element.parent.moveFocusNextChild(element);\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Ctrl+Up\n                                    element.moveUp();\n                                    resetFocus = true;\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Ctrl+Down\n                                    element.moveDown();\n                                    handled = true;\n                                }\n                            }\n                        }\n\n                        if (handled) {\n                            e.preventDefault();\n                        }\n\n                        e.stopPropagation();\n\n                        $scope.$apply(); // Event is not triggered by Angular directive but raw event handler on element.\n\n                        // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\n                        if (resetFocus) {\n                            window.setTimeout(function () {\n                                $scope.$apply(function () {\n                                    element.editor.focusedElement.setIsFocused();\n                                });\n                            }, 100);\n                        }\n                    });\n\n                    $scope.element.setIsFocusedEventHandlers.push(function () {\n                        $element.parent().focus();\n                    });\n\n                    $scope.delete = function (element) {\n                        element.delete();\n                    }\n                },\n\n                configureForContainer: function ($scope, $element) {\n                    var element = $scope.element;\n\n                    //$scope.isReceiving = false; // True when container is receiving an external element via drag/drop.\n                    $scope.getShowChildrenPlaceholder = function () {\n                        return $scope.element.children.length === 0 && !$scope.element.getIsDropTarget();\n                    };\n\n                    $scope.sortableOptions = {\n                        cursor: \"move\",\n                        delay: 150,\n                        disabled: element.getIsSealed(),\n                        distance: 5,\n                        //handle: element.children.length < 2 ? \".imaginary-class\" : false, // For some reason doesn't get re-evaluated after adding more children.\n                        start: function (e, ui) {\n                            $scope.$apply(function () {\n                                element.setIsDropTarget(true);\n                                element.editor.isDragging = true;\n                            });\n                            // Make the drop target placeholder as high as the item being dragged.\n                            ui.placeholder.height(ui.item.height() - 4);\n                            ui.placeholder.css(\"min-height\", 0);\n                        },\n                        stop: function (e, ui) {\n                            $scope.$apply(function () {\n                                element.editor.isDragging = false;\n                                element.setIsDropTarget(false);\n                            });\n                        },\n                        over: function (e, ui) {\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\n                                if (!!ui.sender[0].dropTargetTimeout) {\n                                    $timeout.cancel(ui.sender[0].dropTargetTimeout);\n                                    ui.sender[0].dropTargetTimeout = null;\n                                }\n                                $timeout(function () {\n                                    if (element.type == \"Row\") {\n                                        // If there was a previous drop target and it was a row, roll back any pending column adds to it.\n                                        var previousDropTarget = element.editor.dropTargetElement;\n                                        if (!!previousDropTarget && previousDropTarget.type == \"Row\")\n                                            previousDropTarget.rollbackAddColumn();\n                                    }\n                                    element.setIsDropTarget(false);\n                                });\n                                ui.sender[0].dropTargetTimeout = $timeout(function () {\n                                    if (element.type == \"Row\") {\n                                        var receivedColumn = ui.item.sortable.model;\n                                        var receivedColumnWidth = Math.floor(12 / (element.children.length + 1));\n                                        receivedColumn.width = receivedColumnWidth;\n                                        receivedColumn.offset = 0;\n                                        element.beginAddColumn(receivedColumnWidth);\n                                        // Make the drop target placeholder the correct width and as high as the highest existing column in the row.\n                                        var maxHeight = _.max(_($element.find(\"> .layout-children > .layout-column:not(.ui-sortable-placeholder)\")).map(function (e) {\n                                            return $(e).height();\n                                        }));\n                                        for (i = 1; i <= 12; i++)\n                                            ui.placeholder.removeClass(\"col-xs-\" + i);\n                                        ui.placeholder.addClass(\"col-xs-\" + receivedColumn.width);\n                                        if (maxHeight > 0) {\n                                            ui.placeholder.height(maxHeight);\n                                            ui.placeholder.css(\"min-height\", 0);\n                                        }\n                                        else {\n                                            ui.placeholder.height(0);\n                                            ui.placeholder.css(\"min-height\", \"\");\n                                        }\n                                    }\n                                    element.setIsDropTarget(true);\n                                }, 150);\n                            }\n                        },\n                        receive: function (e, ui) {\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\n                                $scope.$apply(function () {\n                                    var receivedElement = ui.item.sortable.model;\n                                    if (!!receivedElement) {\n                                        if (element.type == \"Row\")\n                                            element.commitAddColumn();\n                                        // Should ideally call LayoutEditor.Container.addChild() instead, but since this handler\n                                        // is run *before* the ui-sortable directive's handler, if we try to add the child to the\n                                        // array that handler will get an exception when trying to do the same.\n                                        // Because of this, we need to invoke \"setParent\" so that specific container types can perform element speficic initialization.\n                                        receivedElement.setEditor(element.editor);\n                                        receivedElement.setParent(element);\n                                        if (!!receivedElement.hasEditor) {\n                                            $scope.$root.editElement(receivedElement).then(function (args) {\n                                                if (!args.cancel) {\n                                                    receivedElement.data = args.element.data;\n\n                                                    if (receivedElement.setHtml)\n                                                        receivedElement.setHtml(args.element.html);\n                                                }\n                                                $timeout(function () {\n                                                    if (!!args.cancel)\n                                                        receivedElement.delete();\n                                                    else\n                                                        receivedElement.setIsFocused();\n                                                    //$scope.isReceiving = false;\n                                                    element.setIsDropTarget(false);\n\n                                                });\n                                                return;\n                                            });\n                                        }\n                                    }\n                                    $timeout(function () {\n                                        //$scope.isReceiving = false;\n                                        element.setIsDropTarget(false);\n                                        if (!!receivedElement)\n                                            receivedElement.setIsFocused();\n                                    });\n                                });\n                            }\n                        }\n                    };\n\n                    $scope.click = function (child, e) {\n                        if (!child.editor.isDragging)\n                            child.setIsFocused();\n                        e.stopPropagation();\n                    };\n\n                    $scope.getClasses = function (child) {\n                        var result = [\"layout-element\"];\n\n                        if (!!child.children) {\n                            result.push(\"layout-container\");\n                            if (child.getIsSealed())\n                                result.push(\"layout-container-sealed\");\n                        }\n\n                        result.push(\"layout-\" + child.type.toLowerCase());\n\n                        if (!!child.dropTargetClass)\n                            result.push(child.dropTargetClass);\n\n                        // TODO: Move these to either the Column directive or the Column model class.\n                        if (child.type == \"Row\") {\n                            result.push(\"row\");\n                            if (!child.canAddColumn())\n                                result.push(\"layout-row-full\");\n                        }\n                        if (child.type == \"Column\") {\n                            result.push(\"col-xs-\" + child.width);\n                            result.push(\"col-xs-offset-\" + child.offset);\n                        }\n                        if (child.type == \"Content\")\n                            result.push(\"layout-content-\" + child.contentTypeClass);\n\n                        if (child.getIsActive())\n                            result.push(\"layout-element-active\");\n                        if (child.getIsFocused())\n                            result.push(\"layout-element-focused\");\n                        if (child.getIsSelected())\n                            result.push(\"layout-element-selected\");\n                        if (child.getIsDropTarget())\n                            result.push(\"layout-element-droptarget\");\n                        if (child.isTemplated)\n                            result.push(\"layout-element-templated\");\n\n                        return result;\n                    };\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutEditor\", [\"environment\",\n        function (environment) {\n            return {\n                restrict: \"E\",\n                scope: {},\n                controller: [\"$scope\", \"$element\", \"$attrs\", \"$compile\", \"clipboard\",\n                    function ($scope, $element, $attrs, $compile, clipboard) {\n                        if (!!$attrs.model)\n                            $scope.element = eval($attrs.model);\n                        else\n                            throw new Error(\"The 'model' attribute must evaluate to a LayoutEditor.Editor object.\");\n\n                        $scope.click = function (canvas, e) {\n                            if (!canvas.editor.isDragging)\n                                canvas.setIsFocused();\n                            e.stopPropagation();\n                        };\n\n                        $scope.getClasses = function (canvas) {\n                            var result = [\"layout-element\", \"layout-container\", \"layout-canvas\"];\n\n                            if (canvas.getIsActive())\n                                result.push(\"layout-element-active\");\n                            if (canvas.getIsFocused())\n                                result.push(\"layout-element-focused\");\n                            if (canvas.getIsSelected())\n                                result.push(\"layout-element-selected\");\n                            if (canvas.getIsDropTarget())\n                                result.push(\"layout-element-droptarget\");\n                            if (canvas.isTemplated)\n                                result.push(\"layout-element-templated\");\n\n                            return result;\n                        };\n\n                        // An unfortunate side-effect of the next hack on line 54 is that the created elements aren't added to the DOM yet, so we can't use it to get to the parent \".layout-desiger\" element.\n                        // Work around: access that element directly (which efectively turns multiple layout editors on a single page impossible). \n                        // //var layoutDesignerHost = $element.closest(\".layout-designer\").data(\"layout-designer-host\");\n                        var layoutDesignerHost = $(\".layout-designer\").data(\"layout-designer-host\");\n\n                        $scope.$root.layoutDesignerHost = layoutDesignerHost;\n\n                        layoutDesignerHost.element.on(\"replacecanvas\", function (e, args) {\n                            var editor = $scope.element;\n                            var canvasData = {\n                                data: args.canvas.data,\n                                htmlId: args.canvas.htmlId,\n                                htmlClass: args.canvas.htmlClass,\n                                htmlStyle: args.canvas.htmlStyle,\n                                isTemplated: args.canvas.isTemplated,\n                                children: args.canvas.children\n                            };\n\n                            // HACK: Instead of simply updating the $scope.element with a new instance, we need to replace the entire orc-layout-editor markup\n                            // in order for angular to rebind starting with the Canvas element. Otherwise, for some reason, it will rebind starting with the first child of Canvas.\n                            // You can see this happening when setting a breakpoint in ScopeConfigurator where containers are initialized with drag & drop: on page load, the first element\n                            // is a Canvas (good), but after having selected another template, the first element is (typically) a Grid (bad).\n                            // Simply recompiling the orc-layout-editor directive will cause the entire thing to be generated, which works just fine as well (even though not is nice as simply leveraging model binding).\n                            layoutDesignerHost.editor = window.layoutEditor = new LayoutEditor.Editor(editor.config, canvasData);\n                            var template = \"<orc-layout-editor\" + \" model='window.layoutEditor' />\";\n                            var html = $compile(template)($scope);\n                            $(\".layout-editor-holder\").html(html);\n                        });\n\n                        $scope.$root.editElement = function (element) {\n                            var host = $scope.$root.layoutDesignerHost;\n                            return host.editElement(element);\n                        };\n\n                        $scope.$root.addElement = function (contentType) {\n                            var host = $scope.$root.layoutDesignerHost;\n                            return host.addElement(contentType);\n                        };\n\n                        $scope.toggleInlineEditing = function () {\n                            if (!$scope.element.inlineEditingIsActive) {\n                                $scope.element.inlineEditingIsActive = true;\n                                $element.find(\".layout-toolbar-container\").show();\n                                var selector = \"#layout-editor-\" + $scope.$id + \" .layout-html .layout-content-markup[data-templated=false]\";\n                                var firstContentEditorId = $(selector).first().attr(\"id\");\n                                tinymce.init({\n                                    selector: selector,\n                                    theme: \"modern\",\n                                    schema: \"html5\",\n                                    plugins: [\n                                        \"advlist autolink lists link image charmap print preview hr anchor pagebreak\",\n                                        \"searchreplace wordcount visualblocks visualchars code fullscreen\",\n                                        \"insertdatetime media nonbreaking table contextmenu directionality\",\n                                        \"emoticons template paste textcolor colorpicker textpattern\",\n                                        \"fullscreen autoresize\"\n                                    ],\n                                    toolbar: \"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close\",\n                                    convert_urls: false,\n                                    valid_elements: \"*[*]\",\n                                    // Shouldn't be needed due to the valid_elements setting, but TinyMCE would strip script.src without it.\n                                    extended_valid_elements: \"script[type|defer|src|language]\",\n                                    statusbar: false,\n                                    skin: \"orchardlightgray\",\n                                    inline: true,\n                                    fixed_toolbar_container: \"#layout-editor-\" + $scope.$id + \" .layout-toolbar-container\",\n                                    init_instance_callback: function (editor) {\n                                        if (editor.id == firstContentEditorId)\n                                            tinymce.execCommand(\"mceFocus\", false, editor.id);\n                                    }\n                                });\n                            }\n                            else {\n                                tinymce.remove(\"#layout-editor-\" + $scope.$id + \" .layout-content-markup\");\n                                $element.find(\".layout-toolbar-container\").hide();\n                                $scope.element.inlineEditingIsActive = false;\n                            }\n                        };\n\n                        $(document).on(\"cut copy paste\", function (e) {\n                            // The real clipboard is supported, so disable the peudo clipboard.\n                            clipboard.disable();\n                            var focusedElement = $scope.element.focusedElement;\n                            if (!!focusedElement) {\n                                $scope.$apply(function () {\n                                    switch (e.type) {\n                                        case \"copy\":\n                                            focusedElement.copy(e.originalEvent.clipboardData);\n                                            break;\n                                        case \"cut\":\n                                            focusedElement.cut(e.originalEvent.clipboardData);\n                                            break;\n                                        case \"paste\":\n                                            focusedElement.paste(e.originalEvent.clipboardData);\n                                            break;\n                                    }\n                                });\n\n                                // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\n                                window.setTimeout(function () {\n                                    $scope.$apply(function () {\n                                        if (!!$scope.element.focusedElement)\n                                            $scope.element.focusedElement.setIsFocused();\n                                    });\n                                }, 100);\n\n                                e.preventDefault();\n                            }\n                        });\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Editor\"),\n                replace: true,\n                link: function (scope, element) {\n                    // No clicks should propagate from the TinyMCE toolbars.\n                    element.find(\".layout-toolbar-container\").click(function (e) {\n                        e.stopPropagation();\n                    });\n                    // Intercept mousedown on editor while in inline editing mode to \n                    // prevent current editor from losing focus.\n                    element.mousedown(function (e) {\n                        if (scope.element.inlineEditingIsActive) {\n                            e.preventDefault();\n                            e.stopPropagation();\n                        }\n                    })\n                    // Unfocus and unselect everything on click outside of canvas.\n                    $(window).click(function (e) {\n                        // Except when in inline editing mode.\n                        if (!scope.element.inlineEditingIsActive) {\n                            scope.$apply(function () {\n                                scope.element.activeElement = null;\n                                scope.element.focusedElement = null;\n                            });\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutCanvas\", [\"scopeConfigurator\", \"environment\",\n        function (scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\", \"$attrs\",\n                    function ($scope, $element, $attrs) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Canvas\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutChild\", [\"$compile\",\n        function ($compile) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                link: function (scope, element) {\n                    var template = \"<orc-layout-\" + scope.element.type.toLowerCase() + \" element='element' />\";\n                    var html = $compile(template)(scope);\n                    $(element).replaceWith(html);\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutColumn\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Column\"),\n                replace: true,\n                link: function (scope, element, attrs) {\n                    element.find(\".layout-column-resize-bar\").draggable({\n                        axis: \"x\",\n                        helper: \"clone\",\n                        revert: true,\n                        start: function (e, ui) {\n                            scope.$apply(function () {\n                                scope.element.editor.isResizing = true;\n                            });\n                        },\n                        drag: function (e, ui) {\n                            var columnElement = element.parent();\n                            var columnSize = columnElement.width() / scope.element.width;\n                            var connectAdjacent = !e.ctrlKey;\n                            if ($(e.target).hasClass(\"layout-column-resize-bar-left\")) {\n                                var delta = ui.offset.left - columnElement.offset().left;\n                                if (delta < -columnSize && scope.element.canExpandLeft(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.expandLeft(connectAdjacent);\n                                    });\n                                }\n                                else if (delta > columnSize && scope.element.canContractLeft(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.contractLeft(connectAdjacent);\n                                    });\n                                }\n                            }\n                            else if ($(e.target).hasClass(\"layout-column-resize-bar-right\")) {\n                                var delta = ui.offset.left - columnElement.width() - columnElement.offset().left;\n                                if (delta > columnSize && scope.element.canExpandRight(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.expandRight(connectAdjacent);\n                                    });\n                                }\n                                else if (delta < -columnSize && scope.element.canContractRight(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.contractRight(connectAdjacent);\n                                    });\n                                }\n                            }\n\n                        },\n                        stop: function (e, ui) {\n                            scope.$apply(function () {\n                              scope.element.editor.isResizing = false;\n                            });\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutContent\", [\"$sce\", \"scopeConfigurator\", \"environment\",\n        function ($sce, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        $scope.edit = function () {\n                            $scope.$root.editElement($scope.element).then(function (args) {\n                                $scope.$apply(function () {\n                                    if (args.cancel)\n                                        return;\n\n                                    $scope.element.data = args.element.data;\n                                    $scope.element.setHtml(args.element.html);\n                                });\n                            });\n                        };\n\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\n                        $scope.element.setHtml = function (html) {\n                            $scope.element.html = html;\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\n                        };\n\n                        $scope.element.setHtml($scope.element.html);\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Content\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutHtml\", [\"$sce\", \"scopeConfigurator\", \"environment\",\n        function ($sce, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        $scope.edit = function () {\n                            $scope.$root.editElement($scope.element).then(function (args) {\n                                $scope.$apply(function () {\n                                    if (args.cancel)\n                                        return;\n\n                                    $scope.element.data = args.element.data;\n                                    $scope.element.setHtml(args.element.html);\n                                });\n                            });\n                        };\n                        $scope.updateContent = function (e) {\n                            $scope.element.setHtml(e.target.innerHTML);\n                        };\n\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\n                        $scope.element.setHtml = function (html) {\n                            $scope.element.html = html;\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\n                        };\n\n                        $scope.element.setHtml($scope.element.html);\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Html\"),\n                replace: true,\n                link: function (scope, element) {\n                    // Mouse down events must not be intercepted by drag and drop while inline editing is active,\n                    // otherwise clicks in inline editors will have no effect.\n                    element.find(\".layout-content-markup\").mousedown(function (e) {\n                        if (scope.element.editor.inlineEditingIsActive) {\n                            e.stopPropagation();\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutGrid\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Grid\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutRow\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"x\";\n                        $scope.sortableOptions[\"ui-floating\"] = true;\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Row\"),\n                replace: true\n            };\n        }\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutPopup\", [\r\n        function () {\r\n            return {\r\n                restrict: \"A\",\r\n                link: function (scope, element, attrs) {\r\n                    var popup = $(element);\r\n                    var trigger = popup.closest(\".layout-popup-trigger\");\r\n                    var parentElement = popup.closest(\".layout-element\");\r\n                    trigger.click(function () {\r\n                        popup.toggle();\r\n                        if (popup.is(\":visible\")) {\r\n                            popup.position({\r\n                                my: attrs.orcLayoutPopupMy || \"left top\",\r\n                                at: attrs.orcLayoutPopupAt || \"left bottom+4px\",\r\n                                of: trigger\r\n                            });\r\n                            popup.find(\"input\").first().focus();\r\n                        }\r\n                    });\r\n                    popup.click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    parentElement.click(function (e) {\r\n                        popup.hide();\r\n                    });\r\n                    popup.keydown(function (e) {\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 27) // Esc\r\n                            popup.hide();\r\n                        e.stopPropagation();\r\n                    });\r\n                    popup.on(\"cut copy paste\", function (e) {\n                        // Allow clipboard operations in popup without invoking clipboard event handlers on parent element.\n                        e.stopPropagation();\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolbox\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n\r\n                        $scope.resetElements = function () {\r\n\r\n                            $scope.gridElements = [\r\n                                LayoutEditor.Grid.from({\r\n                                    toolboxIcon: \"\\uf00a\",\r\n                                    toolboxLabel: \"Grid\",\r\n                                    toolboxDescription: \"Empty grid.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.rowElements = [\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (1 column)\",\r\n                                    toolboxDescription: \"Row with 1 column.\",\r\n                                    children: LayoutEditor.Column.times(1)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (2 columns)\",\r\n                                    toolboxDescription: \"Row with 2 columns.\",\r\n                                    children: LayoutEditor.Column.times(2)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (3 columns)\",\r\n                                    toolboxDescription: \"Row with 3 columns.\",\r\n                                    children: LayoutEditor.Column.times(3)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (4 columns)\",\r\n                                    toolboxDescription: \"Row with 4 columns.\",\r\n                                    children: LayoutEditor.Column.times(4)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (6 columns)\",\r\n                                    toolboxDescription: \"Row with 6 columns.\",\r\n                                    children: LayoutEditor.Column.times(6)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (12 columns)\",\r\n                                    toolboxDescription: \"Row with 12 columns.\",\r\n                                    children: LayoutEditor.Column.times(12)\r\n                                }), LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (empty)\",\r\n                                    toolboxDescription: \"Empty row.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.columnElements = [\r\n                                LayoutEditor.Column.from({\r\n                                    toolboxIcon: \"\\uf0db\",\r\n                                    toolboxLabel: \"Column\",\r\n                                    toolboxDescription: \"Empty column.\",\r\n                                    width: 1,\r\n                                    offset: 0,\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.contentElementCategories = _($scope.element.config.categories).map(function (category) {\r\n                                return {\r\n                                    name: category.name,\r\n                                    elements: _(category.contentTypes).map(function (contentType) {\r\n                                        var type = contentType.type;\r\n                                        var factory = LayoutEditor.factories[type] || LayoutEditor.factories[\"Content\"];\r\n                                        var item = {\r\n                                            isTemplated: false,\r\n                                            contentType: contentType.id,\r\n                                            contentTypeLabel: contentType.label,\r\n                                            contentTypeClass: contentType.typeClass,\r\n                                            data: null,\r\n                                            hasEditor: contentType.hasEditor,\r\n                                            html: contentType.html\r\n                                        };\r\n                                        var element = factory(item);\r\n                                        element.toolboxIcon = contentType.icon || \"\\uf1c9\";\r\n                                        element.toolboxLabel = contentType.label;\r\n                                        element.toolboxDescription = contentType.description;\r\n                                        return element;\r\n                                    })\r\n                                };\r\n                            });\r\n\r\n                        };\r\n\r\n                        $scope.resetElements();\r\n\r\n                        $scope.getSortableOptions = function (type) {\r\n                            var editorId = $element.closest(\".layout-editor\").attr(\"id\");\r\n                            var parentClasses;\r\n                            var placeholderClasses;\r\n                            var floating = false;\r\n\r\n                            switch (type) {\r\n                                case \"Grid\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Row\":\r\n                                    parentClasses = [\".layout-grid\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-row row ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Column\":\r\n                                    parentClasses = [\".layout-row:not(.layout-row-full)\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-column ui-sortable-placeholder\";\r\n                                    floating = true; // To ensure a smooth horizontal-list reordering. https://github.com/angular-ui/ui-sortable#floating\r\n                                    break;\r\n                                case \"Content\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-content ui-sortable-placeholder\";\r\n                                    break;\r\n                            }\r\n\r\n                            return {\r\n                                cursor: \"move\",\r\n                                connectWith: _(parentClasses).map(function (e) { return \"#\" + editorId + \" \" + e + \":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children\"; }).join(\", \"),\r\n                                placeholder: placeholderClasses,\r\n                                \"ui-floating\": floating,\r\n                                create: function (e, ui) {\r\n                                    e.target.isToolbox = true; // Will indicate to connected sortables that dropped items were sent from toolbox.\r\n                                },\r\n                                start: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = true;\r\n                                    });\r\n                                },\r\n                                stop: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = false;\r\n                                        $scope.resetElements();\r\n                                    });\r\n                                },\r\n                                over: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.canvas.setIsDropTarget(false);\r\n                                    });\r\n                                },\r\n                            }\r\n                        };\r\n\r\n                        var layoutIsCollapsedCookieName = \"layoutToolboxCategory_Layout_IsCollapsed\";\r\n                        $scope.layoutIsCollapsed = $.cookie(layoutIsCollapsedCookieName) === \"true\";\r\n\r\n                        $scope.toggleLayoutIsCollapsed = function (e) {\r\n                            $scope.layoutIsCollapsed = !$scope.layoutIsCollapsed;\r\n                            $.cookie(layoutIsCollapsedCookieName, $scope.layoutIsCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Toolbox\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    var toolbox = element.find(\".layout-toolbox\");\r\n                    $(window).on(\"resize scroll\", function (e) {\r\n                        var canvas = element.parent().find(\".layout-canvas\");\r\n                        // If the canvas is taller than the toolbox, make the toolbox sticky-positioned within the editor\r\n                        // to help the user avoid excessive vertical scrolling.\r\n                        var canvasIsTaller = !!canvas && canvas.height() > toolbox.height();\r\n                        var windowPos = $(window).scrollTop();\r\n                        if (canvasIsTaller && windowPos > element.offset().top + element.height() - toolbox.height()) {\r\n                            toolbox.addClass(\"sticky-bottom\");\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                        }\r\n                        else if (canvasIsTaller && windowPos > element.offset().top) {\r\n                            toolbox.addClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                        else {\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutToolboxGroup\", [\"$compile\", \"environment\",\n        function ($compile, environment) {\n            return {\n                restrict: \"E\",\n                scope: { category: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        var isCollapsedCookieName = \"layoutToolboxCategory_\" + $scope.category.name + \"_IsCollapsed\";\n                        $scope.isCollapsed = $.cookie(isCollapsedCookieName) === \"true\";\n                        $scope.toggleIsCollapsed = function (e) {\n                            $scope.isCollapsed = !$scope.isCollapsed;\n                            $.cookie(isCollapsedCookieName, $scope.isCollapsed, { expires: 365 }); // Remember collapsed state for a year.\n                            e.preventDefault();\n                            e.stopPropagation();\n                        };\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"ToolboxGroup\"),\n                replace: true\n            };\n        }\n    ]);"],"sourceRoot":"/source/"} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Module.js","Clipboard.js","ScopeConfigurator.js","Editor.js","Canvas.js","Child.js","Column.js","Content.js","Html.js","Grid.js","Row.js","Popup.js","Toolbox.js","ToolboxGroup.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/TA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC3LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACdA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC/CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AClBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACnBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjMA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"LayoutEditor.js","sourcesContent":["angular.module(\"LayoutEditor\", [\"ngSanitize\", \"ngResource\", \"ui.sortable\"]);","var LayoutEditor;\n(function(LayoutEditor) {\n\n    var Clipboard = function () {\n        var self = this;\n        this._clipboardData = {};\n        this._isDisabled = false;\n        this._wasInvoked = false;\n\n        this.setData = function(contentType, data) {\n            self._clipboardData[contentType] = data;\n            self._wasInvoked = true;\n        };\n        this.getData = function (contentType) {\n            return self._clipboardData[contentType];\n            self._wasInvoked = true;\n        };\n        this.disable = function() {\n            self._isDisabled = true;\n            self._wasInvoked = false;\n            self._clipboardData = {};\n        };\n        this.isDisabled = function () {\r\n            return self._isDisabled;\r\n        }\n        this.wasInvoked = function () {\r\n            return self._wasInvoked;\r\n        }\n    }\n\n    LayoutEditor.Clipboard = new Clipboard();\n\n    angular\n        .module(\"LayoutEditor\")\n        .factory(\"clipboard\", [\n            function() {\n                return {\n                    setData: LayoutEditor.Clipboard.setData,\n                    getData: LayoutEditor.Clipboard.getData,\n                    disable: LayoutEditor.Clipboard.disable,\n                    isDisabled: LayoutEditor.Clipboard.isDisabled,\n                    wasInvoked: LayoutEditor.Clipboard.wasInvoked\n                };\n            }\n        ]);\n})(LayoutEditor || (LayoutEditor = {}));","angular\n    .module(\"LayoutEditor\")\n    .factory(\"scopeConfigurator\", [\"$timeout\", \"clipboard\",\n        function ($timeout, clipboard) {\n            return {\n\n                configureForElement: function ($scope, $element) {\n                \n                    $element.find(\".layout-panel\").click(function (e) {\n                        e.stopPropagation();\n                    });\n\n                    $element.parent().keydown(function (e) {\n                        var handled = false;\n                        var resetFocus = false;\n                        var element = $scope.element;\n                    \n                        if (element.editor.isDragging || element.editor.inlineEditingIsActive)\n                            return;\n\n                        // If native clipboard support exists, the pseudo-clipboard will have been disabled.\n                        if (!clipboard.isDisabled()) {\n                            var focusedElement = element.editor.focusedElement;\n                            if (!!focusedElement) {\n                                // Pseudo clipboard handling for browsers not allowing real clipboard operations.\n                                if (e.ctrlKey) {\n                                    switch (e.which) {\n                                    case 67: // C\n                                        focusedElement.copy(clipboard);\n                                        break;\n                                    case 88: // X\n                                        focusedElement.cut(clipboard);\n                                        break;\n                                    case 86: // V\n                                        focusedElement.paste(clipboard);\n                                        break;\n                                    }\n                                }\n                            }\n                        }\n\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 46) { // Del\n                            $scope.delete(element);\n                            handled = true;\n                        } else if (!e.ctrlKey && !e.shiftKey && !e.altKey && (e.which == 32 || e.which == 27)) { // Space or Esc\n                            $element.find(\".layout-panel-action-properties\").first().click();\n                            handled = true;\n                        }\n\n                        if (element.type == \"Content\") { // This is a content element.\n                            if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 13) { // Enter\n                                $element.find(\".layout-panel-action-edit\").first().click();\n                                handled = true;\n                            }\n                        }\n\n                        if (!!element.children) { // This is a container.\n                            if (!e.ctrlKey && !e.shiftKey && e.altKey && e.which == 40) { // Alt+Down\n                                if (element.children.length > 0)\n                                    element.children[0].setIsFocused();\n                                handled = true;\n                            }\n\n                            if (element.type == \"Column\") { // This is a column.\n                                var connectAdjacent = !e.ctrlKey;\n                                if (e.which == 37) { // Left\n                                    if (e.altKey)\n                                        element.expandLeft(connectAdjacent);\n                                    if (e.shiftKey)\n                                        element.contractRight(connectAdjacent);\n                                    handled = true;\n                                } else if (e.which == 39) { // Right\n                                    if (e.altKey)\n                                        element.contractLeft(connectAdjacent);\n                                    if (e.shiftKey)\n                                        element.expandRight(connectAdjacent);\n                                    handled = true;\n                                }\n                            }\n                        }\n\n                        if (!!element.parent) { // This is a child.\n                            if (e.altKey && e.which == 38) { // Alt+Up\n                                element.parent.setIsFocused();\n                                handled = true;\n                            }\n\n                            if (element.parent.type == \"Row\") { // Parent is a horizontal container.\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Left\n                                    element.parent.moveFocusPrevChild(element);\n                                    handled = true;\n                                }\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Right\n                                    element.parent.moveFocusNextChild(element);\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 37) { // Ctrl+Left\n                                    element.moveUp();\n                                    resetFocus = true;\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 39) { // Ctrl+Right\n                                    element.moveDown();\n                                    handled = true;\n                                }\n                            }\n                            else { // Parent is a vertical container.\n                                if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Up\n                                    element.parent.moveFocusPrevChild(element);\n                                    handled = true;\n                                }\n                                else if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Down\n                                    element.parent.moveFocusNextChild(element);\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 38) { // Ctrl+Up\n                                    element.moveUp();\n                                    resetFocus = true;\n                                    handled = true;\n                                }\n                                else if (e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 40) { // Ctrl+Down\n                                    element.moveDown();\n                                    handled = true;\n                                }\n                            }\n                        }\n\n                        if (handled) {\n                            e.preventDefault();\n                        }\n\n                        e.stopPropagation();\n\n                        $scope.$apply(); // Event is not triggered by Angular directive but raw event handler on element.\n\n                        // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\n                        if (resetFocus) {\n                            window.setTimeout(function () {\n                                $scope.$apply(function () {\n                                    element.editor.focusedElement.setIsFocused();\n                                });\n                            }, 100);\n                        }\n                    });\n\n                    $scope.element.setIsFocusedEventHandlers.push(function () {\n                        $element.parent().focus();\n                    });\n\n                    $scope.delete = function (element) {\n                        element.delete();\n                    }\n                },\n\n                configureForContainer: function ($scope, $element) {\n                    var element = $scope.element;\n\n                    //$scope.isReceiving = false; // True when container is receiving an external element via drag/drop.\n                    $scope.getShowChildrenPlaceholder = function () {\n                        return $scope.element.children.length === 0 && !$scope.element.getIsDropTarget();\n                    };\n\n                    $scope.sortableOptions = {\n                        cursor: \"move\",\n                        delay: 150,\n                        disabled: element.getIsSealed(),\n                        distance: 5,\n                        //handle: element.children.length < 2 ? \".imaginary-class\" : false, // For some reason doesn't get re-evaluated after adding more children.\n                        start: function (e, ui) {\n                            $scope.$apply(function () {\n                                element.setIsDropTarget(true);\n                                element.editor.isDragging = true;\n                            });\n                            // Make the drop target placeholder as high as the item being dragged.\n                            ui.placeholder.height(ui.item.height() - 4);\n                            ui.placeholder.css(\"min-height\", 0);\n                        },\n                        stop: function (e, ui) {\n                            $scope.$apply(function () {\n                                element.editor.isDragging = false;\n                                element.setIsDropTarget(false);\n                            });\n                        },\n                        over: function (e, ui) {\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\n                                if (!!ui.sender[0].dropTargetTimeout) {\n                                    $timeout.cancel(ui.sender[0].dropTargetTimeout);\n                                    ui.sender[0].dropTargetTimeout = null;\n                                }\n                                $timeout(function () {\n                                    if (element.type == \"Row\") {\n                                        // If there was a previous drop target and it was a row, roll back any pending column adds to it.\n                                        var previousDropTarget = element.editor.dropTargetElement;\n                                        if (!!previousDropTarget && previousDropTarget.type == \"Row\")\n                                            previousDropTarget.rollbackAddColumn();\n                                    }\n                                    element.setIsDropTarget(false);\n                                });\n                                ui.sender[0].dropTargetTimeout = $timeout(function () {\n                                    if (element.type == \"Row\") {\n                                        var receivedColumn = ui.item.sortable.model;\n                                        var receivedColumnWidth = Math.floor(12 / (element.children.length + 1));\n                                        receivedColumn.width = receivedColumnWidth;\n                                        receivedColumn.offset = 0;\n                                        element.beginAddColumn(receivedColumnWidth);\n                                        // Make the drop target placeholder the correct width and as high as the highest existing column in the row.\n                                        var maxHeight = _.max(_($element.find(\"> .layout-children > .layout-column:not(.ui-sortable-placeholder)\")).map(function (e) {\n                                            return $(e).height();\n                                        }));\n                                        for (i = 1; i <= 12; i++)\n                                            ui.placeholder.removeClass(\"col-xs-\" + i);\n                                        ui.placeholder.addClass(\"col-xs-\" + receivedColumn.width);\n                                        if (maxHeight > 0) {\n                                            ui.placeholder.height(maxHeight);\n                                            ui.placeholder.css(\"min-height\", 0);\n                                        }\n                                        else {\n                                            ui.placeholder.height(0);\n                                            ui.placeholder.css(\"min-height\", \"\");\n                                        }\n                                    }\n                                    element.setIsDropTarget(true);\n                                }, 150);\n                            }\n                        },\n                        receive: function (e, ui) {\n                            if (!!ui.sender && !!ui.sender[0].isToolbox) {\n                                $scope.$apply(function () {\n                                    var receivedElement = ui.item.sortable.model;\n                                    if (!!receivedElement) {\n                                        if (element.type == \"Row\")\n                                            element.commitAddColumn();\n                                        // Should ideally call LayoutEditor.Container.addChild() instead, but since this handler\n                                        // is run *before* the ui-sortable directive's handler, if we try to add the child to the\n                                        // array that handler will get an exception when trying to do the same.\n                                        // Because of this, we need to invoke \"setParent\" so that specific container types can perform element speficic initialization.\n                                        receivedElement.setEditor(element.editor);\n                                        receivedElement.setParent(element);\n                                        if (!!receivedElement.hasEditor) {\n                                            $scope.$root.editElement(receivedElement).then(function (args) {\n                                                if (!args.cancel) {\n                                                    receivedElement.data = args.element.data;\n\n                                                    if (receivedElement.setHtml)\n                                                        receivedElement.setHtml(args.element.html);\n                                                }\n                                                $timeout(function () {\n                                                    if (!!args.cancel)\n                                                        receivedElement.delete();\n                                                    else\n                                                        receivedElement.setIsFocused();\n                                                    //$scope.isReceiving = false;\n                                                    element.setIsDropTarget(false);\n\n                                                });\n                                                return;\n                                            });\n                                        }\n                                    }\n                                    $timeout(function () {\n                                        //$scope.isReceiving = false;\n                                        element.setIsDropTarget(false);\n                                        if (!!receivedElement)\n                                            receivedElement.setIsFocused();\n                                    });\n                                });\n                            }\n                        }\n                    };\n\n                    $scope.click = function (child, e) {\n                        if (!child.editor.isDragging)\n                            child.setIsFocused();\n                        e.stopPropagation();\n                    };\n\n                    $scope.getClasses = function (child) {\n                        var result = [\"layout-element\"];\n\n                        if (!!child.children) {\n                            result.push(\"layout-container\");\n                            if (child.getIsSealed())\n                                result.push(\"layout-container-sealed\");\n                        }\n\n                        result.push(\"layout-\" + child.type.toLowerCase());\n\n                        if (!!child.dropTargetClass)\n                            result.push(child.dropTargetClass);\n\n                        // TODO: Move these to either the Column directive or the Column model class.\n                        if (child.type == \"Row\") {\n                            result.push(\"row\");\n                            if (!child.canAddColumn())\n                                result.push(\"layout-row-full\");\n                        }\n                        if (child.type == \"Column\") {\n                            result.push(\"col-xs-\" + child.width);\n                            result.push(\"col-xs-offset-\" + child.offset);\n                        }\n                        if (child.type == \"Content\")\n                            result.push(\"layout-content-\" + child.contentTypeClass);\n\n                        if (child.getIsActive())\n                            result.push(\"layout-element-active\");\n                        if (child.getIsFocused())\n                            result.push(\"layout-element-focused\");\n                        if (child.getIsSelected())\n                            result.push(\"layout-element-selected\");\n                        if (child.getIsDropTarget())\n                            result.push(\"layout-element-droptarget\");\n                        if (child.isTemplated)\n                            result.push(\"layout-element-templated\");\n\n                        return result;\n                    };\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutEditor\", [\"environment\",\n        function (environment) {\n            return {\n                restrict: \"E\",\n                scope: {},\n                controller: [\"$scope\", \"$element\", \"$attrs\", \"$compile\", \"clipboard\",\n                    function ($scope, $element, $attrs, $compile, clipboard) {\n                        if (!!$attrs.model)\n                            $scope.element = eval($attrs.model);\n                        else\n                            throw new Error(\"The 'model' attribute must evaluate to a LayoutEditor.Editor object.\");\n\n                        $scope.click = function (canvas, e) {\n                            if (!canvas.editor.isDragging)\n                                canvas.setIsFocused();\n                            e.stopPropagation();\n                        };\n\n                        $scope.getClasses = function (canvas) {\n                            var result = [\"layout-element\", \"layout-container\", \"layout-canvas\"];\n\n                            if (canvas.getIsActive())\n                                result.push(\"layout-element-active\");\n                            if (canvas.getIsFocused())\n                                result.push(\"layout-element-focused\");\n                            if (canvas.getIsSelected())\n                                result.push(\"layout-element-selected\");\n                            if (canvas.getIsDropTarget())\n                                result.push(\"layout-element-droptarget\");\n                            if (canvas.isTemplated)\n                                result.push(\"layout-element-templated\");\n\n                            return result;\n                        };\n\n                        // An unfortunate side-effect of the next hack on line 54 is that the created elements aren't added to the DOM yet, so we can't use it to get to the parent \".layout-desiger\" element.\n                        // Work around: access that element directly (which efectively turns multiple layout editors on a single page impossible). \n                        // //var layoutDesignerHost = $element.closest(\".layout-designer\").data(\"layout-designer-host\");\n                        var layoutDesignerHost = $(\".layout-designer\").data(\"layout-designer-host\");\n\n                        $scope.$root.layoutDesignerHost = layoutDesignerHost;\n\n                        layoutDesignerHost.element.on(\"replacecanvas\", function (e, args) {\n                            var editor = $scope.element;\n                            var canvasData = {\n                                data: args.canvas.data,\n                                htmlId: args.canvas.htmlId,\n                                htmlClass: args.canvas.htmlClass,\n                                htmlStyle: args.canvas.htmlStyle,\n                                isTemplated: args.canvas.isTemplated,\n                                children: args.canvas.children\n                            };\n\n                            // HACK: Instead of simply updating the $scope.element with a new instance, we need to replace the entire orc-layout-editor markup\n                            // in order for angular to rebind starting with the Canvas element. Otherwise, for some reason, it will rebind starting with the first child of Canvas.\n                            // You can see this happening when setting a breakpoint in ScopeConfigurator where containers are initialized with drag & drop: on page load, the first element\n                            // is a Canvas (good), but after having selected another template, the first element is (typically) a Grid (bad).\n                            // Simply recompiling the orc-layout-editor directive will cause the entire thing to be generated, which works just fine as well (even though not is nice as simply leveraging model binding).\n                            layoutDesignerHost.editor = window.layoutEditor = new LayoutEditor.Editor(editor.config, canvasData);\n                            var template = \"<orc-layout-editor\" + \" model='window.layoutEditor' />\";\n                            var html = $compile(template)($scope);\n                            $(\".layout-editor-holder\").html(html);\n                        });\n\n                        $scope.$root.editElement = function (element) {\n                            var host = $scope.$root.layoutDesignerHost;\n                            return host.editElement(element);\n                        };\n\n                        $scope.$root.addElement = function (contentType) {\n                            var host = $scope.$root.layoutDesignerHost;\n                            return host.addElement(contentType);\n                        };\n\n                        $scope.toggleInlineEditing = function () {\n                            if (!$scope.element.inlineEditingIsActive) {\n                                $scope.element.inlineEditingIsActive = true;\n                                $element.find(\".layout-toolbar-container\").show();\n                                var selector = \"#layout-editor-\" + $scope.$id + \" .layout-html .layout-content-markup[data-templated=false]\";\n                                var firstContentEditorId = $(selector).first().attr(\"id\");\n                                tinymce.init({\n                                    selector: selector,\n                                    theme: \"modern\",\n                                    schema: \"html5\",\n                                    plugins: [\n                                        \"advlist autolink lists link image charmap print preview hr anchor pagebreak\",\n                                        \"searchreplace wordcount visualblocks visualchars code fullscreen\",\n                                        \"insertdatetime media nonbreaking table contextmenu directionality\",\n                                        \"emoticons template paste textcolor colorpicker textpattern\",\n                                        \"fullscreen autoresize\"\n                                    ],\n                                    toolbar: \"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close\",\n                                    convert_urls: false,\n                                    valid_elements: \"*[*]\",\n                                    // Shouldn't be needed due to the valid_elements setting, but TinyMCE would strip script.src without it.\n                                    extended_valid_elements: \"script[type|defer|src|language]\",\n                                    statusbar: false,\n                                    skin: \"orchardlightgray\",\n                                    inline: true,\n                                    fixed_toolbar_container: \"#layout-editor-\" + $scope.$id + \" .layout-toolbar-container\",\n                                    init_instance_callback: function (editor) {\n                                        if (editor.id == firstContentEditorId)\n                                            tinymce.execCommand(\"mceFocus\", false, editor.id);\n                                    }\n                                });\n                            }\n                            else {\n                                tinymce.remove(\"#layout-editor-\" + $scope.$id + \" .layout-content-markup\");\n                                $element.find(\".layout-toolbar-container\").hide();\n                                $scope.element.inlineEditingIsActive = false;\n                            }\n                        };\n\n                        $(document).on(\"cut copy paste\", function (e) {\n                            // If the pseudo clipboard was already invoked (which happens on the first clipboard\n                            // operation after page load even if native clipboard support exists) then sit this\n                            // one operation out, but make sure whatever is on the pseudo clipboard gets migrated\n                            // to the native clipboard for subsequent operations.\n                            if (clipboard.wasInvoked()) {\r\n                                e.originalEvent.clipboardData.setData(\"text/plain\", clipboard.getData(\"text/plain\"));\r\n                                e.originalEvent.clipboardData.setData(\"text/json\", clipboard.getData(\"text/json\"));\r\n                                e.preventDefault();\r\n                            }\r\n                            else {\r\n                                var focusedElement = $scope.element.focusedElement;\n                                if (!!focusedElement) {\r\n                                    $scope.$apply(function () {\r\n                                        switch (e.type) {\r\n                                            case \"copy\":\n                                                focusedElement.copy(e.originalEvent.clipboardData);\n                                                break;\n                                            case \"cut\":\n                                                focusedElement.cut(e.originalEvent.clipboardData);\n                                                break;\n                                            case \"paste\":\n                                                focusedElement.paste(e.originalEvent.clipboardData);\n                                                break;\r\n                                        }\r\n                                    });\n\n                                    // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost.\n                                    window.setTimeout(function () {\r\n                                        $scope.$apply(function () {\r\n                                            if (!!$scope.element.focusedElement)\n                                                $scope.element.focusedElement.setIsFocused();\r\n                                        });\r\n                                    }, 100);\n\n                                    e.preventDefault();\r\n                                }\r\n                            }\n\n                            // Native clipboard support obviously exists, so disable the peudo clipboard from now on.\n                            clipboard.disable();\r\n                        });\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Editor\"),\n                replace: true,\n                link: function (scope, element) {\n                    // No clicks should propagate from the TinyMCE toolbars.\n                    element.find(\".layout-toolbar-container\").click(function (e) {\n                        e.stopPropagation();\n                    });\n                    // Intercept mousedown on editor while in inline editing mode to \n                    // prevent current editor from losing focus.\n                    element.mousedown(function (e) {\n                        if (scope.element.inlineEditingIsActive) {\n                            e.preventDefault();\n                            e.stopPropagation();\n                        }\n                    })\n                    // Unfocus and unselect everything on click outside of canvas.\n                    $(window).click(function (e) {\n                        // Except when in inline editing mode.\n                        if (!scope.element.inlineEditingIsActive) {\n                            scope.$apply(function () {\n                                scope.element.activeElement = null;\n                                scope.element.focusedElement = null;\n                            });\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutCanvas\", [\"scopeConfigurator\", \"environment\",\n        function (scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\", \"$attrs\",\n                    function ($scope, $element, $attrs) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Canvas\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutChild\", [\"$compile\",\n        function ($compile) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                link: function (scope, element) {\n                    var template = \"<orc-layout-\" + scope.element.type.toLowerCase() + \" element='element' />\";\n                    var html = $compile(template)(scope);\n                    $(element).replaceWith(html);\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutColumn\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Column\"),\n                replace: true,\n                link: function (scope, element, attrs) {\n                    element.find(\".layout-column-resize-bar\").draggable({\n                        axis: \"x\",\n                        helper: \"clone\",\n                        revert: true,\n                        start: function (e, ui) {\n                            scope.$apply(function () {\n                                scope.element.editor.isResizing = true;\n                            });\n                        },\n                        drag: function (e, ui) {\n                            var columnElement = element.parent();\n                            var columnSize = columnElement.width() / scope.element.width;\n                            var connectAdjacent = !e.ctrlKey;\n                            if ($(e.target).hasClass(\"layout-column-resize-bar-left\")) {\n                                var delta = ui.offset.left - columnElement.offset().left;\n                                if (delta < -columnSize && scope.element.canExpandLeft(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.expandLeft(connectAdjacent);\n                                    });\n                                }\n                                else if (delta > columnSize && scope.element.canContractLeft(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.contractLeft(connectAdjacent);\n                                    });\n                                }\n                            }\n                            else if ($(e.target).hasClass(\"layout-column-resize-bar-right\")) {\n                                var delta = ui.offset.left - columnElement.width() - columnElement.offset().left;\n                                if (delta > columnSize && scope.element.canExpandRight(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.expandRight(connectAdjacent);\n                                    });\n                                }\n                                else if (delta < -columnSize && scope.element.canContractRight(connectAdjacent)) {\n                                    scope.$apply(function () {\n                                        scope.element.contractRight(connectAdjacent);\n                                    });\n                                }\n                            }\n\n                        },\n                        stop: function (e, ui) {\n                            scope.$apply(function () {\n                              scope.element.editor.isResizing = false;\n                            });\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutContent\", [\"$sce\", \"scopeConfigurator\", \"environment\",\n        function ($sce, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        $scope.edit = function () {\n                            $scope.$root.editElement($scope.element).then(function (args) {\n                                $scope.$apply(function () {\n                                    if (args.cancel)\n                                        return;\n\n                                    $scope.element.data = args.element.data;\n                                    $scope.element.setHtml(args.element.html);\n                                });\n                            });\n                        };\n\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\n                        $scope.element.setHtml = function (html) {\n                            $scope.element.html = html;\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\n                        };\n\n                        $scope.element.setHtml($scope.element.html);\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Content\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutHtml\", [\"$sce\", \"scopeConfigurator\", \"environment\",\n        function ($sce, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        $scope.edit = function () {\n                            $scope.$root.editElement($scope.element).then(function (args) {\n                                $scope.$apply(function () {\n                                    if (args.cancel)\n                                        return;\n\n                                    $scope.element.data = args.element.data;\n                                    $scope.element.setHtml(args.element.html);\n                                });\n                            });\n                        };\n                        $scope.updateContent = function (e) {\n                            $scope.element.setHtml(e.target.innerHTML);\n                        };\n\n                        // Overwrite the setHtml function so that we can use the $sce service to trust the html (and not have the html binding strip certain tags).\n                        $scope.element.setHtml = function (html) {\n                            $scope.element.html = html;\n                            $scope.element.htmlUnsafe = $sce.trustAsHtml(html);\n                        };\n\n                        $scope.element.setHtml($scope.element.html);\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Html\"),\n                replace: true,\n                link: function (scope, element) {\n                    // Mouse down events must not be intercepted by drag and drop while inline editing is active,\n                    // otherwise clicks in inline editors will have no effect.\n                    element.find(\".layout-content-markup\").mousedown(function (e) {\n                        if (scope.element.editor.inlineEditingIsActive) {\n                            e.stopPropagation();\n                        }\n                    });\n                }\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutGrid\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"y\";\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Grid\"),\n                replace: true\n            };\n        }\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutRow\", [\"$compile\", \"scopeConfigurator\", \"environment\",\n        function ($compile, scopeConfigurator, environment) {\n            return {\n                restrict: \"E\",\n                scope: { element: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        scopeConfigurator.configureForElement($scope, $element);\n                        scopeConfigurator.configureForContainer($scope, $element);\n                        $scope.sortableOptions[\"axis\"] = \"x\";\n                        $scope.sortableOptions[\"ui-floating\"] = true;\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"Row\"),\n                replace: true\n            };\n        }\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutPopup\", [\r\n        function () {\r\n            return {\r\n                restrict: \"A\",\r\n                link: function (scope, element, attrs) {\r\n                    var popup = $(element);\r\n                    var trigger = popup.closest(\".layout-popup-trigger\");\r\n                    var parentElement = popup.closest(\".layout-element\");\r\n                    trigger.click(function () {\r\n                        popup.toggle();\r\n                        if (popup.is(\":visible\")) {\r\n                            popup.position({\r\n                                my: attrs.orcLayoutPopupMy || \"left top\",\r\n                                at: attrs.orcLayoutPopupAt || \"left bottom+4px\",\r\n                                of: trigger\r\n                            });\r\n                            popup.find(\"input\").first().focus();\r\n                        }\r\n                    });\r\n                    popup.click(function (e) {\r\n                        e.stopPropagation();\r\n                    });\r\n                    parentElement.click(function (e) {\r\n                        popup.hide();\r\n                    });\r\n                    popup.keydown(function (e) {\r\n                        if (!e.ctrlKey && !e.shiftKey && !e.altKey && e.which == 27) // Esc\r\n                            popup.hide();\r\n                        e.stopPropagation();\r\n                    });\r\n                    popup.on(\"cut copy paste\", function (e) {\n                        // Allow clipboard operations in popup without invoking clipboard event handlers on parent element.\n                        e.stopPropagation();\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\r\n    .module(\"LayoutEditor\")\r\n    .directive(\"orcLayoutToolbox\", [\"$compile\", \"environment\",\r\n        function ($compile, environment) {\r\n            return {\r\n                restrict: \"E\",\r\n                controller: [\"$scope\", \"$element\",\r\n                    function ($scope, $element) {\r\n\r\n                        $scope.resetElements = function () {\r\n\r\n                            $scope.gridElements = [\r\n                                LayoutEditor.Grid.from({\r\n                                    toolboxIcon: \"\\uf00a\",\r\n                                    toolboxLabel: \"Grid\",\r\n                                    toolboxDescription: \"Empty grid.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.rowElements = [\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (1 column)\",\r\n                                    toolboxDescription: \"Row with 1 column.\",\r\n                                    children: LayoutEditor.Column.times(1)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (2 columns)\",\r\n                                    toolboxDescription: \"Row with 2 columns.\",\r\n                                    children: LayoutEditor.Column.times(2)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (3 columns)\",\r\n                                    toolboxDescription: \"Row with 3 columns.\",\r\n                                    children: LayoutEditor.Column.times(3)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (4 columns)\",\r\n                                    toolboxDescription: \"Row with 4 columns.\",\r\n                                    children: LayoutEditor.Column.times(4)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (6 columns)\",\r\n                                    toolboxDescription: \"Row with 6 columns.\",\r\n                                    children: LayoutEditor.Column.times(6)\r\n                                }),\r\n                                LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (12 columns)\",\r\n                                    toolboxDescription: \"Row with 12 columns.\",\r\n                                    children: LayoutEditor.Column.times(12)\r\n                                }), LayoutEditor.Row.from({\r\n                                    toolboxIcon: \"\\uf0c9\",\r\n                                    toolboxLabel: \"Row (empty)\",\r\n                                    toolboxDescription: \"Empty row.\",\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.columnElements = [\r\n                                LayoutEditor.Column.from({\r\n                                    toolboxIcon: \"\\uf0db\",\r\n                                    toolboxLabel: \"Column\",\r\n                                    toolboxDescription: \"Empty column.\",\r\n                                    width: 1,\r\n                                    offset: 0,\r\n                                    children: []\r\n                                })\r\n                            ];\r\n\r\n                            $scope.contentElementCategories = _($scope.element.config.categories).map(function (category) {\r\n                                return {\r\n                                    name: category.name,\r\n                                    elements: _(category.contentTypes).map(function (contentType) {\r\n                                        var type = contentType.type;\r\n                                        var factory = LayoutEditor.factories[type] || LayoutEditor.factories[\"Content\"];\r\n                                        var item = {\r\n                                            isTemplated: false,\r\n                                            contentType: contentType.id,\r\n                                            contentTypeLabel: contentType.label,\r\n                                            contentTypeClass: contentType.typeClass,\r\n                                            data: null,\r\n                                            hasEditor: contentType.hasEditor,\r\n                                            html: contentType.html\r\n                                        };\r\n                                        var element = factory(item);\r\n                                        element.toolboxIcon = contentType.icon || \"\\uf1c9\";\r\n                                        element.toolboxLabel = contentType.label;\r\n                                        element.toolboxDescription = contentType.description;\r\n                                        return element;\r\n                                    })\r\n                                };\r\n                            });\r\n\r\n                        };\r\n\r\n                        $scope.resetElements();\r\n\r\n                        $scope.getSortableOptions = function (type) {\r\n                            var editorId = $element.closest(\".layout-editor\").attr(\"id\");\r\n                            var parentClasses;\r\n                            var placeholderClasses;\r\n                            var floating = false;\r\n\r\n                            switch (type) {\r\n                                case \"Grid\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-grid ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Row\":\r\n                                    parentClasses = [\".layout-grid\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-row row ui-sortable-placeholder\";\r\n                                    break;\r\n                                case \"Column\":\r\n                                    parentClasses = [\".layout-row:not(.layout-row-full)\"];\r\n                                    placeholderClasses = \"layout-element layout-container layout-column ui-sortable-placeholder\";\r\n                                    floating = true; // To ensure a smooth horizontal-list reordering. https://github.com/angular-ui/ui-sortable#floating\r\n                                    break;\r\n                                case \"Content\":\r\n                                    parentClasses = [\".layout-canvas\", \".layout-column\", \".layout-common-holder\"];\r\n                                    placeholderClasses = \"layout-element layout-content ui-sortable-placeholder\";\r\n                                    break;\r\n                            }\r\n\r\n                            return {\r\n                                cursor: \"move\",\r\n                                connectWith: _(parentClasses).map(function (e) { return \"#\" + editorId + \" \" + e + \":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children\"; }).join(\", \"),\r\n                                placeholder: placeholderClasses,\r\n                                \"ui-floating\": floating,\r\n                                create: function (e, ui) {\r\n                                    e.target.isToolbox = true; // Will indicate to connected sortables that dropped items were sent from toolbox.\r\n                                },\r\n                                start: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = true;\r\n                                    });\r\n                                },\r\n                                stop: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.isDragging = false;\r\n                                        $scope.resetElements();\r\n                                    });\r\n                                },\r\n                                over: function (e, ui) {\r\n                                    $scope.$apply(function () {\r\n                                        $scope.element.canvas.setIsDropTarget(false);\r\n                                    });\r\n                                },\r\n                            }\r\n                        };\r\n\r\n                        var layoutIsCollapsedCookieName = \"layoutToolboxCategory_Layout_IsCollapsed\";\r\n                        $scope.layoutIsCollapsed = $.cookie(layoutIsCollapsedCookieName) === \"true\";\r\n\r\n                        $scope.toggleLayoutIsCollapsed = function (e) {\r\n                            $scope.layoutIsCollapsed = !$scope.layoutIsCollapsed;\r\n                            $.cookie(layoutIsCollapsedCookieName, $scope.layoutIsCollapsed, { expires: 365 }); // Remember collapsed state for a year.\r\n                            e.preventDefault();\r\n                            e.stopPropagation();\r\n                        };\r\n                    }\r\n                ],\r\n                templateUrl: environment.templateUrl(\"Toolbox\"),\r\n                replace: true,\r\n                link: function (scope, element) {\r\n                    var toolbox = element.find(\".layout-toolbox\");\r\n                    $(window).on(\"resize scroll\", function (e) {\r\n                        var canvas = element.parent().find(\".layout-canvas\");\r\n                        // If the canvas is taller than the toolbox, make the toolbox sticky-positioned within the editor\r\n                        // to help the user avoid excessive vertical scrolling.\r\n                        var canvasIsTaller = !!canvas && canvas.height() > toolbox.height();\r\n                        var windowPos = $(window).scrollTop();\r\n                        if (canvasIsTaller && windowPos > element.offset().top + element.height() - toolbox.height()) {\r\n                            toolbox.addClass(\"sticky-bottom\");\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                        }\r\n                        else if (canvasIsTaller && windowPos > element.offset().top) {\r\n                            toolbox.addClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                        else {\r\n                            toolbox.removeClass(\"sticky-top\");\r\n                            toolbox.removeClass(\"sticky-bottom\");\r\n                        }\r\n                    });\r\n                }\r\n            };\r\n        }\r\n    ]);","angular\n    .module(\"LayoutEditor\")\n    .directive(\"orcLayoutToolboxGroup\", [\"$compile\", \"environment\",\n        function ($compile, environment) {\n            return {\n                restrict: \"E\",\n                scope: { category: \"=\" },\n                controller: [\"$scope\", \"$element\",\n                    function ($scope, $element) {\n                        var isCollapsedCookieName = \"layoutToolboxCategory_\" + $scope.category.name + \"_IsCollapsed\";\n                        $scope.isCollapsed = $.cookie(isCollapsedCookieName) === \"true\";\n                        $scope.toggleIsCollapsed = function (e) {\n                            $scope.isCollapsed = !$scope.isCollapsed;\n                            $.cookie(isCollapsedCookieName, $scope.isCollapsed, { expires: 365 }); // Remember collapsed state for a year.\n                            e.preventDefault();\n                            e.stopPropagation();\n                        };\n                    }\n                ],\n                templateUrl: environment.templateUrl(\"ToolboxGroup\"),\n                replace: true\n            };\n        }\n    ]);"],"sourceRoot":"/source/"} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js index 43793db82..dad14fc95 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor.min.js @@ -1 +1 @@ -angular.module("LayoutEditor",["ngSanitize","ngResource","ui.sortable"]);var LayoutEditor;!function(e){var t=function(){var e=this;this.clipboardData={},this.setData=function(t,o,n){e.clipboardData[t]=o},this.getData=function(t,o){return e.clipboardData[t]},this.disable=function(){this.disabled=!0}};e.Clipboard=new t,angular.module("LayoutEditor").factory("clipboard",[function(){return{setData:e.Clipboard.setData,getData:e.Clipboard.getData,disable:e.Clipboard.disable}}])}(LayoutEditor||(LayoutEditor={})),angular.module("LayoutEditor").factory("scopeConfigurator",["$timeout","clipboard",function(e,t){return{configureForElement:function(e,o){o.find(".layout-panel").click(function(e){e.stopPropagation()}),o.parent().keydown(function(n){var l=!1,i=!1,a=e.element;if(!a.editor.isDragging&&!a.editor.inlineEditingIsActive){if(!t.disabled){var r=a.editor.focusedElement;if(r&&n.ctrlKey)switch(n.which){case 67:r.copy(t);break;case 88:r.cut(t);break;case 86:r.paste(t)}}if(n.ctrlKey||n.shiftKey||n.altKey||46!=n.which?n.ctrlKey||n.shiftKey||n.altKey||32!=n.which&&27!=n.which||(o.find(".layout-panel-action-properties").first().click(),l=!0):(e["delete"](a),l=!0),"Content"==a.type&&(n.ctrlKey||n.shiftKey||n.altKey||13!=n.which||(o.find(".layout-panel-action-edit").first().click(),l=!0)),a.children&&(n.ctrlKey||n.shiftKey||!n.altKey||40!=n.which||(a.children.length>0&&a.children[0].setIsFocused(),l=!0),"Column"==a.type)){var c=!n.ctrlKey;37==n.which?(n.altKey&&a.expandLeft(c),n.shiftKey&&a.contractRight(c),l=!0):39==n.which&&(n.altKey&&a.contractLeft(c),n.shiftKey&&a.expandRight(c),l=!0)}a.parent&&(n.altKey&&38==n.which&&(a.parent.setIsFocused(),l=!0),"Row"==a.parent.type?n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?n.ctrlKey||n.shiftKey||n.altKey||39!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||39!=n.which||(a.moveDown(),l=!0):(a.moveUp(),i=!0,l=!0):(a.parent.moveFocusNextChild(a),l=!0):(a.parent.moveFocusPrevChild(a),l=!0):n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?n.ctrlKey||n.shiftKey||n.altKey||40!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||40!=n.which||(a.moveDown(),l=!0):(a.moveUp(),i=!0,l=!0):(a.parent.moveFocusNextChild(a),l=!0):(a.parent.moveFocusPrevChild(a),l=!0)),l&&n.preventDefault(),n.stopPropagation(),e.$apply(),i&&window.setTimeout(function(){e.$apply(function(){a.editor.focusedElement.setIsFocused()})},100)}}),e.element.setIsFocusedEventHandlers.push(function(){o.parent().focus()}),e["delete"]=function(e){e["delete"]()}},configureForContainer:function(t,o){var n=t.element;t.getShowChildrenPlaceholder=function(){return 0===t.element.children.length&&!t.element.getIsDropTarget()},t.sortableOptions={cursor:"move",delay:150,disabled:n.getIsSealed(),distance:5,start:function(e,o){t.$apply(function(){n.setIsDropTarget(!0),n.editor.isDragging=!0}),o.placeholder.height(o.item.height()-4),o.placeholder.css("min-height",0)},stop:function(e,o){t.$apply(function(){n.editor.isDragging=!1,n.setIsDropTarget(!1)})},over:function(t,l){l.sender&&l.sender[0].isToolbox&&(l.sender[0].dropTargetTimeout&&(e.cancel(l.sender[0].dropTargetTimeout),l.sender[0].dropTargetTimeout=null),e(function(){if("Row"==n.type){var e=n.editor.dropTargetElement;e&&"Row"==e.type&&e.rollbackAddColumn()}n.setIsDropTarget(!1)}),l.sender[0].dropTargetTimeout=e(function(){if("Row"==n.type){var e=l.item.sortable.model,t=Math.floor(12/(n.children.length+1));e.width=t,e.offset=0,n.beginAddColumn(t);var a=_.max(_(o.find("> .layout-children > .layout-column:not(.ui-sortable-placeholder)")).map(function(e){return $(e).height()}));for(i=1;i<=12;i++)l.placeholder.removeClass("col-xs-"+i);l.placeholder.addClass("col-xs-"+e.width),a>0?(l.placeholder.height(a),l.placeholder.css("min-height",0)):(l.placeholder.height(0),l.placeholder.css("min-height",""))}n.setIsDropTarget(!0)},150))},receive:function(o,l){l.sender&&l.sender[0].isToolbox&&t.$apply(function(){var o=l.item.sortable.model;o&&("Row"==n.type&&n.commitAddColumn(),o.setEditor(n.editor),o.setParent(n),o.hasEditor&&t.$root.editElement(o).then(function(t){t.cancel||(o.data=t.element.data,o.setHtml&&o.setHtml(t.element.html)),e(function(){t.cancel?o["delete"]():o.setIsFocused(),n.setIsDropTarget(!1)})})),e(function(){n.setIsDropTarget(!1),o&&o.setIsFocused()})})}},t.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},t.getClasses=function(e){var t=["layout-element"];return e.children&&(t.push("layout-container"),e.getIsSealed()&&t.push("layout-container-sealed")),t.push("layout-"+e.type.toLowerCase()),e.dropTargetClass&&t.push(e.dropTargetClass),"Row"==e.type&&(t.push("row"),e.canAddColumn()||t.push("layout-row-full")),"Column"==e.type&&(t.push("col-xs-"+e.width),t.push("col-xs-offset-"+e.offset)),"Content"==e.type&&t.push("layout-content-"+e.contentTypeClass),e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t}}}}]),angular.module("LayoutEditor").directive("orcLayoutEditor",["environment",function(environment){return{restrict:"E",scope:{},controller:["$scope","$element","$attrs","$compile","clipboard",function($scope,$element,$attrs,$compile,clipboard){if(!$attrs.model)throw new Error("The 'model' attribute must evaluate to a LayoutEditor.Editor object.");$scope.element=eval($attrs.model),$scope.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},$scope.getClasses=function(e){var t=["layout-element","layout-container","layout-canvas"];return e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t};var layoutDesignerHost=$(".layout-designer").data("layout-designer-host");$scope.$root.layoutDesignerHost=layoutDesignerHost,layoutDesignerHost.element.on("replacecanvas",function(e,t){var o=$scope.element,n={data:t.canvas.data,htmlId:t.canvas.htmlId,htmlClass:t.canvas.htmlClass,htmlStyle:t.canvas.htmlStyle,isTemplated:t.canvas.isTemplated,children:t.canvas.children};layoutDesignerHost.editor=window.layoutEditor=new LayoutEditor.Editor(o.config,n);var l="",i=$compile(l)($scope);$(".layout-editor-holder").html(i)}),$scope.$root.editElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.editElement(e)},$scope.$root.addElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.addElement(e)},$scope.toggleInlineEditing=function(){if($scope.element.inlineEditingIsActive)tinymce.remove("#layout-editor-"+$scope.$id+" .layout-content-markup"),$element.find(".layout-toolbar-container").hide(),$scope.element.inlineEditingIsActive=!1;else{$scope.element.inlineEditingIsActive=!0,$element.find(".layout-toolbar-container").show();var e="#layout-editor-"+$scope.$id+" .layout-html .layout-content-markup[data-templated=false]",t=$(e).first().attr("id");tinymce.init({selector:e,theme:"modern",schema:"html5",plugins:["advlist autolink lists link image charmap print preview hr anchor pagebreak","searchreplace wordcount visualblocks visualchars code fullscreen","insertdatetime media nonbreaking table contextmenu directionality","emoticons template paste textcolor colorpicker textpattern","fullscreen autoresize"],toolbar:"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close",convert_urls:!1,valid_elements:"*[*]",extended_valid_elements:"script[type|defer|src|language]",statusbar:!1,skin:"orchardlightgray",inline:!0,fixed_toolbar_container:"#layout-editor-"+$scope.$id+" .layout-toolbar-container",init_instance_callback:function(e){e.id==t&&tinymce.execCommand("mceFocus",!1,e.id)}})}},$(document).on("cut copy paste",function(e){clipboard.disable();var t=$scope.element.focusedElement;t&&($scope.$apply(function(){switch(e.type){case"copy":t.copy(e.originalEvent.clipboardData);break;case"cut":t.cut(e.originalEvent.clipboardData);break;case"paste":t.paste(e.originalEvent.clipboardData)}}),window.setTimeout(function(){$scope.$apply(function(){$scope.element.focusedElement&&$scope.element.focusedElement.setIsFocused()})},100),e.preventDefault())})}],templateUrl:environment.templateUrl("Editor"),replace:!0,link:function(e,t){t.find(".layout-toolbar-container").click(function(e){e.stopPropagation()}),t.mousedown(function(t){e.element.inlineEditingIsActive&&(t.preventDefault(),t.stopPropagation())}),$(window).click(function(t){e.element.inlineEditingIsActive||e.$apply(function(){e.element.activeElement=null,e.element.focusedElement=null})})}}}]),angular.module("LayoutEditor").directive("orcLayoutCanvas",["scopeConfigurator","environment",function(e,t){return{restrict:"E",scope:{element:"="},controller:["$scope","$element","$attrs",function(t,o,n){e.configureForElement(t,o),e.configureForContainer(t,o),t.sortableOptions.axis="y"}],templateUrl:t.templateUrl("Canvas"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutChild",["$compile",function(e){return{restrict:"E",scope:{element:"="},link:function(t,o){var n="",l=e(n)(t);$(o).replaceWith(l)}}}]),angular.module("LayoutEditor").directive("orcLayoutColumn",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Column"),replace:!0,link:function(e,t,o){t.find(".layout-column-resize-bar").draggable({axis:"x",helper:"clone",revert:!0,start:function(t,o){e.$apply(function(){e.element.editor.isResizing=!0})},drag:function(o,n){var l=t.parent(),i=l.width()/e.element.width,a=!o.ctrlKey;if($(o.target).hasClass("layout-column-resize-bar-left")){var r=n.offset.left-l.offset().left;-i>r&&e.element.canExpandLeft(a)?e.$apply(function(){e.element.expandLeft(a)}):r>i&&e.element.canContractLeft(a)&&e.$apply(function(){e.element.contractLeft(a)})}else if($(o.target).hasClass("layout-column-resize-bar-right")){var r=n.offset.left-l.width()-l.offset().left;r>i&&e.element.canExpandRight(a)?e.$apply(function(){e.element.expandRight(a)}):-i>r&&e.element.canContractRight(a)&&e.$apply(function(){e.element.contractRight(a)})}},stop:function(t,o){e.$apply(function(){e.element.editor.isResizing=!1})}})}}}]),angular.module("LayoutEditor").directive("orcLayoutContent",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Content"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutHtml",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.updateContent=function(e){o.element.setHtml(e.target.innerHTML)},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Html"),replace:!0,link:function(e,t){t.find(".layout-content-markup").mousedown(function(t){e.element.editor.inlineEditingIsActive&&t.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutGrid",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Grid"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutRow",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="x",e.sortableOptions["ui-floating"]=!0}],templateUrl:o.templateUrl("Row"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutPopup",[function(){return{restrict:"A",link:function(e,t,o){var n=$(t),l=n.closest(".layout-popup-trigger"),i=n.closest(".layout-element");l.click(function(){n.toggle(),n.is(":visible")&&(n.position({my:o.orcLayoutPopupMy||"left top",at:o.orcLayoutPopupAt||"left bottom+4px",of:l}),n.find("input").first().focus())}),n.click(function(e){e.stopPropagation()}),i.click(function(e){n.hide()}),n.keydown(function(e){e.ctrlKey||e.shiftKey||e.altKey||27!=e.which||n.hide(),e.stopPropagation()}),n.on("cut copy paste",function(e){e.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolbox",["$compile","environment",function(e,t){return{restrict:"E",controller:["$scope","$element",function(e,t){e.resetElements=function(){e.gridElements=[LayoutEditor.Grid.from({toolboxIcon:"",toolboxLabel:"Grid",toolboxDescription:"Empty grid.",children:[]})],e.rowElements=[LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (1 column)",toolboxDescription:"Row with 1 column.",children:LayoutEditor.Column.times(1)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (2 columns)",toolboxDescription:"Row with 2 columns.",children:LayoutEditor.Column.times(2)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (3 columns)",toolboxDescription:"Row with 3 columns.",children:LayoutEditor.Column.times(3)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (4 columns)",toolboxDescription:"Row with 4 columns.",children:LayoutEditor.Column.times(4)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (6 columns)",toolboxDescription:"Row with 6 columns.",children:LayoutEditor.Column.times(6)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (12 columns)",toolboxDescription:"Row with 12 columns.",children:LayoutEditor.Column.times(12)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (empty)",toolboxDescription:"Empty row.",children:[]})],e.columnElements=[LayoutEditor.Column.from({toolboxIcon:"",toolboxLabel:"Column",toolboxDescription:"Empty column.",width:1,offset:0,children:[]})],e.contentElementCategories=_(e.element.config.categories).map(function(e){return{name:e.name,elements:_(e.contentTypes).map(function(e){var t=e.type,o=LayoutEditor.factories[t]||LayoutEditor.factories.Content,n={isTemplated:!1,contentType:e.id,contentTypeLabel:e.label,contentTypeClass:e.typeClass,data:null,hasEditor:e.hasEditor,html:e.html},l=o(n);return l.toolboxIcon=e.icon||"",l.toolboxLabel=e.label,l.toolboxDescription=e.description,l})}})},e.resetElements(),e.getSortableOptions=function(o){var n,l,i=t.closest(".layout-editor").attr("id"),a=!1;switch(o){case"Grid":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder";break;case"Row":n=[".layout-grid"],l="layout-element layout-container layout-row row ui-sortable-placeholder";break;case"Column":n=[".layout-row:not(.layout-row-full)"],l="layout-element layout-container layout-column ui-sortable-placeholder",a=!0;break;case"Content":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-content ui-sortable-placeholder"}return{cursor:"move",connectWith:_(n).map(function(e){return"#"+i+" "+e+":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children"}).join(", "),placeholder:l,"ui-floating":a,create:function(e,t){e.target.isToolbox=!0},start:function(t,o){e.$apply(function(){e.element.isDragging=!0})},stop:function(t,o){e.$apply(function(){e.element.isDragging=!1,e.resetElements()})},over:function(t,o){e.$apply(function(){e.element.canvas.setIsDropTarget(!1)})}}};var o="layoutToolboxCategory_Layout_IsCollapsed";e.layoutIsCollapsed="true"===$.cookie(o),e.toggleLayoutIsCollapsed=function(t){e.layoutIsCollapsed=!e.layoutIsCollapsed,$.cookie(o,e.layoutIsCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("Toolbox"),replace:!0,link:function(e,t){var o=t.find(".layout-toolbox");$(window).on("resize scroll",function(e){var n=t.parent().find(".layout-canvas"),l=!!n&&n.height()>o.height(),i=$(window).scrollTop();l&&i>t.offset().top+t.height()-o.height()?(o.addClass("sticky-bottom"),o.removeClass("sticky-top")):l&&i>t.offset().top?(o.addClass("sticky-top"),o.removeClass("sticky-bottom")):(o.removeClass("sticky-top"),o.removeClass("sticky-bottom"))})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolboxGroup",["$compile","environment",function(e,t){return{restrict:"E",scope:{category:"="},controller:["$scope","$element",function(e,t){var o="layoutToolboxCategory_"+e.category.name+"_IsCollapsed";e.isCollapsed="true"===$.cookie(o),e.toggleIsCollapsed=function(t){e.isCollapsed=!e.isCollapsed,$.cookie(o,e.isCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("ToolboxGroup"),replace:!0}}]); \ No newline at end of file +angular.module("LayoutEditor",["ngSanitize","ngResource","ui.sortable"]);var LayoutEditor;!function(e){var t=function(){var e=this;this._clipboardData={},this._isDisabled=!1,this._wasInvoked=!1,this.setData=function(t,o){e._clipboardData[t]=o,e._wasInvoked=!0},this.getData=function(t){return e._clipboardData[t]},this.disable=function(){e._isDisabled=!0,e._wasInvoked=!1,e._clipboardData={}},this.isDisabled=function(){return e._isDisabled},this.wasInvoked=function(){return e._wasInvoked}};e.Clipboard=new t,angular.module("LayoutEditor").factory("clipboard",[function(){return{setData:e.Clipboard.setData,getData:e.Clipboard.getData,disable:e.Clipboard.disable,isDisabled:e.Clipboard.isDisabled,wasInvoked:e.Clipboard.wasInvoked}}])}(LayoutEditor||(LayoutEditor={})),angular.module("LayoutEditor").factory("scopeConfigurator",["$timeout","clipboard",function(e,t){return{configureForElement:function(e,o){o.find(".layout-panel").click(function(e){e.stopPropagation()}),o.parent().keydown(function(n){var l=!1,i=!1,a=e.element;if(!a.editor.isDragging&&!a.editor.inlineEditingIsActive){if(!t.isDisabled()){var r=a.editor.focusedElement;if(r&&n.ctrlKey)switch(n.which){case 67:r.copy(t);break;case 88:r.cut(t);break;case 86:r.paste(t)}}if(n.ctrlKey||n.shiftKey||n.altKey||46!=n.which?n.ctrlKey||n.shiftKey||n.altKey||32!=n.which&&27!=n.which||(o.find(".layout-panel-action-properties").first().click(),l=!0):(e["delete"](a),l=!0),"Content"==a.type&&(n.ctrlKey||n.shiftKey||n.altKey||13!=n.which||(o.find(".layout-panel-action-edit").first().click(),l=!0)),a.children&&(n.ctrlKey||n.shiftKey||!n.altKey||40!=n.which||(a.children.length>0&&a.children[0].setIsFocused(),l=!0),"Column"==a.type)){var c=!n.ctrlKey;37==n.which?(n.altKey&&a.expandLeft(c),n.shiftKey&&a.contractRight(c),l=!0):39==n.which&&(n.altKey&&a.contractLeft(c),n.shiftKey&&a.expandRight(c),l=!0)}a.parent&&(n.altKey&&38==n.which&&(a.parent.setIsFocused(),l=!0),"Row"==a.parent.type?n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?n.ctrlKey||n.shiftKey||n.altKey||39!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||37!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||39!=n.which||(a.moveDown(),l=!0):(a.moveUp(),i=!0,l=!0):(a.parent.moveFocusNextChild(a),l=!0):(a.parent.moveFocusPrevChild(a),l=!0):n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?n.ctrlKey||n.shiftKey||n.altKey||40!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||38!=n.which?!n.ctrlKey||n.shiftKey||n.altKey||40!=n.which||(a.moveDown(),l=!0):(a.moveUp(),i=!0,l=!0):(a.parent.moveFocusNextChild(a),l=!0):(a.parent.moveFocusPrevChild(a),l=!0)),l&&n.preventDefault(),n.stopPropagation(),e.$apply(),i&&window.setTimeout(function(){e.$apply(function(){a.editor.focusedElement.setIsFocused()})},100)}}),e.element.setIsFocusedEventHandlers.push(function(){o.parent().focus()}),e["delete"]=function(e){e["delete"]()}},configureForContainer:function(t,o){var n=t.element;t.getShowChildrenPlaceholder=function(){return 0===t.element.children.length&&!t.element.getIsDropTarget()},t.sortableOptions={cursor:"move",delay:150,disabled:n.getIsSealed(),distance:5,start:function(e,o){t.$apply(function(){n.setIsDropTarget(!0),n.editor.isDragging=!0}),o.placeholder.height(o.item.height()-4),o.placeholder.css("min-height",0)},stop:function(e,o){t.$apply(function(){n.editor.isDragging=!1,n.setIsDropTarget(!1)})},over:function(t,l){l.sender&&l.sender[0].isToolbox&&(l.sender[0].dropTargetTimeout&&(e.cancel(l.sender[0].dropTargetTimeout),l.sender[0].dropTargetTimeout=null),e(function(){if("Row"==n.type){var e=n.editor.dropTargetElement;e&&"Row"==e.type&&e.rollbackAddColumn()}n.setIsDropTarget(!1)}),l.sender[0].dropTargetTimeout=e(function(){if("Row"==n.type){var e=l.item.sortable.model,t=Math.floor(12/(n.children.length+1));e.width=t,e.offset=0,n.beginAddColumn(t);var a=_.max(_(o.find("> .layout-children > .layout-column:not(.ui-sortable-placeholder)")).map(function(e){return $(e).height()}));for(i=1;i<=12;i++)l.placeholder.removeClass("col-xs-"+i);l.placeholder.addClass("col-xs-"+e.width),a>0?(l.placeholder.height(a),l.placeholder.css("min-height",0)):(l.placeholder.height(0),l.placeholder.css("min-height",""))}n.setIsDropTarget(!0)},150))},receive:function(o,l){l.sender&&l.sender[0].isToolbox&&t.$apply(function(){var o=l.item.sortable.model;o&&("Row"==n.type&&n.commitAddColumn(),o.setEditor(n.editor),o.setParent(n),o.hasEditor&&t.$root.editElement(o).then(function(t){t.cancel||(o.data=t.element.data,o.setHtml&&o.setHtml(t.element.html)),e(function(){t.cancel?o["delete"]():o.setIsFocused(),n.setIsDropTarget(!1)})})),e(function(){n.setIsDropTarget(!1),o&&o.setIsFocused()})})}},t.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},t.getClasses=function(e){var t=["layout-element"];return e.children&&(t.push("layout-container"),e.getIsSealed()&&t.push("layout-container-sealed")),t.push("layout-"+e.type.toLowerCase()),e.dropTargetClass&&t.push(e.dropTargetClass),"Row"==e.type&&(t.push("row"),e.canAddColumn()||t.push("layout-row-full")),"Column"==e.type&&(t.push("col-xs-"+e.width),t.push("col-xs-offset-"+e.offset)),"Content"==e.type&&t.push("layout-content-"+e.contentTypeClass),e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t}}}}]),angular.module("LayoutEditor").directive("orcLayoutEditor",["environment",function(environment){return{restrict:"E",scope:{},controller:["$scope","$element","$attrs","$compile","clipboard",function($scope,$element,$attrs,$compile,clipboard){if(!$attrs.model)throw new Error("The 'model' attribute must evaluate to a LayoutEditor.Editor object.");$scope.element=eval($attrs.model),$scope.click=function(e,t){e.editor.isDragging||e.setIsFocused(),t.stopPropagation()},$scope.getClasses=function(e){var t=["layout-element","layout-container","layout-canvas"];return e.getIsActive()&&t.push("layout-element-active"),e.getIsFocused()&&t.push("layout-element-focused"),e.getIsSelected()&&t.push("layout-element-selected"),e.getIsDropTarget()&&t.push("layout-element-droptarget"),e.isTemplated&&t.push("layout-element-templated"),t};var layoutDesignerHost=$(".layout-designer").data("layout-designer-host");$scope.$root.layoutDesignerHost=layoutDesignerHost,layoutDesignerHost.element.on("replacecanvas",function(e,t){var o=$scope.element,n={data:t.canvas.data,htmlId:t.canvas.htmlId,htmlClass:t.canvas.htmlClass,htmlStyle:t.canvas.htmlStyle,isTemplated:t.canvas.isTemplated,children:t.canvas.children};layoutDesignerHost.editor=window.layoutEditor=new LayoutEditor.Editor(o.config,n);var l="",i=$compile(l)($scope);$(".layout-editor-holder").html(i)}),$scope.$root.editElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.editElement(e)},$scope.$root.addElement=function(e){var t=$scope.$root.layoutDesignerHost;return t.addElement(e)},$scope.toggleInlineEditing=function(){if($scope.element.inlineEditingIsActive)tinymce.remove("#layout-editor-"+$scope.$id+" .layout-content-markup"),$element.find(".layout-toolbar-container").hide(),$scope.element.inlineEditingIsActive=!1;else{$scope.element.inlineEditingIsActive=!0,$element.find(".layout-toolbar-container").show();var e="#layout-editor-"+$scope.$id+" .layout-html .layout-content-markup[data-templated=false]",t=$(e).first().attr("id");tinymce.init({selector:e,theme:"modern",schema:"html5",plugins:["advlist autolink lists link image charmap print preview hr anchor pagebreak","searchreplace wordcount visualblocks visualchars code fullscreen","insertdatetime media nonbreaking table contextmenu directionality","emoticons template paste textcolor colorpicker textpattern","fullscreen autoresize"],toolbar:"undo redo cut copy paste | bold italic | bullist numlist outdent indent formatselect | alignleft aligncenter alignright alignjustify ltr rtl | link unlink charmap | code fullscreen close",convert_urls:!1,valid_elements:"*[*]",extended_valid_elements:"script[type|defer|src|language]",statusbar:!1,skin:"orchardlightgray",inline:!0,fixed_toolbar_container:"#layout-editor-"+$scope.$id+" .layout-toolbar-container",init_instance_callback:function(e){e.id==t&&tinymce.execCommand("mceFocus",!1,e.id)}})}},$(document).on("cut copy paste",function(e){if(clipboard.wasInvoked())e.originalEvent.clipboardData.setData("text/plain",clipboard.getData("text/plain")),e.originalEvent.clipboardData.setData("text/json",clipboard.getData("text/json")),e.preventDefault();else{var t=$scope.element.focusedElement;t&&($scope.$apply(function(){switch(e.type){case"copy":t.copy(e.originalEvent.clipboardData);break;case"cut":t.cut(e.originalEvent.clipboardData);break;case"paste":t.paste(e.originalEvent.clipboardData)}}),window.setTimeout(function(){$scope.$apply(function(){$scope.element.focusedElement&&$scope.element.focusedElement.setIsFocused()})},100),e.preventDefault())}clipboard.disable()})}],templateUrl:environment.templateUrl("Editor"),replace:!0,link:function(e,t){t.find(".layout-toolbar-container").click(function(e){e.stopPropagation()}),t.mousedown(function(t){e.element.inlineEditingIsActive&&(t.preventDefault(),t.stopPropagation())}),$(window).click(function(t){e.element.inlineEditingIsActive||e.$apply(function(){e.element.activeElement=null,e.element.focusedElement=null})})}}}]),angular.module("LayoutEditor").directive("orcLayoutCanvas",["scopeConfigurator","environment",function(e,t){return{restrict:"E",scope:{element:"="},controller:["$scope","$element","$attrs",function(t,o,n){e.configureForElement(t,o),e.configureForContainer(t,o),t.sortableOptions.axis="y"}],templateUrl:t.templateUrl("Canvas"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutChild",["$compile",function(e){return{restrict:"E",scope:{element:"="},link:function(t,o){var n="",l=e(n)(t);$(o).replaceWith(l)}}}]),angular.module("LayoutEditor").directive("orcLayoutColumn",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Column"),replace:!0,link:function(e,t,o){t.find(".layout-column-resize-bar").draggable({axis:"x",helper:"clone",revert:!0,start:function(t,o){e.$apply(function(){e.element.editor.isResizing=!0})},drag:function(o,n){var l=t.parent(),i=l.width()/e.element.width,a=!o.ctrlKey;if($(o.target).hasClass("layout-column-resize-bar-left")){var r=n.offset.left-l.offset().left;-i>r&&e.element.canExpandLeft(a)?e.$apply(function(){e.element.expandLeft(a)}):r>i&&e.element.canContractLeft(a)&&e.$apply(function(){e.element.contractLeft(a)})}else if($(o.target).hasClass("layout-column-resize-bar-right")){var r=n.offset.left-l.width()-l.offset().left;r>i&&e.element.canExpandRight(a)?e.$apply(function(){e.element.expandRight(a)}):-i>r&&e.element.canContractRight(a)&&e.$apply(function(){e.element.contractRight(a)})}},stop:function(t,o){e.$apply(function(){e.element.editor.isResizing=!1})}})}}}]),angular.module("LayoutEditor").directive("orcLayoutContent",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Content"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutHtml",["$sce","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(o,n){t.configureForElement(o,n),o.edit=function(){o.$root.editElement(o.element).then(function(e){o.$apply(function(){e.cancel||(o.element.data=e.element.data,o.element.setHtml(e.element.html))})})},o.updateContent=function(e){o.element.setHtml(e.target.innerHTML)},o.element.setHtml=function(t){o.element.html=t,o.element.htmlUnsafe=e.trustAsHtml(t)},o.element.setHtml(o.element.html)}],templateUrl:o.templateUrl("Html"),replace:!0,link:function(e,t){t.find(".layout-content-markup").mousedown(function(t){e.element.editor.inlineEditingIsActive&&t.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutGrid",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="y"}],templateUrl:o.templateUrl("Grid"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutRow",["$compile","scopeConfigurator","environment",function(e,t,o){return{restrict:"E",scope:{element:"="},controller:["$scope","$element",function(e,o){t.configureForElement(e,o),t.configureForContainer(e,o),e.sortableOptions.axis="x",e.sortableOptions["ui-floating"]=!0}],templateUrl:o.templateUrl("Row"),replace:!0}}]),angular.module("LayoutEditor").directive("orcLayoutPopup",[function(){return{restrict:"A",link:function(e,t,o){var n=$(t),l=n.closest(".layout-popup-trigger"),i=n.closest(".layout-element");l.click(function(){n.toggle(),n.is(":visible")&&(n.position({my:o.orcLayoutPopupMy||"left top",at:o.orcLayoutPopupAt||"left bottom+4px",of:l}),n.find("input").first().focus())}),n.click(function(e){e.stopPropagation()}),i.click(function(e){n.hide()}),n.keydown(function(e){e.ctrlKey||e.shiftKey||e.altKey||27!=e.which||n.hide(),e.stopPropagation()}),n.on("cut copy paste",function(e){e.stopPropagation()})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolbox",["$compile","environment",function(e,t){return{restrict:"E",controller:["$scope","$element",function(e,t){e.resetElements=function(){e.gridElements=[LayoutEditor.Grid.from({toolboxIcon:"",toolboxLabel:"Grid",toolboxDescription:"Empty grid.",children:[]})],e.rowElements=[LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (1 column)",toolboxDescription:"Row with 1 column.",children:LayoutEditor.Column.times(1)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (2 columns)",toolboxDescription:"Row with 2 columns.",children:LayoutEditor.Column.times(2)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (3 columns)",toolboxDescription:"Row with 3 columns.",children:LayoutEditor.Column.times(3)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (4 columns)",toolboxDescription:"Row with 4 columns.",children:LayoutEditor.Column.times(4)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (6 columns)",toolboxDescription:"Row with 6 columns.",children:LayoutEditor.Column.times(6)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (12 columns)",toolboxDescription:"Row with 12 columns.",children:LayoutEditor.Column.times(12)}),LayoutEditor.Row.from({toolboxIcon:"",toolboxLabel:"Row (empty)",toolboxDescription:"Empty row.",children:[]})],e.columnElements=[LayoutEditor.Column.from({toolboxIcon:"",toolboxLabel:"Column",toolboxDescription:"Empty column.",width:1,offset:0,children:[]})],e.contentElementCategories=_(e.element.config.categories).map(function(e){return{name:e.name,elements:_(e.contentTypes).map(function(e){var t=e.type,o=LayoutEditor.factories[t]||LayoutEditor.factories.Content,n={isTemplated:!1,contentType:e.id,contentTypeLabel:e.label,contentTypeClass:e.typeClass,data:null,hasEditor:e.hasEditor,html:e.html},l=o(n);return l.toolboxIcon=e.icon||"",l.toolboxLabel=e.label,l.toolboxDescription=e.description,l})}})},e.resetElements(),e.getSortableOptions=function(o){var n,l,i=t.closest(".layout-editor").attr("id"),a=!1;switch(o){case"Grid":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-container layout-grid ui-sortable-placeholder";break;case"Row":n=[".layout-grid"],l="layout-element layout-container layout-row row ui-sortable-placeholder";break;case"Column":n=[".layout-row:not(.layout-row-full)"],l="layout-element layout-container layout-column ui-sortable-placeholder",a=!0;break;case"Content":n=[".layout-canvas",".layout-column",".layout-common-holder"],l="layout-element layout-content ui-sortable-placeholder"}return{cursor:"move",connectWith:_(n).map(function(e){return"#"+i+" "+e+":not(.layout-container-sealed) > .layout-element-wrapper > .layout-children"}).join(", "),placeholder:l,"ui-floating":a,create:function(e,t){e.target.isToolbox=!0},start:function(t,o){e.$apply(function(){e.element.isDragging=!0})},stop:function(t,o){e.$apply(function(){e.element.isDragging=!1,e.resetElements()})},over:function(t,o){e.$apply(function(){e.element.canvas.setIsDropTarget(!1)})}}};var o="layoutToolboxCategory_Layout_IsCollapsed";e.layoutIsCollapsed="true"===$.cookie(o),e.toggleLayoutIsCollapsed=function(t){e.layoutIsCollapsed=!e.layoutIsCollapsed,$.cookie(o,e.layoutIsCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("Toolbox"),replace:!0,link:function(e,t){var o=t.find(".layout-toolbox");$(window).on("resize scroll",function(e){var n=t.parent().find(".layout-canvas"),l=!!n&&n.height()>o.height(),i=$(window).scrollTop();l&&i>t.offset().top+t.height()-o.height()?(o.addClass("sticky-bottom"),o.removeClass("sticky-top")):l&&i>t.offset().top?(o.addClass("sticky-top"),o.removeClass("sticky-bottom")):(o.removeClass("sticky-top"),o.removeClass("sticky-bottom"))})}}}]),angular.module("LayoutEditor").directive("orcLayoutToolboxGroup",["$compile","environment",function(e,t){return{restrict:"E",scope:{category:"="},controller:["$scope","$element",function(e,t){var o="layoutToolboxCategory_"+e.category.name+"_IsCollapsed";e.isCollapsed="true"===$.cookie(o),e.toggleIsCollapsed=function(t){e.isCollapsed=!e.isCollapsed,$.cookie(o,e.isCollapsed,{expires:365}),t.preventDefault(),t.stopPropagation()}}],templateUrl:t.templateUrl("ToolboxGroup"),replace:!0}}]); \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Directives/Editor.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Directives/Editor.js index 43207591e..8595615e8 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Directives/Editor.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Directives/Editor.js @@ -114,34 +114,46 @@ }; $(document).on("cut copy paste", function (e) { - // The real clipboard is supported, so disable the peudo clipboard. - clipboard.disable(); - var focusedElement = $scope.element.focusedElement; - if (!!focusedElement) { - $scope.$apply(function () { - switch (e.type) { - case "copy": - focusedElement.copy(e.originalEvent.clipboardData); - break; - case "cut": - focusedElement.cut(e.originalEvent.clipboardData); - break; - case "paste": - focusedElement.paste(e.originalEvent.clipboardData); - break; - } - }); - - // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost. - window.setTimeout(function () { - $scope.$apply(function () { - if (!!$scope.element.focusedElement) - $scope.element.focusedElement.setIsFocused(); - }); - }, 100); - + // If the pseudo clipboard was already invoked (which happens on the first clipboard + // operation after page load even if native clipboard support exists) then sit this + // one operation out, but make sure whatever is on the pseudo clipboard gets migrated + // to the native clipboard for subsequent operations. + if (clipboard.wasInvoked()) { + e.originalEvent.clipboardData.setData("text/plain", clipboard.getData("text/plain")); + e.originalEvent.clipboardData.setData("text/json", clipboard.getData("text/json")); e.preventDefault(); } + else { + var focusedElement = $scope.element.focusedElement; + if (!!focusedElement) { + $scope.$apply(function () { + switch (e.type) { + case "copy": + focusedElement.copy(e.originalEvent.clipboardData); + break; + case "cut": + focusedElement.cut(e.originalEvent.clipboardData); + break; + case "paste": + focusedElement.paste(e.originalEvent.clipboardData); + break; + } + }); + + // HACK: Workaround because of how Angular treats the DOM when elements are shifted around - input focus is sometimes lost. + window.setTimeout(function () { + $scope.$apply(function () { + if (!!$scope.element.focusedElement) + $scope.element.focusedElement.setIsFocused(); + }); + }, 100); + + e.preventDefault(); + } + } + + // Native clipboard support obviously exists, so disable the peudo clipboard from now on. + clipboard.disable(); }); } ], diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/Clipboard.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/Clipboard.js index b0be0d474..013672249 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/Clipboard.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/Clipboard.js @@ -3,17 +3,29 @@ var Clipboard = function () { var self = this; - this.clipboardData = {}; - this.setData = function(contentType, data, realClipBoard) { - self.clipboardData[contentType] = data; - }; - this.getData = function (contentType, realClipBoard) { - return self.clipboardData[contentType]; - }; + this._clipboardData = {}; + this._isDisabled = false; + this._wasInvoked = false; - this.disable = function() { - this.disabled = true; + this.setData = function(contentType, data) { + self._clipboardData[contentType] = data; + self._wasInvoked = true; }; + this.getData = function (contentType) { + return self._clipboardData[contentType]; + self._wasInvoked = true; + }; + this.disable = function() { + self._isDisabled = true; + self._wasInvoked = false; + self._clipboardData = {}; + }; + this.isDisabled = function () { + return self._isDisabled; + } + this.wasInvoked = function () { + return self._wasInvoked; + } } LayoutEditor.Clipboard = new Clipboard(); @@ -25,7 +37,9 @@ return { setData: LayoutEditor.Clipboard.setData, getData: LayoutEditor.Clipboard.getData, - disable: LayoutEditor.Clipboard.disable + disable: LayoutEditor.Clipboard.disable, + isDisabled: LayoutEditor.Clipboard.isDisabled, + wasInvoked: LayoutEditor.Clipboard.wasInvoked }; } ]); diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/ScopeConfigurator.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/ScopeConfigurator.js index a6b450e32..d7da1feb8 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/ScopeConfigurator.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/LayoutEditor/Services/ScopeConfigurator.js @@ -15,12 +15,11 @@ var resetFocus = false; var element = $scope.element; - if (element.editor.isDragging || element.editor.inlineEditingIsActive) return; - // If the "real" clipboard works, then the pseudo-clipboard will have been disabled. - if (!clipboard.disabled) { + // If native clipboard support exists, the pseudo-clipboard will have been disabled. + if (!clipboard.isDisabled()) { var focusedElement = element.editor.focusedElement; if (!!focusedElement) { // Pseudo clipboard handling for browsers not allowing real clipboard operations. diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js index 0175273b2..205f708ad 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.js @@ -232,7 +232,6 @@ var LayoutEditor; this.copy = function (clipboardData) { var text = this.getInnerText(); clipboardData.setData("text/plain", text); - console.log(text); var data = this.toObject(); var json = JSON.stringify(data, null, "\t"); @@ -987,4 +986,4 @@ var LayoutEditor; LayoutEditor.registerFactory("Html", function(value) { return LayoutEditor.Html.from(value); }); })(jQuery, LayoutEditor || (LayoutEditor = {})); -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Helpers.js","Editor.js","Element.js","Container.js","Canvas.js","Grid.js","Row.js","Column.js","Content.js","Html.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACjLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"Models.js","sourcesContent":["var LayoutEditor;\n(function (LayoutEditor) {\n\n    Array.prototype.move = function (from, to) {\n        this.splice(to, 0, this.splice(from, 1)[0]);\n    };\n\n    LayoutEditor.childrenFrom = function(values) {\n        return _(values).map(function(value) {\n            return LayoutEditor.elementFrom(value);\n        });\n    };\n\n    var registerFactory = LayoutEditor.registerFactory = function(type, factory) {\n        var factories = LayoutEditor.factories = LayoutEditor.factories || {};\n        factories[type] = factory;\n    };\n\n    registerFactory(\"Grid\", function(value) { return LayoutEditor.Grid.from(value); });\n    registerFactory(\"Row\", function(value) { return LayoutEditor.Row.from(value); });\n    registerFactory(\"Column\", function(value) { return LayoutEditor.Column.from(value); });\n    registerFactory(\"Content\", function(value) { return LayoutEditor.Content.from(value); });\n\n    LayoutEditor.elementFrom = function (value) {\n        var factory = LayoutEditor.factories[value.type];\n\n        if (!factory)\n            throw new Error(\"No element with type \\\"\" + value.type + \"\\\" was found.\");\n\n        var element = factory(value);\n        return element;\n    };\n\n    LayoutEditor.setModel = function (elementSelector, model) {\n        $(elementSelector).scope().element = model;\n    };\n\n    LayoutEditor.getModel = function (elementSelector) {\n        return $(elementSelector).scope().element;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Editor = function (config, canvasData) {\n        this.config = config;\n        this.canvas = LayoutEditor.Canvas.from(canvasData);\n        this.initialState = JSON.stringify(this.canvas.toObject());\n        this.activeElement = null;\n        this.focusedElement = null;\n        this.dropTargetElement = null;\n        this.isDragging = false;\n        this.inlineEditingIsActive = false;\n        this.isResizing = false;\n\n        this.resetToolboxElements = function () {\n            this.toolboxElements = [\n                LayoutEditor.Row.from({\n                    children: []\n                })\n            ];\n        };\n\n        this.isDirty = function() {\n            var currentState = JSON.stringify(this.canvas.toObject());\n            return this.initialState != currentState;\n        };\n\n        this.resetToolboxElements();\n        this.canvas.setEditor(this);\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));\n","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Element = function (type, data, htmlId, htmlClass, htmlStyle, isTemplated) {\n        if (!type)\n            throw new Error(\"Parameter 'type' is required.\");\n\n        this.type = type;\n        this.data = data;\n        this.htmlId = htmlId;\n        this.htmlClass = htmlClass;\n        this.htmlStyle = htmlStyle;\n        this.isTemplated = isTemplated;\n\n        this.editor = null;\n        this.parent = null;\n        this.setIsFocusedEventHandlers = [];\n\n        this.setEditor = function (editor) {\n            this.editor = editor;\n            if (!!this.children && _.isArray(this.children)) {\n                _(this.children).each(function (child) {\n                    child.setEditor(editor);\n                });\n            }\n        };\n\n        this.setParent = function(parentElement) {\n            this.parent = parentElement;\n\n            if (!!this.parent.linkChild)\n                this.parent.linkChild(this);\n        };\n\n        this.setIsTemplated = function (value) {\n            this.isTemplated = value;\n            if (!!this.children && _.isArray(this.children)) {\n                _(this.children).each(function (child) {\n                    child.setIsTemplated(value);\n                });\n            }\n        };\n\n        this.getIsActive = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.activeElement === this && !this.getIsFocused();\n        };\n\n        this.setIsActive = function (value) {\n            if (!this.editor)\n                return;\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\n                return;\n\n            if (value)\n                this.editor.activeElement = this;\n            else\n                this.editor.activeElement = this.parent;\n        };\n\n        this.getIsFocused = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.focusedElement === this;\n        };\n\n        this.setIsFocused = function () {\n            if (!this.editor)\n            \treturn;\n            if (this.isTemplated)\n            \treturn;\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\n                return;\n\n            this.editor.focusedElement = this;\n            _(this.setIsFocusedEventHandlers).each(function (item) {\n                try {\n                    item();\n                }\n                catch (ex) {\n                    // Ignore.\n                }\n            });\n        };\n\n        this.getIsSelected = function () {\n            if (this.getIsFocused())\n                return true;\n\n            if (!!this.children && _.isArray(this.children)) {\n                return _(this.children).any(function(child) {\n                    return child.getIsSelected();\n                });\n            }\n\n            return false;\n        };\n\n        this.getIsDropTarget = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.dropTargetElement === this;\n        }\n\n        this.setIsDropTarget = function (value) {\n            if (!this.editor)\n                return;\n            if (value)\n                this.editor.dropTargetElement = this;\n            else\n                this.editor.dropTargetElement = null;\n        };\n\n        this.delete = function () {\n            if (!!this.parent)\n                this.parent.deleteChild(this);\n        };\n\n        this.canMoveUp = function () {\n            if (!this.parent)\n                return false;\n            return this.parent.canMoveChildUp(this);\n        };\n\n        this.moveUp = function () {\n            if (!!this.parent)\n                this.parent.moveChildUp(this);\n        };\n\n        this.canMoveDown = function () {\n            if (!this.parent)\n                return false;\n            return this.parent.canMoveChildDown(this);\n        };\n\n        this.moveDown = function () {\n            if (!!this.parent)\n                this.parent.moveChildDown(this);\n        };\n\n        this.elementToObject = function () {\n            return {\n                type: this.type,\n                data: this.data,\n                htmlId: this.htmlId,\n                htmlClass: this.htmlClass,\n                htmlStyle: this.htmlStyle,\n                isTemplated: this.isTemplated\n            };\n        };\n\n        this.getEditorObject = function() {\n            return {};\n        };\n\n        this.copy = function (clipboardData) {\n            var text = this.getInnerText();\n            clipboardData.setData(\"text/plain\", text);\n            console.log(text);\n\n            var data = this.toObject();\n            var json = JSON.stringify(data, null, \"\\t\");\n            clipboardData.setData(\"text/json\", json);\n        };\n\n        this.cut = function (clipboardData) {\n            this.copy(clipboardData);\n            this.delete();\n        };\n\n        this.paste = function (clipboardData) {\n            if (!!this.parent)\n                this.parent.paste(clipboardData);\n        };\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Container = function (allowedChildTypes, children) {\n\n        this.allowedChildTypes = allowedChildTypes;\n        this.children = children;\n        this.isContainer = true;\n\n        var _self = this;\n\n        this.setChildren = function (children) {\n            this.children = children;\n            _(this.children).each(function (child) {\n                child.parent = _self;\n            });\n        };\n\n        this.setChildren(children);\n\n        this.getIsSealed = function () {\n            return _(this.children).any(function (child) {\n                return child.isTemplated;\n            });\n        };\n\n        this.addChild = function (child) {\n            if (!_(this.children).contains(child) && (_(this.allowedChildTypes).contains(child.type) || child.isContainable))\n                this.children.push(child);\n            child.setEditor(this.editor);\n            child.setIsTemplated(false);\n            child.parent = this;\n        };\n\n        this.deleteChild = function (child) {\n            var index = _(this.children).indexOf(child);\n            if (index >= 0) {\n                this.children.splice(index, 1);\n                if (child.getIsActive())\n                    this.editor.activeElement = null;\n                if (child.getIsFocused()) {\n                    // If the deleted child was focused, try to set new focus to the most appropriate sibling or parent.\n                    if (this.children.length > index)\n                        this.children[index].setIsFocused();\n                    else if (index > 0)\n                        this.children[index - 1].setIsFocused();\n                    else\n                        this.setIsFocused();\n                }\n            }\n        };\n\n        this.moveFocusPrevChild = function (child) {\n            if (this.children.length < 2)\n                return;\n            var index = _(this.children).indexOf(child);\n            if (index > 0)\n                this.children[index - 1].setIsFocused();\n        };\n\n        this.moveFocusNextChild = function (child) {\n            if (this.children.length < 2)\n                return;\n            var index = _(this.children).indexOf(child);\n            if (index < this.children.length - 1)\n                this.children[index + 1].setIsFocused();\n        };\n\n        this.insertChild = function (child, afterChild) {\n            if (!_(this.children).contains(child)) {\n                var index = Math.max(_(this.children).indexOf(afterChild), 0);\n                this.children.splice(index + 1, 0, child);\n                child.setEditor(this.editor);\n                child.parent = this;\n            }\n        };\n\n        this.moveChildUp = function (child) {\n            if (!this.canMoveChildUp(child))\n                return;\n            var index = _(this.children).indexOf(child);\n            this.children.move(index, index - 1);\n        };\n\n        this.moveChildDown = function (child) {\n            if (!this.canMoveChildDown(child))\n                return;\n            var index = _(this.children).indexOf(child);\n            this.children.move(index, index + 1);\n        };\n\n        this.canMoveChildUp = function (child) {\n            var index = _(this.children).indexOf(child);\n            return index > 0;\n        };\n\n        this.canMoveChildDown = function (child) {\n            var index = _(this.children).indexOf(child);\n            return index < this.children.length - 1;\n        };\n\n        this.childrenToObject = function () {\n            return _(this.children).map(function (child) {\n                return child.toObject();\n            });\n        };\n\n        this.getInnerText = function () {\n            return _(this.children).reduce(function (memo, child) {\n                return memo + \"\\n\" + child.getInnerText();\n            }, \"\");\n        }\n\n        this.paste = function (clipboardData) {\n            var json = clipboardData.getData(\"text/json\");\n            if (!!json) {\n                var data = JSON.parse(json);\n                var child = LayoutEditor.elementFrom(data);\n                this.pasteChild(child);\n            }\n        };\n\n        this.pasteChild = function (child) {\n            if (_(this.allowedChildTypes).contains(child.type) || child.isContainable) {\n                this.addChild(child);\n                child.setIsFocused();\n            }\n            else if (!!this.parent)\n                this.parent.pasteChild(child);\n        }\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Canvas = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Canvas\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Canvas.from = function (value) {\n        return new LayoutEditor.Canvas(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));\n","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Grid = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Grid\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Row\"], children);\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Grid.from = function (value) {\n        var result = new LayoutEditor.Grid(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Row = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Row\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Column\"], children);\n\n        var _self = this;\n\n        function _getTotalColumnsWidth() {\n            return _(_self.children).reduce(function (memo, child) {\n                return memo + child.offset + child.width;\n            }, 0);\n        }\n\n        // Implements a simple algorithm to distribute space (either positive or negative)\n        // between the child columns of the row. Negative space is distributed when making\n        // room for a new column (e.c. clipboard paste or dropping from the toolbox) while\n        // positive space is distributed when filling the grap of a removed column.\n        function _distributeSpace(space) {\n            if (space == 0)\n                return true;\n             \n            var undistributedSpace = space;\n\n            if (undistributedSpace < 0) {\n                var vacantSpace = 12 - _getTotalColumnsWidth();\n                undistributedSpace += vacantSpace;\n                if (undistributedSpace > 0)\n                    undistributedSpace = 0;\n            }\n\n            // If space is negative, try to decrease offsets first.\n            while (undistributedSpace < 0 && _(_self.children).any(function (column) { return column.offset > 0; })) { // While there is still offset left to remove.\n                for (i = 0; i < _self.children.length && undistributedSpace < 0; i++) {\n                    var column = _self.children[i];\n                    if (column.offset > 0) {\n                        column.offset--;\n                        undistributedSpace++;\n                    }\n                }\n            }\n\n            function hasWidth(column) {\n                if (undistributedSpace > 0)\n                    return column.width < 12;\n                else if (undistributedSpace < 0)\n                    return column.width > 1;\n                return false;\n            }\n\n            // Try to distribute remaining space (could be negative or positive) using widths.\n            while (undistributedSpace != 0) {\n                // Any more column width available for distribution?\n                if (!_(_self.children).any(hasWidth))\n                    break;\n                for (i = 0; i < _self.children.length && undistributedSpace != 0; i++) {\n                    var column = _self.children[i % _self.children.length];\n                    if (hasWidth(column)) {\n                        var delta = undistributedSpace / Math.abs(undistributedSpace);\n                        column.width += delta;\n                        undistributedSpace -= delta;\n                    }\n                }                \n            }\n\n            return undistributedSpace == 0;\n        }\n\n        var _isAddingColumn = false;\n\n        this.canAddColumn = function () {\n            return this.children.length < 12;\n        };\n\n        this.beginAddColumn = function (newColumnWidth) {\n            if (!!_isAddingColumn)\n                throw new Error(\"Column add operation is already in progress.\")\n            _(this.children).each(function (column) {\n                column.beginChange();\n            });\n            if (_distributeSpace(-newColumnWidth)) {\n                _isAddingColumn = true;\n                return true;\n            }\n            _(this.children).each(function (column) {\n                column.rollbackChange();\n            });\n            return false;\n        };\n\n        this.commitAddColumn = function () {\n            if (!_isAddingColumn)\n                throw new Error(\"No column add operation in progress.\")\n            _(this.children).each(function (column) {\n                column.commitChange();\n            });\n            _isAddingColumn = false;\n        };\n\n        this.rollbackAddColumn = function () {\n            if (!_isAddingColumn)\n                throw new Error(\"No column add operation in progress.\")\n            _(this.children).each(function (column) {\n                column.rollbackChange();\n            });\n            _isAddingColumn = false;\n        };\n\n        var _baseDeleteChild = this.deleteChild;\n        this.deleteChild = function (column) { \n            var width = column.width;\n            _baseDeleteChild.call(this, column);\n            _distributeSpace(width);\n        };\n\n        this.canContractColumnRight = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0)\n                return column.width > 1;\n            return false;\n        };\n\n        this.contractColumnRight = function (column, connectAdjacent) {\n            if (!this.canContractColumnRight(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width > 1) {\n                    column.width--;\n                    if (this.children.length > index + 1) {\n                        var nextColumn = this.children[index + 1];\n                        if (connectAdjacent && nextColumn.offset == 0)\n                            nextColumn.width++;\n                        else\n                            nextColumn.offset++;\n                    }\n                }\n            }\n        };\n\n        this.canExpandColumnRight = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width >= 12)\n                    return false;\n                if (this.children.length > index + 1) {\n                    var nextColumn = this.children[index + 1];\n                    if (connectAdjacent && nextColumn.offset == 0)\n                        return nextColumn.width > 1;\n                    else\n                        return nextColumn.offset > 0;\n                }\n                return _getTotalColumnsWidth() < 12;\n            }\n            return false;\n        };\n\n        this.expandColumnRight = function (column, connectAdjacent) {\n            if (!this.canExpandColumnRight(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (this.children.length > index + 1) {\n                    var nextColumn = this.children[index + 1];\n                    if (connectAdjacent && nextColumn.offset == 0)\n                        nextColumn.width--;\n                    else\n                        nextColumn.offset--;\n                }\n                column.width++;\n            }\n        };\n\n        this.canExpandColumnLeft = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width >= 12)\n                    return false;\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        return prevColumn.width > 1;\n                }\n                return column.offset > 0;\n            }\n            return false;\n        };\n\n        this.expandColumnLeft = function (column, connectAdjacent) {\n            if (!this.canExpandColumnLeft(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        prevColumn.width--;\n                    else\n                        column.offset--;\n                }\n                else\n                    column.offset--;\n                column.width++;\n            }\n        };\n\n        this.canContractColumnLeft = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0)\n                return column.width > 1;\n            return false;\n        };\n\n        this.contractColumnLeft = function (column, connectAdjacent) {\n            if (!this.canContractColumnLeft(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        prevColumn.width++;\n                    else\n                        column.offset++;\n                }\n                else\n                    column.offset++;\n                column.width--;\n            }\n        };\n\n        this.evenColumns = function () {\n            if (this.children.length == 0)\n                return;\n\n            var evenWidth = Math.floor(12 / this.children.length);\n            _(this.children).each(function (column) {\n                column.width = evenWidth;\n                column.offset = 0;\n            });\n\n            var rest = 12 % this.children.length;\n            if (rest > 0)\n                _distributeSpace(rest);\n        };\n\n        var _basePasteChild = this.pasteChild;\n        this.pasteChild = function (child) {\n            if (child.type == \"Column\") {\n                if (this.beginAddColumn(child.width)) {\n                    this.commitAddColumn();\n                    _basePasteChild.call(this, child)\n                }\n            }\n            else if (!!this.parent)\n                this.parent.pasteChild(child);\n        }\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Row.from = function (value) {\n        var result = new LayoutEditor.Row(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Column = function (data, htmlId, htmlClass, htmlStyle, isTemplated, width, offset, children) {\n        LayoutEditor.Element.call(this, \"Column\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\n\n        this.width = width;\n        this.offset = offset;\n\n        var _hasPendingChange = false;\n        var _origWidth = 0;\n        var _origOffset = 0;\n\n        this.beginChange = function () {\n            if (!!_hasPendingChange)\n                throw new Error(\"Column already has a pending change.\")\n            _hasPendingChange = true;\n            _origWidth = this.width;\n            _origOffset = this.offset;\n        };\n\n        this.commitChange = function () {\n            if (!_hasPendingChange)\n                throw new Error(\"Column has no pending change.\")\n            _origWidth = 0;\n            _origOffset = 0;\n            _hasPendingChange = false;\n        };\n\n        this.rollbackChange = function () {\n            if (!_hasPendingChange)\n                throw new Error(\"Column has no pending change.\")\n            this.width = _origWidth;\n            this.offset = _origOffset;\n            _origWidth = 0;\n            _origOffset = 0;\n            _hasPendingChange = false;\n        };\n\n        this.canSplit = function () {\n            return this.width > 1;\n        };\n\n        this.split = function () {\n            if (!this.canSplit())\n                return;\n\n            var newColumnWidth = Math.floor(this.width / 2);\n            var newColumn = LayoutEditor.Column.from({\n                data: null,\n                htmlId: null,\n                htmlClass: null,\n                htmlStyle: null,\n                width: newColumnWidth,\n                offset: 0,\n                children: []\n            });\n            \n            this.width = this.width - newColumnWidth;\n            this.parent.insertChild(newColumn, this);\n            newColumn.setIsFocused();\n        };\n\n        this.canContractRight = function (connectAdjacent) {\n            return this.parent.canContractColumnRight(this, connectAdjacent);\n        };\n\n        this.contractRight = function (connectAdjacent) {\n            this.parent.contractColumnRight(this, connectAdjacent);\n        };\n\n        this.canExpandRight = function (connectAdjacent) {\n            return this.parent.canExpandColumnRight(this, connectAdjacent);\n        };\n\n        this.expandRight = function (connectAdjacent) {\n            this.parent.expandColumnRight(this, connectAdjacent);\n        };\n\n        this.canExpandLeft = function (connectAdjacent) {\n            return this.parent.canExpandColumnLeft(this, connectAdjacent);\n        };\n\n        this.expandLeft = function (connectAdjacent) {\n            this.parent.expandColumnLeft(this, connectAdjacent);\n        };\n\n        this.canContractLeft = function (connectAdjacent) {\n            return this.parent.canContractColumnLeft(this, connectAdjacent);\n        };\n\n        this.contractLeft = function (connectAdjacent) {\n            this.parent.contractColumnLeft(this, connectAdjacent);\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.width = this.width;\n            result.offset = this.offset;\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Column.from = function (value) {\n        var result = new LayoutEditor.Column(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.width,\n            value.offset,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n    LayoutEditor.Column.times = function (value) {\n        return _.times(value, function (n) {\n            return LayoutEditor.Column.from({\n                data: null,\n                htmlId: null,\n                htmlClass: null,\n                isTemplated: false,\n                width: 12 / value,\n                offset: 0,\n                children: []\n            });\n        });\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Content = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor) {\n        LayoutEditor.Element.call(this, \"Content\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n\n        this.contentType = contentType;\n        this.contentTypeLabel = contentTypeLabel;\n        this.contentTypeClass = contentTypeClass;\n        this.html = html;\n        this.hasEditor = hasEditor;\n\n        this.getInnerText = function () {\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\n        };\n\n        // This function will be overwritten by the Content directive.\n        this.setHtml = function (html) {\n            this.html = html;\n            this.htmlUnsafe = html;\n        }\n\n        this.toObject = function () {\n            return {\n                \"type\": \"Content\"\n            };\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.contentType = this.contentType;\n            result.contentTypeLabel = this.contentTypeLabel;\n            result.contentTypeClass = this.contentTypeClass;\n            result.html = this.html;\n            result.hasEditor = hasEditor;\n            return result;\n        };\n\n        this.setHtml(html);\n    };\n\n    LayoutEditor.Content.from = function (value) {\n        var result = new LayoutEditor.Content(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.contentType,\n            value.contentTypeLabel,\n            value.contentTypeClass,\n            value.html,\n            value.hasEditor);\n\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function ($, LayoutEditor) {\n\n    LayoutEditor.Html = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor) {\n        LayoutEditor.Element.call(this, \"Html\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n\n        this.contentType = contentType;\n        this.contentTypeLabel = contentTypeLabel;\n        this.contentTypeClass = contentTypeClass;\n        this.html = html;\n        this.hasEditor = hasEditor;\n        this.isContainable = true;\n\n        this.getInnerText = function () {\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\n        };\n\n        // This function will be overwritten by the Content directive.\n        this.setHtml = function (html) {\n            this.html = html;\n            this.htmlUnsafe = html;\n        }\n\n        this.toObject = function () {\n            return {\n                \"type\": \"Html\"\n            };\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.contentType = this.contentType;\n            result.contentTypeLabel = this.contentTypeLabel;\n            result.contentTypeClass = this.contentTypeClass;\n            result.html = this.html;\n            result.hasEditor = hasEditor;\n            return result;\n        };\n\n        var getEditorObject = this.getEditorObject;\n        this.getEditorObject = function () {\n            var dto = getEditorObject();\n            return $.extend(dto, {\n                Content: this.html\n            });\n        }\n\n        this.setHtml(html);\n    };\n\n    LayoutEditor.Html.from = function (value) {\n        var result = new LayoutEditor.Html(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.contentType,\n            value.contentTypeLabel,\n            value.contentTypeClass,\n            value.html,\n            value.hasEditor);\n\n        return result;\n    };\n\n    LayoutEditor.registerFactory(\"Html\", function(value) { return LayoutEditor.Html.from(value); });\n\n})(jQuery, LayoutEditor || (LayoutEditor = {}));"],"sourceRoot":"/source/"} \ No newline at end of file +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["Helpers.js","Editor.js","Element.js","Container.js","Canvas.js","Grid.js","Row.js","Column.js","Content.js","Html.js"],"names":[],"mappings":"AAAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChCA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AChLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACpIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AC5RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACvIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;ACzDA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"Models.js","sourcesContent":["var LayoutEditor;\n(function (LayoutEditor) {\n\n    Array.prototype.move = function (from, to) {\n        this.splice(to, 0, this.splice(from, 1)[0]);\n    };\n\n    LayoutEditor.childrenFrom = function(values) {\n        return _(values).map(function(value) {\n            return LayoutEditor.elementFrom(value);\n        });\n    };\n\n    var registerFactory = LayoutEditor.registerFactory = function(type, factory) {\n        var factories = LayoutEditor.factories = LayoutEditor.factories || {};\n        factories[type] = factory;\n    };\n\n    registerFactory(\"Grid\", function(value) { return LayoutEditor.Grid.from(value); });\n    registerFactory(\"Row\", function(value) { return LayoutEditor.Row.from(value); });\n    registerFactory(\"Column\", function(value) { return LayoutEditor.Column.from(value); });\n    registerFactory(\"Content\", function(value) { return LayoutEditor.Content.from(value); });\n\n    LayoutEditor.elementFrom = function (value) {\n        var factory = LayoutEditor.factories[value.type];\n\n        if (!factory)\n            throw new Error(\"No element with type \\\"\" + value.type + \"\\\" was found.\");\n\n        var element = factory(value);\n        return element;\n    };\n\n    LayoutEditor.setModel = function (elementSelector, model) {\n        $(elementSelector).scope().element = model;\n    };\n\n    LayoutEditor.getModel = function (elementSelector) {\n        return $(elementSelector).scope().element;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Editor = function (config, canvasData) {\n        this.config = config;\n        this.canvas = LayoutEditor.Canvas.from(canvasData);\n        this.initialState = JSON.stringify(this.canvas.toObject());\n        this.activeElement = null;\n        this.focusedElement = null;\n        this.dropTargetElement = null;\n        this.isDragging = false;\n        this.inlineEditingIsActive = false;\n        this.isResizing = false;\n\n        this.resetToolboxElements = function () {\n            this.toolboxElements = [\n                LayoutEditor.Row.from({\n                    children: []\n                })\n            ];\n        };\n\n        this.isDirty = function() {\n            var currentState = JSON.stringify(this.canvas.toObject());\n            return this.initialState != currentState;\n        };\n\n        this.resetToolboxElements();\n        this.canvas.setEditor(this);\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));\n","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Element = function (type, data, htmlId, htmlClass, htmlStyle, isTemplated) {\n        if (!type)\n            throw new Error(\"Parameter 'type' is required.\");\n\n        this.type = type;\n        this.data = data;\n        this.htmlId = htmlId;\n        this.htmlClass = htmlClass;\n        this.htmlStyle = htmlStyle;\n        this.isTemplated = isTemplated;\n\n        this.editor = null;\n        this.parent = null;\n        this.setIsFocusedEventHandlers = [];\n\n        this.setEditor = function (editor) {\n            this.editor = editor;\n            if (!!this.children && _.isArray(this.children)) {\n                _(this.children).each(function (child) {\n                    child.setEditor(editor);\n                });\n            }\n        };\n\n        this.setParent = function(parentElement) {\n            this.parent = parentElement;\n\n            if (!!this.parent.linkChild)\n                this.parent.linkChild(this);\n        };\n\n        this.setIsTemplated = function (value) {\n            this.isTemplated = value;\n            if (!!this.children && _.isArray(this.children)) {\n                _(this.children).each(function (child) {\n                    child.setIsTemplated(value);\n                });\n            }\n        };\n\n        this.getIsActive = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.activeElement === this && !this.getIsFocused();\n        };\n\n        this.setIsActive = function (value) {\n            if (!this.editor)\n                return;\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\n                return;\n\n            if (value)\n                this.editor.activeElement = this;\n            else\n                this.editor.activeElement = this.parent;\n        };\n\n        this.getIsFocused = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.focusedElement === this;\n        };\n\n        this.setIsFocused = function () {\n            if (!this.editor)\n            \treturn;\n            if (this.isTemplated)\n            \treturn;\n            if (this.editor.isDragging || this.editor.inlineEditingIsActive || this.editor.isResizing)\n                return;\n\n            this.editor.focusedElement = this;\n            _(this.setIsFocusedEventHandlers).each(function (item) {\n                try {\n                    item();\n                }\n                catch (ex) {\n                    // Ignore.\n                }\n            });\n        };\n\n        this.getIsSelected = function () {\n            if (this.getIsFocused())\n                return true;\n\n            if (!!this.children && _.isArray(this.children)) {\n                return _(this.children).any(function(child) {\n                    return child.getIsSelected();\n                });\n            }\n\n            return false;\n        };\n\n        this.getIsDropTarget = function () {\n            if (!this.editor)\n                return false;\n            return this.editor.dropTargetElement === this;\n        }\n\n        this.setIsDropTarget = function (value) {\n            if (!this.editor)\n                return;\n            if (value)\n                this.editor.dropTargetElement = this;\n            else\n                this.editor.dropTargetElement = null;\n        };\n\n        this.delete = function () {\n            if (!!this.parent)\n                this.parent.deleteChild(this);\n        };\n\n        this.canMoveUp = function () {\n            if (!this.parent)\n                return false;\n            return this.parent.canMoveChildUp(this);\n        };\n\n        this.moveUp = function () {\n            if (!!this.parent)\n                this.parent.moveChildUp(this);\n        };\n\n        this.canMoveDown = function () {\n            if (!this.parent)\n                return false;\n            return this.parent.canMoveChildDown(this);\n        };\n\n        this.moveDown = function () {\n            if (!!this.parent)\n                this.parent.moveChildDown(this);\n        };\n\n        this.elementToObject = function () {\n            return {\n                type: this.type,\n                data: this.data,\n                htmlId: this.htmlId,\n                htmlClass: this.htmlClass,\n                htmlStyle: this.htmlStyle,\n                isTemplated: this.isTemplated\n            };\n        };\n\n        this.getEditorObject = function() {\n            return {};\n        };\n\n        this.copy = function (clipboardData) {\n            var text = this.getInnerText();\n            clipboardData.setData(\"text/plain\", text);\n\n            var data = this.toObject();\n            var json = JSON.stringify(data, null, \"\\t\");\n            clipboardData.setData(\"text/json\", json);\n        };\n\n        this.cut = function (clipboardData) {\n            this.copy(clipboardData);\n            this.delete();\n        };\n\n        this.paste = function (clipboardData) {\n            if (!!this.parent)\n                this.parent.paste(clipboardData);\n        };\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Container = function (allowedChildTypes, children) {\n\n        this.allowedChildTypes = allowedChildTypes;\n        this.children = children;\n        this.isContainer = true;\n\n        var _self = this;\n\n        this.setChildren = function (children) {\n            this.children = children;\n            _(this.children).each(function (child) {\n                child.parent = _self;\n            });\n        };\n\n        this.setChildren(children);\n\n        this.getIsSealed = function () {\n            return _(this.children).any(function (child) {\n                return child.isTemplated;\n            });\n        };\n\n        this.addChild = function (child) {\n            if (!_(this.children).contains(child) && (_(this.allowedChildTypes).contains(child.type) || child.isContainable))\n                this.children.push(child);\n            child.setEditor(this.editor);\n            child.setIsTemplated(false);\n            child.parent = this;\n        };\n\n        this.deleteChild = function (child) {\n            var index = _(this.children).indexOf(child);\n            if (index >= 0) {\n                this.children.splice(index, 1);\n                if (child.getIsActive())\n                    this.editor.activeElement = null;\n                if (child.getIsFocused()) {\n                    // If the deleted child was focused, try to set new focus to the most appropriate sibling or parent.\n                    if (this.children.length > index)\n                        this.children[index].setIsFocused();\n                    else if (index > 0)\n                        this.children[index - 1].setIsFocused();\n                    else\n                        this.setIsFocused();\n                }\n            }\n        };\n\n        this.moveFocusPrevChild = function (child) {\n            if (this.children.length < 2)\n                return;\n            var index = _(this.children).indexOf(child);\n            if (index > 0)\n                this.children[index - 1].setIsFocused();\n        };\n\n        this.moveFocusNextChild = function (child) {\n            if (this.children.length < 2)\n                return;\n            var index = _(this.children).indexOf(child);\n            if (index < this.children.length - 1)\n                this.children[index + 1].setIsFocused();\n        };\n\n        this.insertChild = function (child, afterChild) {\n            if (!_(this.children).contains(child)) {\n                var index = Math.max(_(this.children).indexOf(afterChild), 0);\n                this.children.splice(index + 1, 0, child);\n                child.setEditor(this.editor);\n                child.parent = this;\n            }\n        };\n\n        this.moveChildUp = function (child) {\n            if (!this.canMoveChildUp(child))\n                return;\n            var index = _(this.children).indexOf(child);\n            this.children.move(index, index - 1);\n        };\n\n        this.moveChildDown = function (child) {\n            if (!this.canMoveChildDown(child))\n                return;\n            var index = _(this.children).indexOf(child);\n            this.children.move(index, index + 1);\n        };\n\n        this.canMoveChildUp = function (child) {\n            var index = _(this.children).indexOf(child);\n            return index > 0;\n        };\n\n        this.canMoveChildDown = function (child) {\n            var index = _(this.children).indexOf(child);\n            return index < this.children.length - 1;\n        };\n\n        this.childrenToObject = function () {\n            return _(this.children).map(function (child) {\n                return child.toObject();\n            });\n        };\n\n        this.getInnerText = function () {\n            return _(this.children).reduce(function (memo, child) {\n                return memo + \"\\n\" + child.getInnerText();\n            }, \"\");\n        }\n\n        this.paste = function (clipboardData) {\n            var json = clipboardData.getData(\"text/json\");\n            if (!!json) {\n                var data = JSON.parse(json);\n                var child = LayoutEditor.elementFrom(data);\n                this.pasteChild(child);\n            }\n        };\n\n        this.pasteChild = function (child) {\n            if (_(this.allowedChildTypes).contains(child.type) || child.isContainable) {\n                this.addChild(child);\n                child.setIsFocused();\n            }\n            else if (!!this.parent)\n                this.parent.pasteChild(child);\n        }\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Canvas = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Canvas\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Canvas.from = function (value) {\n        return new LayoutEditor.Canvas(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));\n","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Grid = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Grid\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Row\"], children);\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Grid.from = function (value) {\n        var result = new LayoutEditor.Grid(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Row = function (data, htmlId, htmlClass, htmlStyle, isTemplated, children) {\n        LayoutEditor.Element.call(this, \"Row\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Column\"], children);\n\n        var _self = this;\n\n        function _getTotalColumnsWidth() {\n            return _(_self.children).reduce(function (memo, child) {\n                return memo + child.offset + child.width;\n            }, 0);\n        }\n\n        // Implements a simple algorithm to distribute space (either positive or negative)\n        // between the child columns of the row. Negative space is distributed when making\n        // room for a new column (e.c. clipboard paste or dropping from the toolbox) while\n        // positive space is distributed when filling the grap of a removed column.\n        function _distributeSpace(space) {\n            if (space == 0)\n                return true;\n             \n            var undistributedSpace = space;\n\n            if (undistributedSpace < 0) {\n                var vacantSpace = 12 - _getTotalColumnsWidth();\n                undistributedSpace += vacantSpace;\n                if (undistributedSpace > 0)\n                    undistributedSpace = 0;\n            }\n\n            // If space is negative, try to decrease offsets first.\n            while (undistributedSpace < 0 && _(_self.children).any(function (column) { return column.offset > 0; })) { // While there is still offset left to remove.\n                for (i = 0; i < _self.children.length && undistributedSpace < 0; i++) {\n                    var column = _self.children[i];\n                    if (column.offset > 0) {\n                        column.offset--;\n                        undistributedSpace++;\n                    }\n                }\n            }\n\n            function hasWidth(column) {\n                if (undistributedSpace > 0)\n                    return column.width < 12;\n                else if (undistributedSpace < 0)\n                    return column.width > 1;\n                return false;\n            }\n\n            // Try to distribute remaining space (could be negative or positive) using widths.\n            while (undistributedSpace != 0) {\n                // Any more column width available for distribution?\n                if (!_(_self.children).any(hasWidth))\n                    break;\n                for (i = 0; i < _self.children.length && undistributedSpace != 0; i++) {\n                    var column = _self.children[i % _self.children.length];\n                    if (hasWidth(column)) {\n                        var delta = undistributedSpace / Math.abs(undistributedSpace);\n                        column.width += delta;\n                        undistributedSpace -= delta;\n                    }\n                }                \n            }\n\n            return undistributedSpace == 0;\n        }\n\n        var _isAddingColumn = false;\n\n        this.canAddColumn = function () {\n            return this.children.length < 12;\n        };\n\n        this.beginAddColumn = function (newColumnWidth) {\n            if (!!_isAddingColumn)\n                throw new Error(\"Column add operation is already in progress.\")\n            _(this.children).each(function (column) {\n                column.beginChange();\n            });\n            if (_distributeSpace(-newColumnWidth)) {\n                _isAddingColumn = true;\n                return true;\n            }\n            _(this.children).each(function (column) {\n                column.rollbackChange();\n            });\n            return false;\n        };\n\n        this.commitAddColumn = function () {\n            if (!_isAddingColumn)\n                throw new Error(\"No column add operation in progress.\")\n            _(this.children).each(function (column) {\n                column.commitChange();\n            });\n            _isAddingColumn = false;\n        };\n\n        this.rollbackAddColumn = function () {\n            if (!_isAddingColumn)\n                throw new Error(\"No column add operation in progress.\")\n            _(this.children).each(function (column) {\n                column.rollbackChange();\n            });\n            _isAddingColumn = false;\n        };\n\n        var _baseDeleteChild = this.deleteChild;\n        this.deleteChild = function (column) { \n            var width = column.width;\n            _baseDeleteChild.call(this, column);\n            _distributeSpace(width);\n        };\n\n        this.canContractColumnRight = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0)\n                return column.width > 1;\n            return false;\n        };\n\n        this.contractColumnRight = function (column, connectAdjacent) {\n            if (!this.canContractColumnRight(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width > 1) {\n                    column.width--;\n                    if (this.children.length > index + 1) {\n                        var nextColumn = this.children[index + 1];\n                        if (connectAdjacent && nextColumn.offset == 0)\n                            nextColumn.width++;\n                        else\n                            nextColumn.offset++;\n                    }\n                }\n            }\n        };\n\n        this.canExpandColumnRight = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width >= 12)\n                    return false;\n                if (this.children.length > index + 1) {\n                    var nextColumn = this.children[index + 1];\n                    if (connectAdjacent && nextColumn.offset == 0)\n                        return nextColumn.width > 1;\n                    else\n                        return nextColumn.offset > 0;\n                }\n                return _getTotalColumnsWidth() < 12;\n            }\n            return false;\n        };\n\n        this.expandColumnRight = function (column, connectAdjacent) {\n            if (!this.canExpandColumnRight(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (this.children.length > index + 1) {\n                    var nextColumn = this.children[index + 1];\n                    if (connectAdjacent && nextColumn.offset == 0)\n                        nextColumn.width--;\n                    else\n                        nextColumn.offset--;\n                }\n                column.width++;\n            }\n        };\n\n        this.canExpandColumnLeft = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (column.width >= 12)\n                    return false;\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        return prevColumn.width > 1;\n                }\n                return column.offset > 0;\n            }\n            return false;\n        };\n\n        this.expandColumnLeft = function (column, connectAdjacent) {\n            if (!this.canExpandColumnLeft(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        prevColumn.width--;\n                    else\n                        column.offset--;\n                }\n                else\n                    column.offset--;\n                column.width++;\n            }\n        };\n\n        this.canContractColumnLeft = function (column, connectAdjacent) {\n            var index = _(this.children).indexOf(column);\n            if (index >= 0)\n                return column.width > 1;\n            return false;\n        };\n\n        this.contractColumnLeft = function (column, connectAdjacent) {\n            if (!this.canContractColumnLeft(column, connectAdjacent))\n                return;\n\n            var index = _(this.children).indexOf(column);\n            if (index >= 0) {\n                if (index > 0) {\n                    var prevColumn = this.children[index - 1];\n                    if (connectAdjacent && column.offset == 0)\n                        prevColumn.width++;\n                    else\n                        column.offset++;\n                }\n                else\n                    column.offset++;\n                column.width--;\n            }\n        };\n\n        this.evenColumns = function () {\n            if (this.children.length == 0)\n                return;\n\n            var evenWidth = Math.floor(12 / this.children.length);\n            _(this.children).each(function (column) {\n                column.width = evenWidth;\n                column.offset = 0;\n            });\n\n            var rest = 12 % this.children.length;\n            if (rest > 0)\n                _distributeSpace(rest);\n        };\n\n        var _basePasteChild = this.pasteChild;\n        this.pasteChild = function (child) {\n            if (child.type == \"Column\") {\n                if (this.beginAddColumn(child.width)) {\n                    this.commitAddColumn();\n                    _basePasteChild.call(this, child)\n                }\n            }\n            else if (!!this.parent)\n                this.parent.pasteChild(child);\n        }\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Row.from = function (value) {\n        var result = new LayoutEditor.Row(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Column = function (data, htmlId, htmlClass, htmlStyle, isTemplated, width, offset, children) {\n        LayoutEditor.Element.call(this, \"Column\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n        LayoutEditor.Container.call(this, [\"Grid\", \"Content\"], children);\n\n        this.width = width;\n        this.offset = offset;\n\n        var _hasPendingChange = false;\n        var _origWidth = 0;\n        var _origOffset = 0;\n\n        this.beginChange = function () {\n            if (!!_hasPendingChange)\n                throw new Error(\"Column already has a pending change.\")\n            _hasPendingChange = true;\n            _origWidth = this.width;\n            _origOffset = this.offset;\n        };\n\n        this.commitChange = function () {\n            if (!_hasPendingChange)\n                throw new Error(\"Column has no pending change.\")\n            _origWidth = 0;\n            _origOffset = 0;\n            _hasPendingChange = false;\n        };\n\n        this.rollbackChange = function () {\n            if (!_hasPendingChange)\n                throw new Error(\"Column has no pending change.\")\n            this.width = _origWidth;\n            this.offset = _origOffset;\n            _origWidth = 0;\n            _origOffset = 0;\n            _hasPendingChange = false;\n        };\n\n        this.canSplit = function () {\n            return this.width > 1;\n        };\n\n        this.split = function () {\n            if (!this.canSplit())\n                return;\n\n            var newColumnWidth = Math.floor(this.width / 2);\n            var newColumn = LayoutEditor.Column.from({\n                data: null,\n                htmlId: null,\n                htmlClass: null,\n                htmlStyle: null,\n                width: newColumnWidth,\n                offset: 0,\n                children: []\n            });\n            \n            this.width = this.width - newColumnWidth;\n            this.parent.insertChild(newColumn, this);\n            newColumn.setIsFocused();\n        };\n\n        this.canContractRight = function (connectAdjacent) {\n            return this.parent.canContractColumnRight(this, connectAdjacent);\n        };\n\n        this.contractRight = function (connectAdjacent) {\n            this.parent.contractColumnRight(this, connectAdjacent);\n        };\n\n        this.canExpandRight = function (connectAdjacent) {\n            return this.parent.canExpandColumnRight(this, connectAdjacent);\n        };\n\n        this.expandRight = function (connectAdjacent) {\n            this.parent.expandColumnRight(this, connectAdjacent);\n        };\n\n        this.canExpandLeft = function (connectAdjacent) {\n            return this.parent.canExpandColumnLeft(this, connectAdjacent);\n        };\n\n        this.expandLeft = function (connectAdjacent) {\n            this.parent.expandColumnLeft(this, connectAdjacent);\n        };\n\n        this.canContractLeft = function (connectAdjacent) {\n            return this.parent.canContractColumnLeft(this, connectAdjacent);\n        };\n\n        this.contractLeft = function (connectAdjacent) {\n            this.parent.contractColumnLeft(this, connectAdjacent);\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.width = this.width;\n            result.offset = this.offset;\n            result.children = this.childrenToObject();\n            return result;\n        };\n    };\n\n    LayoutEditor.Column.from = function (value) {\n        var result = new LayoutEditor.Column(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.width,\n            value.offset,\n            LayoutEditor.childrenFrom(value.children));\n        result.toolboxIcon = value.toolboxIcon;\n        result.toolboxLabel = value.toolboxLabel;\n        result.toolboxDescription = value.toolboxDescription;\n        return result;\n    };\n\n    LayoutEditor.Column.times = function (value) {\n        return _.times(value, function (n) {\n            return LayoutEditor.Column.from({\n                data: null,\n                htmlId: null,\n                htmlClass: null,\n                isTemplated: false,\n                width: 12 / value,\n                offset: 0,\n                children: []\n            });\n        });\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function (LayoutEditor) {\n\n    LayoutEditor.Content = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor) {\n        LayoutEditor.Element.call(this, \"Content\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n\n        this.contentType = contentType;\n        this.contentTypeLabel = contentTypeLabel;\n        this.contentTypeClass = contentTypeClass;\n        this.html = html;\n        this.hasEditor = hasEditor;\n\n        this.getInnerText = function () {\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\n        };\n\n        // This function will be overwritten by the Content directive.\n        this.setHtml = function (html) {\n            this.html = html;\n            this.htmlUnsafe = html;\n        }\n\n        this.toObject = function () {\n            return {\n                \"type\": \"Content\"\n            };\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.contentType = this.contentType;\n            result.contentTypeLabel = this.contentTypeLabel;\n            result.contentTypeClass = this.contentTypeClass;\n            result.html = this.html;\n            result.hasEditor = hasEditor;\n            return result;\n        };\n\n        this.setHtml(html);\n    };\n\n    LayoutEditor.Content.from = function (value) {\n        var result = new LayoutEditor.Content(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.contentType,\n            value.contentTypeLabel,\n            value.contentTypeClass,\n            value.html,\n            value.hasEditor);\n\n        return result;\n    };\n\n})(LayoutEditor || (LayoutEditor = {}));","var LayoutEditor;\n(function ($, LayoutEditor) {\n\n    LayoutEditor.Html = function (data, htmlId, htmlClass, htmlStyle, isTemplated, contentType, contentTypeLabel, contentTypeClass, html, hasEditor) {\n        LayoutEditor.Element.call(this, \"Html\", data, htmlId, htmlClass, htmlStyle, isTemplated);\n\n        this.contentType = contentType;\n        this.contentTypeLabel = contentTypeLabel;\n        this.contentTypeClass = contentTypeClass;\n        this.html = html;\n        this.hasEditor = hasEditor;\n        this.isContainable = true;\n\n        this.getInnerText = function () {\n            return $($.parseHTML(\"<div>\" + this.html + \"</div>\")).text();\n        };\n\n        // This function will be overwritten by the Content directive.\n        this.setHtml = function (html) {\n            this.html = html;\n            this.htmlUnsafe = html;\n        }\n\n        this.toObject = function () {\n            return {\n                \"type\": \"Html\"\n            };\n        };\n\n        this.toObject = function () {\n            var result = this.elementToObject();\n            result.contentType = this.contentType;\n            result.contentTypeLabel = this.contentTypeLabel;\n            result.contentTypeClass = this.contentTypeClass;\n            result.html = this.html;\n            result.hasEditor = hasEditor;\n            return result;\n        };\n\n        var getEditorObject = this.getEditorObject;\n        this.getEditorObject = function () {\n            var dto = getEditorObject();\n            return $.extend(dto, {\n                Content: this.html\n            });\n        }\n\n        this.setHtml(html);\n    };\n\n    LayoutEditor.Html.from = function (value) {\n        var result = new LayoutEditor.Html(\n            value.data,\n            value.htmlId,\n            value.htmlClass,\n            value.htmlStyle,\n            value.isTemplated,\n            value.contentType,\n            value.contentTypeLabel,\n            value.contentTypeClass,\n            value.html,\n            value.hasEditor);\n\n        return result;\n    };\n\n    LayoutEditor.registerFactory(\"Html\", function(value) { return LayoutEditor.Html.from(value); });\n\n})(jQuery, LayoutEditor || (LayoutEditor = {}));"],"sourceRoot":"/source/"} \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js index c354fee03..9e2520803 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models.min.js @@ -1 +1 @@ -var LayoutEditor;!function(t){Array.prototype.move=function(t,i){this.splice(i,0,this.splice(t,1)[0])},t.childrenFrom=function(i){return _(i).map(function(i){return t.elementFrom(i)})};var i=t.registerFactory=function(i,n){var e=t.factories=t.factories||{};e[i]=n};i("Grid",function(i){return t.Grid.from(i)}),i("Row",function(i){return t.Row.from(i)}),i("Column",function(i){return t.Column.from(i)}),i("Content",function(i){return t.Content.from(i)}),t.elementFrom=function(i){var n=t.factories[i.type];if(!n)throw new Error('No element with type "'+i.type+'" was found.');var e=n(i);return e},t.setModel=function(t,i){$(t).scope().element=i},t.getModel=function(t){return $(t).scope().element}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Editor=function(i,n){this.config=i,this.canvas=t.Canvas.from(n),this.initialState=JSON.stringify(this.canvas.toObject()),this.activeElement=null,this.focusedElement=null,this.dropTargetElement=null,this.isDragging=!1,this.inlineEditingIsActive=!1,this.isResizing=!1,this.resetToolboxElements=function(){this.toolboxElements=[t.Row.from({children:[]})]},this.isDirty=function(){var t=JSON.stringify(this.canvas.toObject());return this.initialState!=t},this.resetToolboxElements(),this.canvas.setEditor(this)}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Element=function(t,i,n,e,h,o){if(!t)throw new Error("Parameter 'type' is required.");this.type=t,this.data=i,this.htmlId=n,this.htmlClass=e,this.htmlStyle=h,this.isTemplated=o,this.editor=null,this.parent=null,this.setIsFocusedEventHandlers=[],this.setEditor=function(t){this.editor=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setEditor(t)})},this.setParent=function(t){this.parent=t,this.parent.linkChild&&this.parent.linkChild(this)},this.setIsTemplated=function(t){this.isTemplated=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setIsTemplated(t)})},this.getIsActive=function(){return this.editor?this.editor.activeElement===this&&!this.getIsFocused():!1},this.setIsActive=function(t){this.editor&&(this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.activeElement=t?this:this.parent))},this.getIsFocused=function(){return this.editor?this.editor.focusedElement===this:!1},this.setIsFocused=function(){this.editor&&(this.isTemplated||this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.focusedElement=this,_(this.setIsFocusedEventHandlers).each(function(t){try{t()}catch(i){}})))},this.getIsSelected=function(){return this.getIsFocused()?!0:this.children&&_.isArray(this.children)?_(this.children).any(function(t){return t.getIsSelected()}):!1},this.getIsDropTarget=function(){return this.editor?this.editor.dropTargetElement===this:!1},this.setIsDropTarget=function(t){this.editor&&(this.editor.dropTargetElement=t?this:null)},this["delete"]=function(){this.parent&&this.parent.deleteChild(this)},this.canMoveUp=function(){return this.parent?this.parent.canMoveChildUp(this):!1},this.moveUp=function(){this.parent&&this.parent.moveChildUp(this)},this.canMoveDown=function(){return this.parent?this.parent.canMoveChildDown(this):!1},this.moveDown=function(){this.parent&&this.parent.moveChildDown(this)},this.elementToObject=function(){return{type:this.type,data:this.data,htmlId:this.htmlId,htmlClass:this.htmlClass,htmlStyle:this.htmlStyle,isTemplated:this.isTemplated}},this.getEditorObject=function(){return{}},this.copy=function(t){var i=this.getInnerText();t.setData("text/plain",i),console.log(i);var n=this.toObject(),e=JSON.stringify(n,null," ");t.setData("text/json",e)},this.cut=function(t){this.copy(t),this["delete"]()},this.paste=function(t){this.parent&&this.parent.paste(t)}}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Container=function(i,n){this.allowedChildTypes=i,this.children=n,this.isContainer=!0;var e=this;this.setChildren=function(t){this.children=t,_(this.children).each(function(t){t.parent=e})},this.setChildren(n),this.getIsSealed=function(){return _(this.children).any(function(t){return t.isTemplated})},this.addChild=function(t){_(this.children).contains(t)||!_(this.allowedChildTypes).contains(t.type)&&!t.isContainable||this.children.push(t),t.setEditor(this.editor),t.setIsTemplated(!1),t.parent=this},this.deleteChild=function(t){var i=_(this.children).indexOf(t);i>=0&&(this.children.splice(i,1),t.getIsActive()&&(this.editor.activeElement=null),t.getIsFocused()&&(this.children.length>i?this.children[i].setIsFocused():i>0?this.children[i-1].setIsFocused():this.setIsFocused()))},this.moveFocusPrevChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i>0&&this.children[i-1].setIsFocused()}},this.moveFocusNextChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i0},this.canMoveChildDown=function(t){var i=_(this.children).indexOf(t);return i0?t.width<12:0>e?t.width>1:!1}if(0==t)return!0;var e=t;if(0>e){var h=12-l();e+=h,e>0&&(e=0)}for(;0>e&&_(a.children).any(function(t){return t.offset>0});)for(i=0;ie;i++){var o=a.children[i];o.offset>0&&(o.offset--,e++)}for(;0!=e&&_(a.children).any(n);)for(i=0;i=0?t.width>1:!1},this.contractColumnRight=function(t,i){if(this.canContractColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0&&t.width>1&&(t.width--,this.children.length>n+1)){var e=this.children[n+1];i&&0==e.offset?e.width++:e.offset++}}},this.canExpandColumnRight=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(this.children.length>n+1){var e=this.children[n+1];return i&&0==e.offset?e.width>1:e.offset>0}return l()<12}return!1},this.expandColumnRight=function(t,i){if(this.canExpandColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(this.children.length>n+1){var e=this.children[n+1];i&&0==e.offset?e.width--:e.offset--}t.width++}}},this.canExpandColumnLeft=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(n>0){var e=this.children[n-1];if(i&&0==t.offset)return e.width>1}return t.offset>0}return!1},this.expandColumnLeft=function(t,i){if(this.canExpandColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width--:t.offset--}else t.offset--;t.width++}}},this.canContractColumnLeft=function(t,i){var n=_(this.children).indexOf(t);return n>=0?t.width>1:!1},this.contractColumnLeft=function(t,i){if(this.canContractColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width++:t.offset++}else t.offset++;t.width--}}},this.evenColumns=function(){if(0!=this.children.length){var t=Math.floor(12/this.children.length);_(this.children).each(function(i){i.width=t,i.offset=0});var i=12%this.children.length;i>0&&c(i)}};var f=this.pasteChild;this.pasteChild=function(t){"Column"==t.type?this.beginAddColumn(t.width)&&(this.commitAddColumn(),f.call(this,t)):this.parent&&this.parent.pasteChild(t)},this.toObject=function(){var t=this.elementToObject();return t.children=this.childrenToObject(),t}},t.Row.from=function(i){var n=new t.Row(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Column=function(i,n,e,h,o,s,r,l){t.Element.call(this,"Column",i,n,e,h,o),t.Container.call(this,["Grid","Content"],l),this.width=s,this.offset=r;var c=!1,a=0,d=0;this.beginChange=function(){if(c)throw new Error("Column already has a pending change.");c=!0,a=this.width,d=this.offset},this.commitChange=function(){if(!c)throw new Error("Column has no pending change.");a=0,d=0,c=!1},this.rollbackChange=function(){if(!c)throw new Error("Column has no pending change.");this.width=a,this.offset=d,a=0,d=0,c=!1},this.canSplit=function(){return this.width>1},this.split=function(){if(this.canSplit()){var i=Math.floor(this.width/2),n=t.Column.from({data:null,htmlId:null,htmlClass:null,htmlStyle:null,width:i,offset:0,children:[]});this.width=this.width-i,this.parent.insertChild(n,this),n.setIsFocused()}},this.canContractRight=function(t){return this.parent.canContractColumnRight(this,t)},this.contractRight=function(t){this.parent.contractColumnRight(this,t)},this.canExpandRight=function(t){return this.parent.canExpandColumnRight(this,t)},this.expandRight=function(t){this.parent.expandColumnRight(this,t)},this.canExpandLeft=function(t){return this.parent.canExpandColumnLeft(this,t)},this.expandLeft=function(t){this.parent.expandColumnLeft(this,t)},this.canContractLeft=function(t){return this.parent.canContractColumnLeft(this,t)},this.contractLeft=function(t){this.parent.contractColumnLeft(this,t)},this.toObject=function(){var t=this.elementToObject();return t.width=this.width,t.offset=this.offset,t.children=this.childrenToObject(),t}},t.Column.from=function(i){var n=new t.Column(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.width,i.offset,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n},t.Column.times=function(i){return _.times(i,function(n){return t.Column.from({data:null,htmlId:null,htmlClass:null,isTemplated:!1,width:12/i,offset:0,children:[]})})}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Content=function(i,n,e,h,o,s,r,l,c,a){t.Element.call(this,"Content",i,n,e,h,o),this.contentType=s,this.contentTypeLabel=r,this.contentTypeClass=l,this.html=c,this.hasEditor=a,this.getInnerText=function(){return $($.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Content"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=a,t},this.setHtml(c)},t.Content.from=function(i){var n=new t.Content(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.contentType,i.contentTypeLabel,i.contentTypeClass,i.html,i.hasEditor);return n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t,i){i.Html=function(n,e,h,o,s,r,l,c,a,d){i.Element.call(this,"Html",n,e,h,o,s),this.contentType=r,this.contentTypeLabel=l,this.contentTypeClass=c,this.html=a,this.hasEditor=d,this.isContainable=!0,this.getInnerText=function(){return t(t.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Html"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=d,t};var u=this.getEditorObject;this.getEditorObject=function(){var i=u();return t.extend(i,{Content:this.html})},this.setHtml(a)},i.Html.from=function(t){var n=new i.Html(t.data,t.htmlId,t.htmlClass,t.htmlStyle,t.isTemplated,t.contentType,t.contentTypeLabel,t.contentTypeClass,t.html,t.hasEditor);return n},i.registerFactory("Html",function(t){return i.Html.from(t)})}(jQuery,LayoutEditor||(LayoutEditor={})); \ No newline at end of file +var LayoutEditor;!function(t){Array.prototype.move=function(t,i){this.splice(i,0,this.splice(t,1)[0])},t.childrenFrom=function(i){return _(i).map(function(i){return t.elementFrom(i)})};var i=t.registerFactory=function(i,n){var e=t.factories=t.factories||{};e[i]=n};i("Grid",function(i){return t.Grid.from(i)}),i("Row",function(i){return t.Row.from(i)}),i("Column",function(i){return t.Column.from(i)}),i("Content",function(i){return t.Content.from(i)}),t.elementFrom=function(i){var n=t.factories[i.type];if(!n)throw new Error('No element with type "'+i.type+'" was found.');var e=n(i);return e},t.setModel=function(t,i){$(t).scope().element=i},t.getModel=function(t){return $(t).scope().element}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Editor=function(i,n){this.config=i,this.canvas=t.Canvas.from(n),this.initialState=JSON.stringify(this.canvas.toObject()),this.activeElement=null,this.focusedElement=null,this.dropTargetElement=null,this.isDragging=!1,this.inlineEditingIsActive=!1,this.isResizing=!1,this.resetToolboxElements=function(){this.toolboxElements=[t.Row.from({children:[]})]},this.isDirty=function(){var t=JSON.stringify(this.canvas.toObject());return this.initialState!=t},this.resetToolboxElements(),this.canvas.setEditor(this)}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Element=function(t,i,n,e,h,o){if(!t)throw new Error("Parameter 'type' is required.");this.type=t,this.data=i,this.htmlId=n,this.htmlClass=e,this.htmlStyle=h,this.isTemplated=o,this.editor=null,this.parent=null,this.setIsFocusedEventHandlers=[],this.setEditor=function(t){this.editor=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setEditor(t)})},this.setParent=function(t){this.parent=t,this.parent.linkChild&&this.parent.linkChild(this)},this.setIsTemplated=function(t){this.isTemplated=t,this.children&&_.isArray(this.children)&&_(this.children).each(function(i){i.setIsTemplated(t)})},this.getIsActive=function(){return this.editor?this.editor.activeElement===this&&!this.getIsFocused():!1},this.setIsActive=function(t){this.editor&&(this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.activeElement=t?this:this.parent))},this.getIsFocused=function(){return this.editor?this.editor.focusedElement===this:!1},this.setIsFocused=function(){this.editor&&(this.isTemplated||this.editor.isDragging||this.editor.inlineEditingIsActive||this.editor.isResizing||(this.editor.focusedElement=this,_(this.setIsFocusedEventHandlers).each(function(t){try{t()}catch(i){}})))},this.getIsSelected=function(){return this.getIsFocused()?!0:this.children&&_.isArray(this.children)?_(this.children).any(function(t){return t.getIsSelected()}):!1},this.getIsDropTarget=function(){return this.editor?this.editor.dropTargetElement===this:!1},this.setIsDropTarget=function(t){this.editor&&(this.editor.dropTargetElement=t?this:null)},this["delete"]=function(){this.parent&&this.parent.deleteChild(this)},this.canMoveUp=function(){return this.parent?this.parent.canMoveChildUp(this):!1},this.moveUp=function(){this.parent&&this.parent.moveChildUp(this)},this.canMoveDown=function(){return this.parent?this.parent.canMoveChildDown(this):!1},this.moveDown=function(){this.parent&&this.parent.moveChildDown(this)},this.elementToObject=function(){return{type:this.type,data:this.data,htmlId:this.htmlId,htmlClass:this.htmlClass,htmlStyle:this.htmlStyle,isTemplated:this.isTemplated}},this.getEditorObject=function(){return{}},this.copy=function(t){var i=this.getInnerText();t.setData("text/plain",i);var n=this.toObject(),e=JSON.stringify(n,null," ");t.setData("text/json",e)},this.cut=function(t){this.copy(t),this["delete"]()},this.paste=function(t){this.parent&&this.parent.paste(t)}}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Container=function(i,n){this.allowedChildTypes=i,this.children=n,this.isContainer=!0;var e=this;this.setChildren=function(t){this.children=t,_(this.children).each(function(t){t.parent=e})},this.setChildren(n),this.getIsSealed=function(){return _(this.children).any(function(t){return t.isTemplated})},this.addChild=function(t){_(this.children).contains(t)||!_(this.allowedChildTypes).contains(t.type)&&!t.isContainable||this.children.push(t),t.setEditor(this.editor),t.setIsTemplated(!1),t.parent=this},this.deleteChild=function(t){var i=_(this.children).indexOf(t);i>=0&&(this.children.splice(i,1),t.getIsActive()&&(this.editor.activeElement=null),t.getIsFocused()&&(this.children.length>i?this.children[i].setIsFocused():i>0?this.children[i-1].setIsFocused():this.setIsFocused()))},this.moveFocusPrevChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i>0&&this.children[i-1].setIsFocused()}},this.moveFocusNextChild=function(t){if(!(this.children.length<2)){var i=_(this.children).indexOf(t);i0},this.canMoveChildDown=function(t){var i=_(this.children).indexOf(t);return i0?t.width<12:0>e?t.width>1:!1}if(0==t)return!0;var e=t;if(0>e){var h=12-l();e+=h,e>0&&(e=0)}for(;0>e&&_(a.children).any(function(t){return t.offset>0});)for(i=0;ie;i++){var o=a.children[i];o.offset>0&&(o.offset--,e++)}for(;0!=e&&_(a.children).any(n);)for(i=0;i=0?t.width>1:!1},this.contractColumnRight=function(t,i){if(this.canContractColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0&&t.width>1&&(t.width--,this.children.length>n+1)){var e=this.children[n+1];i&&0==e.offset?e.width++:e.offset++}}},this.canExpandColumnRight=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(this.children.length>n+1){var e=this.children[n+1];return i&&0==e.offset?e.width>1:e.offset>0}return l()<12}return!1},this.expandColumnRight=function(t,i){if(this.canExpandColumnRight(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(this.children.length>n+1){var e=this.children[n+1];i&&0==e.offset?e.width--:e.offset--}t.width++}}},this.canExpandColumnLeft=function(t,i){var n=_(this.children).indexOf(t);if(n>=0){if(t.width>=12)return!1;if(n>0){var e=this.children[n-1];if(i&&0==t.offset)return e.width>1}return t.offset>0}return!1},this.expandColumnLeft=function(t,i){if(this.canExpandColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width--:t.offset--}else t.offset--;t.width++}}},this.canContractColumnLeft=function(t,i){var n=_(this.children).indexOf(t);return n>=0?t.width>1:!1},this.contractColumnLeft=function(t,i){if(this.canContractColumnLeft(t,i)){var n=_(this.children).indexOf(t);if(n>=0){if(n>0){var e=this.children[n-1];i&&0==t.offset?e.width++:t.offset++}else t.offset++;t.width--}}},this.evenColumns=function(){if(0!=this.children.length){var t=Math.floor(12/this.children.length);_(this.children).each(function(i){i.width=t,i.offset=0});var i=12%this.children.length;i>0&&c(i)}};var f=this.pasteChild;this.pasteChild=function(t){"Column"==t.type?this.beginAddColumn(t.width)&&(this.commitAddColumn(),f.call(this,t)):this.parent&&this.parent.pasteChild(t)},this.toObject=function(){var t=this.elementToObject();return t.children=this.childrenToObject(),t}},t.Row.from=function(i){var n=new t.Row(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Column=function(i,n,e,h,o,s,r,l){t.Element.call(this,"Column",i,n,e,h,o),t.Container.call(this,["Grid","Content"],l),this.width=s,this.offset=r;var c=!1,a=0,d=0;this.beginChange=function(){if(c)throw new Error("Column already has a pending change.");c=!0,a=this.width,d=this.offset},this.commitChange=function(){if(!c)throw new Error("Column has no pending change.");a=0,d=0,c=!1},this.rollbackChange=function(){if(!c)throw new Error("Column has no pending change.");this.width=a,this.offset=d,a=0,d=0,c=!1},this.canSplit=function(){return this.width>1},this.split=function(){if(this.canSplit()){var i=Math.floor(this.width/2),n=t.Column.from({data:null,htmlId:null,htmlClass:null,htmlStyle:null,width:i,offset:0,children:[]});this.width=this.width-i,this.parent.insertChild(n,this),n.setIsFocused()}},this.canContractRight=function(t){return this.parent.canContractColumnRight(this,t)},this.contractRight=function(t){this.parent.contractColumnRight(this,t)},this.canExpandRight=function(t){return this.parent.canExpandColumnRight(this,t)},this.expandRight=function(t){this.parent.expandColumnRight(this,t)},this.canExpandLeft=function(t){return this.parent.canExpandColumnLeft(this,t)},this.expandLeft=function(t){this.parent.expandColumnLeft(this,t)},this.canContractLeft=function(t){return this.parent.canContractColumnLeft(this,t)},this.contractLeft=function(t){this.parent.contractColumnLeft(this,t)},this.toObject=function(){var t=this.elementToObject();return t.width=this.width,t.offset=this.offset,t.children=this.childrenToObject(),t}},t.Column.from=function(i){var n=new t.Column(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.width,i.offset,t.childrenFrom(i.children));return n.toolboxIcon=i.toolboxIcon,n.toolboxLabel=i.toolboxLabel,n.toolboxDescription=i.toolboxDescription,n},t.Column.times=function(i){return _.times(i,function(n){return t.Column.from({data:null,htmlId:null,htmlClass:null,isTemplated:!1,width:12/i,offset:0,children:[]})})}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t){t.Content=function(i,n,e,h,o,s,r,l,c,a){t.Element.call(this,"Content",i,n,e,h,o),this.contentType=s,this.contentTypeLabel=r,this.contentTypeClass=l,this.html=c,this.hasEditor=a,this.getInnerText=function(){return $($.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Content"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=a,t},this.setHtml(c)},t.Content.from=function(i){var n=new t.Content(i.data,i.htmlId,i.htmlClass,i.htmlStyle,i.isTemplated,i.contentType,i.contentTypeLabel,i.contentTypeClass,i.html,i.hasEditor);return n}}(LayoutEditor||(LayoutEditor={}));var LayoutEditor;!function(t,i){i.Html=function(n,e,h,o,s,r,l,c,a,d){i.Element.call(this,"Html",n,e,h,o,s),this.contentType=r,this.contentTypeLabel=l,this.contentTypeClass=c,this.html=a,this.hasEditor=d,this.isContainable=!0,this.getInnerText=function(){return t(t.parseHTML("
"+this.html+"
")).text()},this.setHtml=function(t){this.html=t,this.htmlUnsafe=t},this.toObject=function(){return{type:"Html"}},this.toObject=function(){var t=this.elementToObject();return t.contentType=this.contentType,t.contentTypeLabel=this.contentTypeLabel,t.contentTypeClass=this.contentTypeClass,t.html=this.html,t.hasEditor=d,t};var u=this.getEditorObject;this.getEditorObject=function(){var i=u();return t.extend(i,{Content:this.html})},this.setHtml(a)},i.Html.from=function(t){var n=new i.Html(t.data,t.htmlId,t.htmlClass,t.htmlStyle,t.isTemplated,t.contentType,t.contentTypeLabel,t.contentTypeClass,t.html,t.hasEditor);return n},i.registerFactory("Html",function(t){return i.Html.from(t)})}(jQuery,LayoutEditor||(LayoutEditor={})); \ No newline at end of file diff --git a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models/Element.js b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models/Element.js index bff13edd9..2cd44511e 100644 --- a/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models/Element.js +++ b/src/Orchard.Web/Modules/Orchard.Layouts/Scripts/Models/Element.js @@ -157,7 +157,6 @@ this.copy = function (clipboardData) { var text = this.getInnerText(); clipboardData.setData("text/plain", text); - console.log(text); var data = this.toObject(); var json = JSON.stringify(data, null, "\t");