dever 6 years ago
parent
commit
19c7e29229

+ 0 - 0
assets/lib/layui/plugin/citys.js → assets/lib/layui/admin/modules/plugin/citys.js


File diff suppressed because it is too large
+ 10 - 0
assets/lib/layui/admin/modules/plugin/cropper/cropper.css


+ 3087 - 0
assets/lib/layui/admin/modules/plugin/cropper/cropper.js

@@ -0,0 +1,3087 @@
+/*!
+ * Cropper v3.0.0
+ */
+
+layui.define(['jquery'], function (exports) {
+    var $ = layui.jquery;
+    $ = $ && $.hasOwnProperty('default') ? $['default'] : $;
+
+    var DEFAULTS = {
+        // Define the view mode of the cropper
+        viewMode: 0, // 0, 1, 2, 3
+
+        // Define the dragging mode of the cropper
+        dragMode: 'crop', // 'crop', 'move' or 'none'
+
+        // Define the aspect ratio of the crop box
+        aspectRatio: NaN,
+
+        // An object with the previous cropping result data
+        data: null,
+
+        // A selector for adding extra containers to preview
+        preview: '',
+
+        // Re-render the cropper when resize the window
+        responsive: true,
+
+        // Restore the cropped area after resize the window
+        restore: true,
+
+        // Check if the current image is a cross-origin image
+        checkCrossOrigin: true,
+
+        // Check the current image's Exif Orientation information
+        checkOrientation: true,
+
+        // Show the black modal
+        modal: true,
+
+        // Show the dashed lines for guiding
+        guides: true,
+
+        // Show the center indicator for guiding
+        center: true,
+
+        // Show the white modal to highlight the crop box
+        highlight: true,
+
+        // Show the grid background
+        background: true,
+
+        // Enable to crop the image automatically when initialize
+        autoCrop: true,
+
+        // Define the percentage of automatic cropping area when initializes
+        autoCropArea: 0.8,
+
+        // Enable to move the image
+        movable: true,
+
+        // Enable to rotate the image
+        rotatable: true,
+
+        // Enable to scale the image
+        scalable: true,
+
+        // Enable to zoom the image
+        zoomable: true,
+
+        // Enable to zoom the image by dragging touch
+        zoomOnTouch: true,
+
+        // Enable to zoom the image by wheeling mouse
+        zoomOnWheel: true,
+
+        // Define zoom ratio when zoom the image by wheeling mouse
+        wheelZoomRatio: 0.1,
+
+        // Enable to move the crop box
+        cropBoxMovable: true,
+
+        // Enable to resize the crop box
+        cropBoxResizable: true,
+
+        // Toggle drag mode between "crop" and "move" when click twice on the cropper
+        toggleDragModeOnDblclick: true,
+
+        // Size limitation
+        minCanvasWidth: 0,
+        minCanvasHeight: 0,
+        minCropBoxWidth: 0,
+        minCropBoxHeight: 0,
+        minContainerWidth: 200,
+        minContainerHeight: 100,
+
+        // Shortcuts of events
+        ready: null,
+        cropstart: null,
+        cropmove: null,
+        cropend: null,
+        crop: null,
+        zoom: null
+    };
+
+    var TEMPLATE = '<div class="cropper-container">' + '<div class="cropper-wrap-box">' + '<div class="cropper-canvas"></div>' + '</div>' + '<div class="cropper-drag-box"></div>' + '<div class="cropper-crop-box">' + '<span class="cropper-view-box"></span>' + '<span class="cropper-dashed dashed-h"></span>' + '<span class="cropper-dashed dashed-v"></span>' + '<span class="cropper-center"></span>' + '<span class="cropper-face"></span>' + '<span class="cropper-line line-e" data-action="e"></span>' + '<span class="cropper-line line-n" data-action="n"></span>' + '<span class="cropper-line line-w" data-action="w"></span>' + '<span class="cropper-line line-s" data-action="s"></span>' + '<span class="cropper-point point-e" data-action="e"></span>' + '<span class="cropper-point point-n" data-action="n"></span>' + '<span class="cropper-point point-w" data-action="w"></span>' + '<span class="cropper-point point-s" data-action="s"></span>' + '<span class="cropper-point point-ne" data-action="ne"></span>' + '<span class="cropper-point point-nw" data-action="nw"></span>' + '<span class="cropper-point point-sw" data-action="sw"></span>' + '<span class="cropper-point point-se" data-action="se"></span>' + '</div>' + '</div>';
+
+    var REGEXP_DATA_URL_HEAD = /^data:.*,/;
+    var REGEXP_USERAGENT = /(Macintosh|iPhone|iPod|iPad).*AppleWebKit/i;
+    var navigator = typeof window !== 'undefined' ? window.navigator : null;
+    var IS_SAFARI_OR_UIWEBVIEW = navigator && REGEXP_USERAGENT.test(navigator.userAgent);
+    var fromCharCode = String.fromCharCode;
+
+    function isNumber(n) {
+        return typeof n === 'number' && !isNaN(n);
+    }
+
+    function isUndefined(n) {
+        return typeof n === 'undefined';
+    }
+
+    function toArray(obj, offset) {
+        var args = [];
+
+        // This is necessary for IE8
+        if (isNumber(offset)) {
+            args.push(offset);
+        }
+
+        return args.slice.apply(obj, args);
+    }
+
+    // Custom proxy to avoid jQuery's guid
+    function proxy(fn, context) {
+        for (var _len = arguments.length, args = Array(_len > 2 ? _len - 2 : 0), _key = 2; _key < _len; _key++) {
+            args[_key - 2] = arguments[_key];
+        }
+
+        return function () {
+            for (var _len2 = arguments.length, args2 = Array(_len2), _key2 = 0; _key2 < _len2; _key2++) {
+                args2[_key2] = arguments[_key2];
+            }
+
+            return fn.apply(context, args.concat(toArray(args2)));
+        };
+    }
+
+    function objectKeys(obj) {
+        var keys = [];
+
+        $.each(obj, function (key) {
+            keys.push(key);
+        });
+
+        return keys;
+    }
+
+    function isCrossOriginURL(url) {
+        var parts = url.match(/^(https?:)\/\/([^:/?#]+):?(\d*)/i);
+
+        return parts && (parts[1] !== location.protocol || parts[2] !== location.hostname || parts[3] !== location.port);
+    }
+
+    function addTimestamp(url) {
+        var timestamp = 'timestamp=' + new Date().getTime();
+
+        return url + (url.indexOf('?') === -1 ? '?' : '&') + timestamp;
+    }
+
+    function getImageSize(image, callback) {
+        // Modern browsers (ignore Safari, #120 & #509)
+        if (image.naturalWidth && !IS_SAFARI_OR_UIWEBVIEW) {
+            callback(image.naturalWidth, image.naturalHeight);
+            return;
+        }
+
+        // IE8: Don't use `new Image()` here (#319)
+        var newImage = document.createElement('img');
+
+        newImage.onload = function load() {
+            callback(this.width, this.height);
+        };
+
+        newImage.src = image.src;
+    }
+
+    function getTransform(options) {
+        var transforms = [];
+        var translateX = options.translateX;
+        var translateY = options.translateY;
+        var rotate = options.rotate;
+        var scaleX = options.scaleX;
+        var scaleY = options.scaleY;
+
+        if (isNumber(translateX) && translateX !== 0) {
+            transforms.push('translateX(' + translateX + 'px)');
+        }
+
+        if (isNumber(translateY) && translateY !== 0) {
+            transforms.push('translateY(' + translateY + 'px)');
+        }
+
+        // Rotate should come first before scale to match orientation transform
+        if (isNumber(rotate) && rotate !== 0) {
+            transforms.push('rotate(' + rotate + 'deg)');
+        }
+
+        if (isNumber(scaleX) && scaleX !== 1) {
+            transforms.push('scaleX(' + scaleX + ')');
+        }
+
+        if (isNumber(scaleY) && scaleY !== 1) {
+            transforms.push('scaleY(' + scaleY + ')');
+        }
+
+        return transforms.length ? transforms.join(' ') : 'none';
+    }
+
+    function getRotatedSizes(data, isReversed) {
+        var deg = Math.abs(data.degree) % 180;
+        var arc = (deg > 90 ? 180 - deg : deg) * Math.PI / 180;
+        var sinArc = Math.sin(arc);
+        var cosArc = Math.cos(arc);
+        var width = data.width;
+        var height = data.height;
+        var aspectRatio = data.aspectRatio;
+        var newWidth = void 0;
+        var newHeight = void 0;
+
+        if (!isReversed) {
+            newWidth = width * cosArc + height * sinArc;
+            newHeight = width * sinArc + height * cosArc;
+        } else {
+            newWidth = width / (cosArc + sinArc / aspectRatio);
+            newHeight = newWidth / aspectRatio;
+        }
+
+        return {
+            width: newWidth,
+            height: newHeight
+        };
+    }
+
+    function getSourceCanvas(image, data, options) {
+        var canvas = $('<canvas>')[0];
+        var context = canvas.getContext('2d');
+        var dstX = 0;
+        var dstY = 0;
+        var dstWidth = data.naturalWidth;
+        var dstHeight = data.naturalHeight;
+        var rotate = data.rotate;
+        var scaleX = data.scaleX;
+        var scaleY = data.scaleY;
+        var scalable = isNumber(scaleX) && isNumber(scaleY) && (scaleX !== 1 || scaleY !== 1);
+        var rotatable = isNumber(rotate) && rotate !== 0;
+        var advanced = rotatable || scalable;
+        var canvasWidth = dstWidth * Math.abs(scaleX || 1);
+        var canvasHeight = dstHeight * Math.abs(scaleY || 1);
+        var translateX = void 0;
+        var translateY = void 0;
+        var rotated = void 0;
+
+        if (scalable) {
+            translateX = canvasWidth / 2;
+            translateY = canvasHeight / 2;
+        }
+
+        if (rotatable) {
+            rotated = getRotatedSizes({
+                width: canvasWidth,
+                height: canvasHeight,
+                degree: rotate
+            });
+
+            canvasWidth = rotated.width;
+            canvasHeight = rotated.height;
+            translateX = canvasWidth / 2;
+            translateY = canvasHeight / 2;
+        }
+
+        canvas.width = canvasWidth;
+        canvas.height = canvasHeight;
+
+        if (options.fillColor) {
+            context.fillStyle = options.fillColor;
+            context.fillRect(0, 0, canvasWidth, canvasHeight);
+        }
+
+        if (advanced) {
+            dstX = -dstWidth / 2;
+            dstY = -dstHeight / 2;
+
+            context.save();
+            context.translate(translateX, translateY);
+        }
+
+        // Rotate should come first before scale as in the "getTransform" function
+        if (rotatable) {
+            context.rotate(rotate * Math.PI / 180);
+        }
+
+        if (scalable) {
+            context.scale(scaleX, scaleY);
+        }
+
+        context.imageSmoothingEnabled = !!options.imageSmoothingEnabled;
+
+        if (options.imageSmoothingQuality) {
+            context.imageSmoothingQuality = options.imageSmoothingQuality;
+        }
+
+        context.drawImage(image, Math.floor(dstX), Math.floor(dstY), Math.floor(dstWidth), Math.floor(dstHeight));
+
+        if (advanced) {
+            context.restore();
+        }
+
+        return canvas;
+    }
+
+    function getStringFromCharCode(dataView, start, length) {
+        var str = '';
+        var i = void 0;
+
+        for (i = start, length += start; i < length; i += 1) {
+            str += fromCharCode(dataView.getUint8(i));
+        }
+
+        return str;
+    }
+
+    function getOrientation(arrayBuffer) {
+        var dataView = new DataView(arrayBuffer);
+        var length = dataView.byteLength;
+        var orientation = void 0;
+        var exifIDCode = void 0;
+        var tiffOffset = void 0;
+        var firstIFDOffset = void 0;
+        var littleEndian = void 0;
+        var endianness = void 0;
+        var app1Start = void 0;
+        var ifdStart = void 0;
+        var offset = void 0;
+        var i = void 0;
+
+        // Only handle JPEG image (start by 0xFFD8)
+        if (dataView.getUint8(0) === 0xFF && dataView.getUint8(1) === 0xD8) {
+            offset = 2;
+
+            while (offset < length) {
+                if (dataView.getUint8(offset) === 0xFF && dataView.getUint8(offset + 1) === 0xE1) {
+                    app1Start = offset;
+                    break;
+                }
+
+                offset += 1;
+            }
+        }
+
+        if (app1Start) {
+            exifIDCode = app1Start + 4;
+            tiffOffset = app1Start + 10;
+
+            if (getStringFromCharCode(dataView, exifIDCode, 4) === 'Exif') {
+                endianness = dataView.getUint16(tiffOffset);
+                littleEndian = endianness === 0x4949;
+
+                if (littleEndian || endianness === 0x4D4D /* bigEndian */) {
+                    if (dataView.getUint16(tiffOffset + 2, littleEndian) === 0x002A) {
+                        firstIFDOffset = dataView.getUint32(tiffOffset + 4, littleEndian);
+
+                        if (firstIFDOffset >= 0x00000008) {
+                            ifdStart = tiffOffset + firstIFDOffset;
+                        }
+                    }
+                }
+            }
+        }
+
+        if (ifdStart) {
+            length = dataView.getUint16(ifdStart, littleEndian);
+
+            for (i = 0; i < length; i += 1) {
+                offset = ifdStart + i * 12 + 2;
+
+                if (dataView.getUint16(offset, littleEndian) === 0x0112 /* Orientation */) {
+                    // 8 is the offset of the current tag's value
+                    offset += 8;
+
+                    // Get the original orientation value
+                    orientation = dataView.getUint16(offset, littleEndian);
+
+                    // Override the orientation with its default value for Safari (#120)
+                    if (IS_SAFARI_OR_UIWEBVIEW) {
+                        dataView.setUint16(offset, 1, littleEndian);
+                    }
+
+                    break;
+                }
+            }
+        }
+
+        return orientation;
+    }
+
+    function dataURLToArrayBuffer(dataURL) {
+        var base64 = dataURL.replace(REGEXP_DATA_URL_HEAD, '');
+        var binary = atob(base64);
+        var length = binary.length;
+        var arrayBuffer = new ArrayBuffer(length);
+        var dataView = new Uint8Array(arrayBuffer);
+        var i = void 0;
+
+        for (i = 0; i < length; i += 1) {
+            dataView[i] = binary.charCodeAt(i);
+        }
+
+        return arrayBuffer;
+    }
+
+    // Only available for JPEG image
+    function arrayBufferToDataURL(arrayBuffer) {
+        var dataView = new Uint8Array(arrayBuffer);
+        var length = dataView.length;
+        var base64 = '';
+        var i = void 0;
+
+        for (i = 0; i < length; i += 1) {
+            base64 += fromCharCode(dataView[i]);
+        }
+
+        return 'data:image/jpeg;base64,' + btoa(base64);
+    }
+
+    var render = {
+        render: function render() {
+            var self = this;
+
+            self.initContainer();
+            self.initCanvas();
+            self.initCropBox();
+
+            self.renderCanvas();
+
+            if (self.cropped) {
+                self.renderCropBox();
+            }
+        },
+        initContainer: function initContainer() {
+            var self = this;
+            var options = self.options;
+            var $this = self.$element;
+            var $container = self.$container;
+            var $cropper = self.$cropper;
+            var hidden = 'cropper-hidden';
+
+            $cropper.addClass(hidden);
+            $this.removeClass(hidden);
+
+            $cropper.css(self.container = {
+                width: Math.max($container.width(), Number(options.minContainerWidth) || 200),
+                height: Math.max($container.height(), Number(options.minContainerHeight) || 100)
+            });
+
+            $this.addClass(hidden);
+            $cropper.removeClass(hidden);
+        },
+
+
+        // Canvas (image wrapper)
+        initCanvas: function initCanvas() {
+            var self = this;
+            var viewMode = self.options.viewMode;
+            var container = self.container;
+            var containerWidth = container.width;
+            var containerHeight = container.height;
+            var image = self.image;
+            var imageNaturalWidth = image.naturalWidth;
+            var imageNaturalHeight = image.naturalHeight;
+            var is90Degree = Math.abs(image.rotate) % 180 === 90;
+            var naturalWidth = is90Degree ? imageNaturalHeight : imageNaturalWidth;
+            var naturalHeight = is90Degree ? imageNaturalWidth : imageNaturalHeight;
+            var aspectRatio = naturalWidth / naturalHeight;
+            var canvasWidth = containerWidth;
+            var canvasHeight = containerHeight;
+
+            if (containerHeight * aspectRatio > containerWidth) {
+                if (viewMode === 3) {
+                    canvasWidth = containerHeight * aspectRatio;
+                } else {
+                    canvasHeight = containerWidth / aspectRatio;
+                }
+            } else if (viewMode === 3) {
+                canvasHeight = containerWidth / aspectRatio;
+            } else {
+                canvasWidth = containerHeight * aspectRatio;
+            }
+
+            var canvas = {
+                naturalWidth: naturalWidth,
+                naturalHeight: naturalHeight,
+                aspectRatio: aspectRatio,
+                width: canvasWidth,
+                height: canvasHeight
+            };
+
+            canvas.left = (containerWidth - canvasWidth) / 2;
+            canvas.top = (containerHeight - canvasHeight) / 2;
+            canvas.oldLeft = canvas.left;
+            canvas.oldTop = canvas.top;
+
+            self.canvas = canvas;
+            self.limited = viewMode === 1 || viewMode === 2;
+            self.limitCanvas(true, true);
+            self.initialImage = $.extend({}, image);
+            self.initialCanvas = $.extend({}, canvas);
+        },
+        limitCanvas: function limitCanvas(isSizeLimited, isPositionLimited) {
+            var self = this;
+            var options = self.options;
+            var viewMode = options.viewMode;
+            var container = self.container;
+            var containerWidth = container.width;
+            var containerHeight = container.height;
+            var canvas = self.canvas;
+            var aspectRatio = canvas.aspectRatio;
+            var cropBox = self.cropBox;
+            var cropped = self.cropped && cropBox;
+
+            if (isSizeLimited) {
+                var minCanvasWidth = Number(options.minCanvasWidth) || 0;
+                var minCanvasHeight = Number(options.minCanvasHeight) || 0;
+
+                if (viewMode) {
+                    if (viewMode > 1) {
+                        minCanvasWidth = Math.max(minCanvasWidth, containerWidth);
+                        minCanvasHeight = Math.max(minCanvasHeight, containerHeight);
+
+                        if (viewMode === 3) {
+                            if (minCanvasHeight * aspectRatio > minCanvasWidth) {
+                                minCanvasWidth = minCanvasHeight * aspectRatio;
+                            } else {
+                                minCanvasHeight = minCanvasWidth / aspectRatio;
+                            }
+                        }
+                    } else if (minCanvasWidth) {
+                        minCanvasWidth = Math.max(minCanvasWidth, cropped ? cropBox.width : 0);
+                    } else if (minCanvasHeight) {
+                        minCanvasHeight = Math.max(minCanvasHeight, cropped ? cropBox.height : 0);
+                    } else if (cropped) {
+                        minCanvasWidth = cropBox.width;
+                        minCanvasHeight = cropBox.height;
+
+                        if (minCanvasHeight * aspectRatio > minCanvasWidth) {
+                            minCanvasWidth = minCanvasHeight * aspectRatio;
+                        } else {
+                            minCanvasHeight = minCanvasWidth / aspectRatio;
+                        }
+                    }
+                }
+
+                if (minCanvasWidth && minCanvasHeight) {
+                    if (minCanvasHeight * aspectRatio > minCanvasWidth) {
+                        minCanvasHeight = minCanvasWidth / aspectRatio;
+                    } else {
+                        minCanvasWidth = minCanvasHeight * aspectRatio;
+                    }
+                } else if (minCanvasWidth) {
+                    minCanvasHeight = minCanvasWidth / aspectRatio;
+                } else if (minCanvasHeight) {
+                    minCanvasWidth = minCanvasHeight * aspectRatio;
+                }
+
+                canvas.minWidth = minCanvasWidth;
+                canvas.minHeight = minCanvasHeight;
+                canvas.maxWidth = Infinity;
+                canvas.maxHeight = Infinity;
+            }
+
+            if (isPositionLimited) {
+                if (viewMode) {
+                    var newCanvasLeft = containerWidth - canvas.width;
+                    var newCanvasTop = containerHeight - canvas.height;
+
+                    canvas.minLeft = Math.min(0, newCanvasLeft);
+                    canvas.minTop = Math.min(0, newCanvasTop);
+                    canvas.maxLeft = Math.max(0, newCanvasLeft);
+                    canvas.maxTop = Math.max(0, newCanvasTop);
+
+                    if (cropped && self.limited) {
+                        canvas.minLeft = Math.min(cropBox.left, cropBox.left + cropBox.width - canvas.width);
+                        canvas.minTop = Math.min(cropBox.top, cropBox.top + cropBox.height - canvas.height);
+                        canvas.maxLeft = cropBox.left;
+                        canvas.maxTop = cropBox.top;
+
+                        if (viewMode === 2) {
+                            if (canvas.width >= containerWidth) {
+                                canvas.minLeft = Math.min(0, newCanvasLeft);
+                                canvas.maxLeft = Math.max(0, newCanvasLeft);
+                            }
+
+                            if (canvas.height >= containerHeight) {
+                                canvas.minTop = Math.min(0, newCanvasTop);
+                                canvas.maxTop = Math.max(0, newCanvasTop);
+                            }
+                        }
+                    }
+                } else {
+                    canvas.minLeft = -canvas.width;
+                    canvas.minTop = -canvas.height;
+                    canvas.maxLeft = containerWidth;
+                    canvas.maxTop = containerHeight;
+                }
+            }
+        },
+        renderCanvas: function renderCanvas(isChanged) {
+            var self = this;
+            var canvas = self.canvas;
+            var image = self.image;
+            var rotate = image.rotate;
+            var naturalWidth = image.naturalWidth;
+            var naturalHeight = image.naturalHeight;
+
+            if (self.rotated) {
+                self.rotated = false;
+
+                // Computes rotated sizes with image sizes
+                var rotated = getRotatedSizes({
+                    width: image.width,
+                    height: image.height,
+                    degree: rotate
+                });
+                var aspectRatio = rotated.width / rotated.height;
+                var isSquareImage = image.aspectRatio === 1;
+
+                if (isSquareImage || aspectRatio !== canvas.aspectRatio) {
+                    canvas.left -= (rotated.width - canvas.width) / 2;
+                    canvas.top -= (rotated.height - canvas.height) / 2;
+                    canvas.width = rotated.width;
+                    canvas.height = rotated.height;
+                    canvas.aspectRatio = aspectRatio;
+                    canvas.naturalWidth = naturalWidth;
+                    canvas.naturalHeight = naturalHeight;
+
+                    // Computes rotated sizes with natural image sizes
+                    if (isSquareImage && rotate % 90 || rotate % 180) {
+                        var rotated2 = getRotatedSizes({
+                            width: naturalWidth,
+                            height: naturalHeight,
+                            degree: rotate
+                        });
+
+                        canvas.naturalWidth = rotated2.width;
+                        canvas.naturalHeight = rotated2.height;
+                    }
+
+                    self.limitCanvas(true, false);
+                }
+            }
+
+            if (canvas.width > canvas.maxWidth || canvas.width < canvas.minWidth) {
+                canvas.left = canvas.oldLeft;
+            }
+
+            if (canvas.height > canvas.maxHeight || canvas.height < canvas.minHeight) {
+                canvas.top = canvas.oldTop;
+            }
+
+            canvas.width = Math.min(Math.max(canvas.width, canvas.minWidth), canvas.maxWidth);
+            canvas.height = Math.min(Math.max(canvas.height, canvas.minHeight), canvas.maxHeight);
+
+            self.limitCanvas(false, true);
+
+            canvas.left = Math.min(Math.max(canvas.left, canvas.minLeft), canvas.maxLeft);
+            canvas.top = Math.min(Math.max(canvas.top, canvas.minTop), canvas.maxTop);
+            canvas.oldLeft = canvas.left;
+            canvas.oldTop = canvas.top;
+
+            self.$canvas.css({
+                width: canvas.width,
+                height: canvas.height,
+                transform: getTransform({
+                    translateX: canvas.left,
+                    translateY: canvas.top
+                })
+            });
+
+            self.renderImage();
+
+            if (self.cropped && self.limited) {
+                self.limitCropBox(true, true);
+            }
+
+            if (isChanged) {
+                self.output();
+            }
+        },
+        renderImage: function renderImage(isChanged) {
+            var self = this;
+            var canvas = self.canvas;
+            var image = self.image;
+            var reversed = void 0;
+
+            if (image.rotate) {
+                reversed = getRotatedSizes({
+                    width: canvas.width,
+                    height: canvas.height,
+                    degree: image.rotate,
+                    aspectRatio: image.aspectRatio
+                }, true);
+            }
+
+            $.extend(image, reversed ? {
+                width: reversed.width,
+                height: reversed.height,
+                left: (canvas.width - reversed.width) / 2,
+                top: (canvas.height - reversed.height) / 2
+            } : {
+                width: canvas.width,
+                height: canvas.height,
+                left: 0,
+                top: 0
+            });
+
+            self.$clone.css({
+                width: image.width,
+                height: image.height,
+                transform: getTransform($.extend({
+                    translateX: image.left,
+                    translateY: image.top
+                }, image))
+            });
+
+            if (isChanged) {
+                self.output();
+            }
+        },
+        initCropBox: function initCropBox() {
+            var self = this;
+            var options = self.options;
+            var canvas = self.canvas;
+            var aspectRatio = options.aspectRatio;
+            var autoCropArea = Number(options.autoCropArea) || 0.8;
+            var cropBox = {
+                width: canvas.width,
+                height: canvas.height
+            };
+
+            if (aspectRatio) {
+                if (canvas.height * aspectRatio > canvas.width) {
+                    cropBox.height = cropBox.width / aspectRatio;
+                } else {
+                    cropBox.width = cropBox.height * aspectRatio;
+                }
+            }
+
+            self.cropBox = cropBox;
+            self.limitCropBox(true, true);
+
+            // Initialize auto crop area
+            cropBox.width = Math.min(Math.max(cropBox.width, cropBox.minWidth), cropBox.maxWidth);
+            cropBox.height = Math.min(Math.max(cropBox.height, cropBox.minHeight), cropBox.maxHeight);
+
+            // The width of auto crop area must large than "minWidth", and the height too. (#164)
+            cropBox.width = Math.max(cropBox.minWidth, cropBox.width * autoCropArea);
+            cropBox.height = Math.max(cropBox.minHeight, cropBox.height * autoCropArea);
+            cropBox.left = canvas.left + (canvas.width - cropBox.width) / 2;
+            cropBox.top = canvas.top + (canvas.height - cropBox.height) / 2;
+            cropBox.oldLeft = cropBox.left;
+            cropBox.oldTop = cropBox.top;
+
+            self.initialCropBox = $.extend({}, cropBox);
+        },
+        limitCropBox: function limitCropBox(isSizeLimited, isPositionLimited) {
+            var self = this;
+            var options = self.options;
+            var aspectRatio = options.aspectRatio;
+            var container = self.container;
+            var containerWidth = container.width;
+            var containerHeight = container.height;
+            var canvas = self.canvas;
+            var cropBox = self.cropBox;
+            var limited = self.limited;
+
+            if (isSizeLimited) {
+                var minCropBoxWidth = Number(options.minCropBoxWidth) || 0;
+                var minCropBoxHeight = Number(options.minCropBoxHeight) || 0;
+                var maxCropBoxWidth = Math.min(containerWidth, limited ? canvas.width : containerWidth);
+                var maxCropBoxHeight = Math.min(containerHeight, limited ? canvas.height : containerHeight);
+
+                // The min/maxCropBoxWidth/Height must be less than containerWidth/Height
+                minCropBoxWidth = Math.min(minCropBoxWidth, containerWidth);
+                minCropBoxHeight = Math.min(minCropBoxHeight, containerHeight);
+
+                if (aspectRatio) {
+                    if (minCropBoxWidth && minCropBoxHeight) {
+                        if (minCropBoxHeight * aspectRatio > minCropBoxWidth) {
+                            minCropBoxHeight = minCropBoxWidth / aspectRatio;
+                        } else {
+                            minCropBoxWidth = minCropBoxHeight * aspectRatio;
+                        }
+                    } else if (minCropBoxWidth) {
+                        minCropBoxHeight = minCropBoxWidth / aspectRatio;
+                    } else if (minCropBoxHeight) {
+                        minCropBoxWidth = minCropBoxHeight * aspectRatio;
+                    }
+
+                    if (maxCropBoxHeight * aspectRatio > maxCropBoxWidth) {
+                        maxCropBoxHeight = maxCropBoxWidth / aspectRatio;
+                    } else {
+                        maxCropBoxWidth = maxCropBoxHeight * aspectRatio;
+                    }
+                }
+
+                // The minWidth/Height must be less than maxWidth/Height
+                cropBox.minWidth = Math.min(minCropBoxWidth, maxCropBoxWidth);
+                cropBox.minHeight = Math.min(minCropBoxHeight, maxCropBoxHeight);
+                cropBox.maxWidth = maxCropBoxWidth;
+                cropBox.maxHeight = maxCropBoxHeight;
+            }
+
+            if (isPositionLimited) {
+                if (limited) {
+                    cropBox.minLeft = Math.max(0, canvas.left);
+                    cropBox.minTop = Math.max(0, canvas.top);
+                    cropBox.maxLeft = Math.min(containerWidth, canvas.left + canvas.width) - cropBox.width;
+                    cropBox.maxTop = Math.min(containerHeight, canvas.top + canvas.height) - cropBox.height;
+                } else {
+                    cropBox.minLeft = 0;
+                    cropBox.minTop = 0;
+                    cropBox.maxLeft = containerWidth - cropBox.width;
+                    cropBox.maxTop = containerHeight - cropBox.height;
+                }
+            }
+        },
+        renderCropBox: function renderCropBox() {
+            var self = this;
+            var options = self.options;
+            var container = self.container;
+            var containerWidth = container.width;
+            var containerHeight = container.height;
+            var cropBox = self.cropBox;
+
+            if (cropBox.width > cropBox.maxWidth || cropBox.width < cropBox.minWidth) {
+                cropBox.left = cropBox.oldLeft;
+            }
+
+            if (cropBox.height > cropBox.maxHeight || cropBox.height < cropBox.minHeight) {
+                cropBox.top = cropBox.oldTop;
+            }
+
+            cropBox.width = Math.min(Math.max(cropBox.width, cropBox.minWidth), cropBox.maxWidth);
+            cropBox.height = Math.min(Math.max(cropBox.height, cropBox.minHeight), cropBox.maxHeight);
+
+            self.limitCropBox(false, true);
+
+            cropBox.left = Math.min(Math.max(cropBox.left, cropBox.minLeft), cropBox.maxLeft);
+            cropBox.top = Math.min(Math.max(cropBox.top, cropBox.minTop), cropBox.maxTop);
+            cropBox.oldLeft = cropBox.left;
+            cropBox.oldTop = cropBox.top;
+
+            if (options.movable && options.cropBoxMovable) {
+                // Turn to move the canvas when the crop box is equal to the container
+                self.$face.data('action', cropBox.width === containerWidth && cropBox.height === containerHeight ? 'move' : 'all');
+            }
+
+            self.$cropBox.css({
+                width: cropBox.width,
+                height: cropBox.height,
+                transform: getTransform({
+                    translateX: cropBox.left,
+                    translateY: cropBox.top
+                })
+            });
+
+            if (self.cropped && self.limited) {
+                self.limitCanvas(true, true);
+            }
+
+            if (!self.disabled) {
+                self.output();
+            }
+        },
+        output: function output() {
+            var self = this;
+
+            self.preview();
+
+            if (self.completed) {
+                self.trigger('crop', self.getData());
+            }
+        }
+    };
+
+    var DATA_PREVIEW = 'preview';
+
+    var preview = {
+        initPreview: function initPreview() {
+            var self = this;
+            var crossOrigin = self.crossOrigin;
+            var url = crossOrigin ? self.crossOriginUrl : self.url;
+            var image = document.createElement('img');
+
+            if (crossOrigin) {
+                image.crossOrigin = crossOrigin;
+            }
+
+            image.src = url;
+
+            var $clone2 = $(image);
+
+            self.$preview = $(self.options.preview);
+            self.$clone2 = $clone2;
+            self.$viewBox.html($clone2);
+            self.$preview.each(function (i, element) {
+                var $this = $(element);
+                var img = document.createElement('img');
+
+                // Save the original size for recover
+                $this.data(DATA_PREVIEW, {
+                    width: $this.width(),
+                    height: $this.height(),
+                    html: $this.html()
+                });
+
+                if (crossOrigin) {
+                    img.crossOrigin = crossOrigin;
+                }
+
+                img.src = url;
+
+                /**
+                 * Override img element styles
+                 * Add `display:block` to avoid margin top issue
+                 * Add `height:auto` to override `height` attribute on IE8
+                 * (Occur only when margin-top <= -height)
+                 */
+                img.style.cssText = 'display:block;' + 'width:100%;' + 'height:auto;' + 'min-width:0!important;' + 'min-height:0!important;' + 'max-width:none!important;' + 'max-height:none!important;' + 'image-orientation:0deg!important;"';
+
+                $this.html(img);
+            });
+        },
+        resetPreview: function resetPreview() {
+            this.$preview.each(function (i, element) {
+                var $this = $(element);
+                var data = $this.data(DATA_PREVIEW);
+
+                $this.css({
+                    width: data.width,
+                    height: data.height
+                }).html(data.html).removeData(DATA_PREVIEW);
+            });
+        },
+        preview: function preview() {
+            var self = this;
+            var image = self.image;
+            var canvas = self.canvas;
+            var cropBox = self.cropBox;
+            var cropBoxWidth = cropBox.width;
+            var cropBoxHeight = cropBox.height;
+            var width = image.width;
+            var height = image.height;
+            var left = cropBox.left - canvas.left - image.left;
+            var top = cropBox.top - canvas.top - image.top;
+
+            if (!self.cropped || self.disabled) {
+                return;
+            }
+
+            self.$clone2.css({
+                width: width,
+                height: height,
+                transform: getTransform($.extend({
+                    translateX: -left,
+                    translateY: -top
+                }, image))
+            });
+
+            self.$preview.each(function (i, element) {
+                var $this = $(element);
+                var data = $this.data(DATA_PREVIEW);
+                var originalWidth = data.width;
+                var originalHeight = data.height;
+                var newWidth = originalWidth;
+                var newHeight = originalHeight;
+                var ratio = 1;
+
+                if (cropBoxWidth) {
+                    ratio = originalWidth / cropBoxWidth;
+                    newHeight = cropBoxHeight * ratio;
+                }
+
+                if (cropBoxHeight && newHeight > originalHeight) {
+                    ratio = originalHeight / cropBoxHeight;
+                    newWidth = cropBoxWidth * ratio;
+                    newHeight = originalHeight;
+                }
+
+                $this.css({
+                    width: newWidth,
+                    height: newHeight
+                }).find('img').css({
+                    width: width * ratio,
+                    height: height * ratio,
+                    transform: getTransform($.extend({
+                        translateX: -left * ratio,
+                        translateY: -top * ratio
+                    }, image))
+                });
+            });
+        }
+    };
+
+    // Globals
+    var PointerEvent = typeof window !== 'undefined' ? window.PointerEvent : null;
+
+    // Events
+    var EVENT_POINTER_DOWN = PointerEvent ? 'pointerdown' : 'touchstart mousedown';
+    var EVENT_POINTER_MOVE = PointerEvent ? 'pointermove' : 'touchmove mousemove';
+    var EVENT_POINTER_UP = PointerEvent ? ' pointerup pointercancel' : 'touchend touchcancel mouseup';
+    var EVENT_WHEEL = 'wheel mousewheel DOMMouseScroll';
+    var EVENT_DBLCLICK = 'dblclick';
+    var EVENT_RESIZE = 'resize';
+    var EVENT_CROP_START = 'cropstart';
+    var EVENT_CROP_MOVE = 'cropmove';
+    var EVENT_CROP_END = 'cropend';
+    var EVENT_CROP = 'crop';
+    var EVENT_ZOOM = 'zoom';
+
+    var events = {
+        bind: function bind() {
+            var self = this;
+            var options = self.options;
+            var $this = self.$element;
+            var $cropper = self.$cropper;
+
+            if ($.isFunction(options.cropstart)) {
+                $this.on(EVENT_CROP_START, options.cropstart);
+            }
+
+            if ($.isFunction(options.cropmove)) {
+                $this.on(EVENT_CROP_MOVE, options.cropmove);
+            }
+
+            if ($.isFunction(options.cropend)) {
+                $this.on(EVENT_CROP_END, options.cropend);
+            }
+
+            if ($.isFunction(options.crop)) {
+                $this.on(EVENT_CROP, options.crop);
+            }
+
+            if ($.isFunction(options.zoom)) {
+                $this.on(EVENT_ZOOM, options.zoom);
+            }
+
+            $cropper.on(EVENT_POINTER_DOWN, proxy(self.cropStart, this));
+
+            if (options.zoomable && options.zoomOnWheel) {
+                $cropper.on(EVENT_WHEEL, proxy(self.wheel, this));
+            }
+
+            if (options.toggleDragModeOnDblclick) {
+                $cropper.on(EVENT_DBLCLICK, proxy(self.dblclick, this));
+            }
+
+            $(document).on(EVENT_POINTER_MOVE, self.onCropMove = proxy(self.cropMove, this)).on(EVENT_POINTER_UP, self.onCropEnd = proxy(self.cropEnd, this));
+
+            if (options.responsive) {
+                $(window).on(EVENT_RESIZE, self.onResize = proxy(self.resize, this));
+            }
+        },
+        unbind: function unbind() {
+            var self = this;
+            var options = self.options;
+            var $this = self.$element;
+            var $cropper = self.$cropper;
+
+            if ($.isFunction(options.cropstart)) {
+                $this.off(EVENT_CROP_START, options.cropstart);
+            }
+
+            if ($.isFunction(options.cropmove)) {
+                $this.off(EVENT_CROP_MOVE, options.cropmove);
+            }
+
+            if ($.isFunction(options.cropend)) {
+                $this.off(EVENT_CROP_END, options.cropend);
+            }
+
+            if ($.isFunction(options.crop)) {
+                $this.off(EVENT_CROP, options.crop);
+            }
+
+            if ($.isFunction(options.zoom)) {
+                $this.off(EVENT_ZOOM, options.zoom);
+            }
+
+            $cropper.off(EVENT_POINTER_DOWN, self.cropStart);
+
+            if (options.zoomable && options.zoomOnWheel) {
+                $cropper.off(EVENT_WHEEL, self.wheel);
+            }
+
+            if (options.toggleDragModeOnDblclick) {
+                $cropper.off(EVENT_DBLCLICK, self.dblclick);
+            }
+
+            $(document).off(EVENT_POINTER_MOVE, self.onCropMove).off(EVENT_POINTER_UP, self.onCropEnd);
+
+            if (options.responsive) {
+                $(window).off(EVENT_RESIZE, self.onResize);
+            }
+        }
+    };
+
+    var REGEXP_ACTIONS = /^(e|w|s|n|se|sw|ne|nw|all|crop|move|zoom)$/;
+
+    function getPointer(_ref, endOnly) {
+        var pageX = _ref.pageX,
+            pageY = _ref.pageY;
+
+        var end = {
+            endX: pageX,
+            endY: pageY
+        };
+
+        if (endOnly) {
+            return end;
+        }
+
+        return $.extend({
+            startX: pageX,
+            startY: pageY
+        }, end);
+    }
+
+    var handlers = {
+        resize: function resize() {
+            var self = this;
+            var options = self.options;
+            var $container = self.$container;
+            var container = self.container;
+            var minContainerWidth = Number(options.minContainerWidth) || 200;
+            var minContainerHeight = Number(options.minContainerHeight) || 100;
+
+            if (self.disabled || container.width === minContainerWidth || container.height === minContainerHeight) {
+                return;
+            }
+
+            var ratio = $container.width() / container.width;
+
+            // Resize when width changed or height changed
+            if (ratio !== 1 || $container.height() !== container.height) {
+                var canvasData = void 0;
+                var cropBoxData = void 0;
+
+                if (options.restore) {
+                    canvasData = self.getCanvasData();
+                    cropBoxData = self.getCropBoxData();
+                }
+
+                self.render();
+
+                if (options.restore) {
+                    self.setCanvasData($.each(canvasData, function (i, n) {
+                        canvasData[i] = n * ratio;
+                    }));
+                    self.setCropBoxData($.each(cropBoxData, function (i, n) {
+                        cropBoxData[i] = n * ratio;
+                    }));
+                }
+            }
+        },
+        dblclick: function dblclick() {
+            var self = this;
+
+            if (self.disabled || self.options.dragMode === 'none') {
+                return;
+            }
+
+            self.setDragMode(self.$dragBox.hasClass('cropper-crop') ? 'move' : 'crop');
+        },
+        wheel: function wheel(event) {
+            var self = this;
+            var e = event.originalEvent || event;
+            var ratio = Number(self.options.wheelZoomRatio) || 0.1;
+
+            if (self.disabled) {
+                return;
+            }
+
+            event.preventDefault();
+
+            // Limit wheel speed to prevent zoom too fast
+            if (self.wheeling) {
+                return;
+            }
+
+            self.wheeling = true;
+
+            setTimeout(function () {
+                self.wheeling = false;
+            }, 50);
+
+            var delta = 1;
+
+            if (e.deltaY) {
+                delta = e.deltaY > 0 ? 1 : -1;
+            } else if (e.wheelDelta) {
+                delta = -e.wheelDelta / 120;
+            } else if (e.detail) {
+                delta = e.detail > 0 ? 1 : -1;
+            }
+
+            self.zoom(-delta * ratio, event);
+        },
+        cropStart: function cropStart(e) {
+            var self = this;
+
+            if (self.disabled) {
+                return;
+            }
+
+            var options = self.options;
+            var pointers = self.pointers;
+            var originalEvent = e.originalEvent;
+            var action = void 0;
+
+            if (originalEvent && originalEvent.changedTouches) {
+                // Handle touch event
+                $.each(originalEvent.changedTouches, function (i, touch) {
+                    pointers[touch.identifier] = getPointer(touch);
+                });
+            } else {
+                // Handle mouse event and pointer event
+                pointers[originalEvent && originalEvent.pointerId || 0] = getPointer(originalEvent || e);
+            }
+
+            if (objectKeys(pointers).length > 1 && options.zoomable && options.zoomOnTouch) {
+                action = 'zoom';
+            } else {
+                action = $(e.target).data('action');
+            }
+
+            if (!REGEXP_ACTIONS.test(action)) {
+                return;
+            }
+
+            if (self.trigger('cropstart', {
+                originalEvent: originalEvent,
+                action: action
+            }).isDefaultPrevented()) {
+                return;
+            }
+
+            e.preventDefault();
+
+            self.action = action;
+            self.cropping = false;
+
+            if (action === 'crop') {
+                self.cropping = true;
+                self.$dragBox.addClass('cropper-modal');
+            }
+        },
+        cropMove: function cropMove(e) {
+            var self = this;
+            var action = self.action;
+
+            if (self.disabled || !action) {
+                return;
+            }
+
+            var pointers = self.pointers;
+            var originalEvent = e.originalEvent;
+
+            e.preventDefault();
+
+            if (self.trigger('cropmove', {
+                originalEvent: originalEvent,
+                action: action
+            }).isDefaultPrevented()) {
+                return;
+            }
+
+            if (originalEvent && originalEvent.changedTouches) {
+                $.each(originalEvent.changedTouches, function (i, touch) {
+                    $.extend(pointers[touch.identifier], getPointer(touch, true));
+                });
+            } else {
+                $.extend(pointers[originalEvent && originalEvent.pointerId || 0], getPointer(originalEvent || e, true));
+            }
+
+            self.change(e);
+        },
+        cropEnd: function cropEnd(e) {
+            var self = this;
+
+            if (self.disabled) {
+                return;
+            }
+
+            var action = self.action;
+            var pointers = self.pointers;
+            var originalEvent = e.originalEvent;
+
+            if (originalEvent && originalEvent.changedTouches) {
+                $.each(originalEvent.changedTouches, function (i, touch) {
+                    delete pointers[touch.identifier];
+                });
+            } else {
+                delete pointers[originalEvent && originalEvent.pointerId || 0];
+            }
+
+            if (!action) {
+                return;
+            }
+
+            e.preventDefault();
+
+            if (!objectKeys(pointers).length) {
+                self.action = '';
+            }
+
+            if (self.cropping) {
+                self.cropping = false;
+                self.$dragBox.toggleClass('cropper-modal', self.cropped && self.options.modal);
+            }
+
+            self.trigger('cropend', {
+                originalEvent: originalEvent,
+                action: action
+            });
+        }
+    };
+
+    // Actions
+    var ACTION_EAST = 'e';
+    var ACTION_WEST = 'w';
+    var ACTION_SOUTH = 's';
+    var ACTION_NORTH = 'n';
+    var ACTION_SOUTH_EAST = 'se';
+    var ACTION_SOUTH_WEST = 'sw';
+    var ACTION_NORTH_EAST = 'ne';
+    var ACTION_NORTH_WEST = 'nw';
+
+    function getMaxZoomRatio(pointers) {
+        var pointers2 = $.extend({}, pointers);
+        var ratios = [];
+
+        $.each(pointers, function (pointerId, pointer) {
+            delete pointers2[pointerId];
+
+            $.each(pointers2, function (pointerId2, pointer2) {
+                var x1 = Math.abs(pointer.startX - pointer2.startX);
+                var y1 = Math.abs(pointer.startY - pointer2.startY);
+                var x2 = Math.abs(pointer.endX - pointer2.endX);
+                var y2 = Math.abs(pointer.endY - pointer2.endY);
+                var z1 = Math.sqrt(x1 * x1 + y1 * y1);
+                var z2 = Math.sqrt(x2 * x2 + y2 * y2);
+                var ratio = (z2 - z1) / z1;
+
+                ratios.push(ratio);
+            });
+        });
+
+        ratios.sort(function (a, b) {
+            return Math.abs(a) < Math.abs(b);
+        });
+
+        return ratios[0];
+    }
+
+    var change = {
+        change: function change(e) {
+            var self = this;
+            var options = self.options;
+            var pointers = self.pointers;
+            var pointer = pointers[objectKeys(pointers)[0]];
+            var container = self.container;
+            var canvas = self.canvas;
+            var cropBox = self.cropBox;
+            var action = self.action;
+            var aspectRatio = options.aspectRatio;
+            var width = cropBox.width;
+            var height = cropBox.height;
+            var left = cropBox.left;
+            var top = cropBox.top;
+            var right = left + width;
+            var bottom = top + height;
+            var minLeft = 0;
+            var minTop = 0;
+            var maxWidth = container.width;
+            var maxHeight = container.height;
+            var renderable = true;
+            var offset = void 0;
+
+            // Locking aspect ratio in "free mode" by holding shift key (#259)
+            if (!aspectRatio && e.shiftKey) {
+                aspectRatio = width && height ? width / height : 1;
+            }
+
+            if (self.limited) {
+                minLeft = cropBox.minLeft;
+                minTop = cropBox.minTop;
+                maxWidth = minLeft + Math.min(container.width, canvas.width, canvas.left + canvas.width);
+                maxHeight = minTop + Math.min(container.height, canvas.height, canvas.top + canvas.height);
+            }
+
+            var range = {
+                x: pointer.endX - pointer.startX,
+                y: pointer.endY - pointer.startY
+            };
+
+            switch (action) {
+                // Move crop box
+                case 'all':
+                    left += range.x;
+                    top += range.y;
+                    break;
+
+                    // Resize crop box
+                case ACTION_EAST:
+                    if (range.x >= 0 && (right >= maxWidth || aspectRatio && (top <= minTop || bottom >= maxHeight))) {
+                        renderable = false;
+                        break;
+                    }
+
+                    if (right + range.x > maxWidth) {
+                        range.x = maxWidth - right;
+                    }
+
+                    width += range.x;
+
+                    if (aspectRatio) {
+                        height = width / aspectRatio;
+                        top -= range.x / aspectRatio / 2;
+                    }
+
+                    if (width < 0) {
+                        action = ACTION_WEST;
+                        width = 0;
+                    }
+
+                    break;
+
+                case ACTION_NORTH:
+                    if (range.y <= 0 && (top <= minTop || aspectRatio && (left <= minLeft || right >= maxWidth))) {
+                        renderable = false;
+                        break;
+                    }
+
+                    if (top + range.y < minTop) {
+                        range.y = minTop - top;
+                    }
+
+                    height -= range.y;
+                    top += range.y;
+
+                    if (aspectRatio) {
+                        width = height * aspectRatio;
+                        left += range.y * aspectRatio / 2;
+                    }
+
+                    if (height < 0) {
+                        action = ACTION_SOUTH;
+                        height = 0;
+                    }
+
+                    break;
+
+                case ACTION_WEST:
+                    if (range.x <= 0 && (left <= minLeft || aspectRatio && (top <= minTop || bottom >= maxHeight))) {
+                        renderable = false;
+                        break;
+                    }
+
+                    if (left + range.x < minLeft) {
+                        range.x = minLeft - left;
+                    }
+
+                    width -= range.x;
+                    left += range.x;
+
+                    if (aspectRatio) {
+                        height = width / aspectRatio;
+                        top += range.x / aspectRatio / 2;
+                    }
+
+                    if (width < 0) {
+                        action = ACTION_EAST;
+                        width = 0;
+                    }
+
+                    break;
+
+                case ACTION_SOUTH:
+                    if (range.y >= 0 && (bottom >= maxHeight || aspectRatio && (left <= minLeft || right >= maxWidth))) {
+                        renderable = false;
+                        break;
+                    }
+
+                    if (bottom + range.y > maxHeight) {
+                        range.y = maxHeight - bottom;
+                    }
+
+                    height += range.y;
+
+                    if (aspectRatio) {
+                        width = height * aspectRatio;
+                        left -= range.y * aspectRatio / 2;
+                    }
+
+                    if (height < 0) {
+                        action = ACTION_NORTH;
+                        height = 0;
+                    }
+
+                    break;
+
+                case ACTION_NORTH_EAST:
+                    if (aspectRatio) {
+                        if (range.y <= 0 && (top <= minTop || right >= maxWidth)) {
+                            renderable = false;
+                            break;
+                        }
+
+                        height -= range.y;
+                        top += range.y;
+                        width = height * aspectRatio;
+                    } else {
+                        if (range.x >= 0) {
+                            if (right < maxWidth) {
+                                width += range.x;
+                            } else if (range.y <= 0 && top <= minTop) {
+                                renderable = false;
+                            }
+                        } else {
+                            width += range.x;
+                        }
+
+                        if (range.y <= 0) {
+                            if (top > minTop) {
+                                height -= range.y;
+                                top += range.y;
+                            }
+                        } else {
+                            height -= range.y;
+                            top += range.y;
+                        }
+                    }
+
+                    if (width < 0 && height < 0) {
+                        action = ACTION_SOUTH_WEST;
+                        height = 0;
+                        width = 0;
+                    } else if (width < 0) {
+                        action = ACTION_NORTH_WEST;
+                        width = 0;
+                    } else if (height < 0) {
+                        action = ACTION_SOUTH_EAST;
+                        height = 0;
+                    }
+
+                    break;
+
+                case ACTION_NORTH_WEST:
+                    if (aspectRatio) {
+                        if (range.y <= 0 && (top <= minTop || left <= minLeft)) {
+                            renderable = false;
+                            break;
+                        }
+
+                        height -= range.y;
+                        top += range.y;
+                        width = height * aspectRatio;
+                        left += range.y * aspectRatio;
+                    } else {
+                        if (range.x <= 0) {
+                            if (left > minLeft) {
+                                width -= range.x;
+                                left += range.x;
+                            } else if (range.y <= 0 && top <= minTop) {
+                                renderable = false;
+                            }
+                        } else {
+                            width -= range.x;
+                            left += range.x;
+                        }
+
+                        if (range.y <= 0) {
+                            if (top > minTop) {
+                                height -= range.y;
+                                top += range.y;
+                            }
+                        } else {
+                            height -= range.y;
+                            top += range.y;
+                        }
+                    }
+
+                    if (width < 0 && height < 0) {
+                        action = ACTION_SOUTH_EAST;
+                        height = 0;
+                        width = 0;
+                    } else if (width < 0) {
+                        action = ACTION_NORTH_EAST;
+                        width = 0;
+                    } else if (height < 0) {
+                        action = ACTION_SOUTH_WEST;
+                        height = 0;
+                    }
+
+                    break;
+
+                case ACTION_SOUTH_WEST:
+                    if (aspectRatio) {
+                        if (range.x <= 0 && (left <= minLeft || bottom >= maxHeight)) {
+                            renderable = false;
+                            break;
+                        }
+
+                        width -= range.x;
+                        left += range.x;
+                        height = width / aspectRatio;
+                    } else {
+                        if (range.x <= 0) {
+                            if (left > minLeft) {
+                                width -= range.x;
+                                left += range.x;
+                            } else if (range.y >= 0 && bottom >= maxHeight) {
+                                renderable = false;
+                            }
+                        } else {
+                            width -= range.x;
+                            left += range.x;
+                        }
+
+                        if (range.y >= 0) {
+                            if (bottom < maxHeight) {
+                                height += range.y;
+                            }
+                        } else {
+                            height += range.y;
+                        }
+                    }
+
+                    if (width < 0 && height < 0) {
+                        action = ACTION_NORTH_EAST;
+                        height = 0;
+                        width = 0;
+                    } else if (width < 0) {
+                        action = ACTION_SOUTH_EAST;
+                        width = 0;
+                    } else if (height < 0) {
+                        action = ACTION_NORTH_WEST;
+                        height = 0;
+                    }
+
+                    break;
+
+                case ACTION_SOUTH_EAST:
+                    if (aspectRatio) {
+                        if (range.x >= 0 && (right >= maxWidth || bottom >= maxHeight)) {
+                            renderable = false;
+                            break;
+                        }
+
+                        width += range.x;
+                        height = width / aspectRatio;
+                    } else {
+                        if (range.x >= 0) {
+                            if (right < maxWidth) {
+                                width += range.x;
+                            } else if (range.y >= 0 && bottom >= maxHeight) {
+                                renderable = false;
+                            }
+                        } else {
+                            width += range.x;
+                        }
+
+                        if (range.y >= 0) {
+                            if (bottom < maxHeight) {
+                                height += range.y;
+                            }
+                        } else {
+                            height += range.y;
+                        }
+                    }
+
+                    if (width < 0 && height < 0) {
+                        action = ACTION_NORTH_WEST;
+                        height = 0;
+                        width = 0;
+                    } else if (width < 0) {
+                        action = ACTION_SOUTH_WEST;
+                        width = 0;
+                    } else if (height < 0) {
+                        action = ACTION_NORTH_EAST;
+                        height = 0;
+                    }
+
+                    break;
+
+                    // Move canvas
+                case 'move':
+                    self.move(range.x, range.y);
+                    renderable = false;
+                    break;
+
+                    // Zoom canvas
+                case 'zoom':
+                    self.zoom(getMaxZoomRatio(pointers), e.originalEvent);
+                    renderable = false;
+                    break;
+
+                    // Create crop box
+                case 'crop':
+                    if (!range.x || !range.y) {
+                        renderable = false;
+                        break;
+                    }
+
+                    offset = self.$cropper.offset();
+                    left = pointer.startX - offset.left;
+                    top = pointer.startY - offset.top;
+                    width = cropBox.minWidth;
+                    height = cropBox.minHeight;
+
+                    if (range.x > 0) {
+                        action = range.y > 0 ? ACTION_SOUTH_EAST : ACTION_NORTH_EAST;
+                    } else if (range.x < 0) {
+                        left -= width;
+                        action = range.y > 0 ? ACTION_SOUTH_WEST : ACTION_NORTH_WEST;
+                    }
+
+                    if (range.y < 0) {
+                        top -= height;
+                    }
+
+                    // Show the crop box if is hidden
+                    if (!self.cropped) {
+                        self.$cropBox.removeClass('cropper-hidden');
+                        self.cropped = true;
+
+                        if (self.limited) {
+                            self.limitCropBox(true, true);
+                        }
+                    }
+
+                    break;
+
+                default:
+            }
+
+            if (renderable) {
+                cropBox.width = width;
+                cropBox.height = height;
+                cropBox.left = left;
+                cropBox.top = top;
+                self.action = action;
+                self.renderCropBox();
+            }
+
+            // Override
+            $.each(pointers, function (i, p) {
+                p.startX = p.endX;
+                p.startY = p.endY;
+            });
+        }
+    };
+
+    function _toConsumableArray(arr) { if (Array.isArray(arr)) { for (var i = 0, arr2 = Array(arr.length) ; i < arr.length; i++) { arr2[i] = arr[i]; } return arr2; } else { return Array.from(arr); } }
+
+    function getPointersCenter(pointers) {
+        var pageX = 0;
+        var pageY = 0;
+        var count = 0;
+
+        $.each(pointers, function (i, _ref) {
+            var startX = _ref.startX,
+                startY = _ref.startY;
+
+            pageX += startX;
+            pageY += startY;
+            count += 1;
+        });
+
+        pageX /= count;
+        pageY /= count;
+
+        return {
+            pageX: pageX,
+            pageY: pageY
+        };
+    }
+
+    var methods = {
+        // Show the crop box manually
+        crop: function crop() {
+            var self = this;
+
+            if (!self.ready || self.disabled) {
+                return;
+            }
+
+            if (!self.cropped) {
+                self.cropped = true;
+                self.limitCropBox(true, true);
+
+                if (self.options.modal) {
+                    self.$dragBox.addClass('cropper-modal');
+                }
+
+                self.$cropBox.removeClass('cropper-hidden');
+            }
+
+            self.setCropBoxData(self.initialCropBox);
+        },
+
+
+        // Reset the image and crop box to their initial states
+        reset: function reset() {
+            var self = this;
+
+            if (!self.ready || self.disabled) {
+                return;
+            }
+
+            self.image = $.extend({}, self.initialImage);
+            self.canvas = $.extend({}, self.initialCanvas);
+            self.cropBox = $.extend({}, self.initialCropBox);
+
+            self.renderCanvas();
+
+            if (self.cropped) {
+                self.renderCropBox();
+            }
+        },
+
+
+        // Clear the crop box
+        clear: function clear() {
+            var self = this;
+
+            if (!self.cropped || self.disabled) {
+                return;
+            }
+
+            $.extend(self.cropBox, {
+                left: 0,
+                top: 0,
+                width: 0,
+                height: 0
+            });
+
+            self.cropped = false;
+            self.renderCropBox();
+
+            self.limitCanvas(true, true);
+
+            // Render canvas after crop box rendered
+            self.renderCanvas();
+
+            self.$dragBox.removeClass('cropper-modal');
+            self.$cropBox.addClass('cropper-hidden');
+        },
+
+
+        /**
+         * Replace the image's src and rebuild the cropper
+         *
+         * @param {String} url
+         * @param {Boolean} onlyColorChanged (optional)
+         */
+        replace: function replace(url, onlyColorChanged) {
+            var self = this;
+
+            if (!self.disabled && url) {
+                if (self.isImg) {
+                    self.$element.attr('src', url);
+                }
+
+                if (onlyColorChanged) {
+                    self.url = url;
+                    self.$clone.attr('src', url);
+
+                    if (self.ready) {
+                        self.$preview.find('img').add(self.$clone2).attr('src', url);
+                    }
+                } else {
+                    if (self.isImg) {
+                        self.replaced = true;
+                    }
+
+                    // Clear previous data
+                    self.options.data = null;
+                    self.load(url);
+                }
+            }
+        },
+
+
+        // Enable (unfreeze) the cropper
+        enable: function enable() {
+            var self = this;
+
+            if (self.ready) {
+                self.disabled = false;
+                self.$cropper.removeClass('cropper-disabled');
+            }
+        },
+
+
+        // Disable (freeze) the cropper
+        disable: function disable() {
+            var self = this;
+
+            if (self.ready) {
+                self.disabled = true;
+                self.$cropper.addClass('cropper-disabled');
+            }
+        },
+
+
+        // Destroy the cropper and remove the instance from the image
+        destroy: function destroy() {
+            var self = this;
+            var $this = self.$element;
+
+            if (self.loaded) {
+                if (self.isImg && self.replaced) {
+                    $this.attr('src', self.originalUrl);
+                }
+
+                self.unbuild();
+                $this.removeClass('cropper-hidden');
+            } else if (self.isImg) {
+                $this.off('load', self.start);
+            } else if (self.$clone) {
+                self.$clone.remove();
+            }
+
+            $this.removeData('cropper');
+        },
+
+
+        /**
+         * Move the canvas with relative offsets
+         *
+         * @param {Number} offsetX
+         * @param {Number} offsetY (optional)
+         */
+        move: function move(offsetX, offsetY) {
+            var self = this;
+            var canvas = self.canvas;
+
+            self.moveTo(isUndefined(offsetX) ? offsetX : canvas.left + Number(offsetX), isUndefined(offsetY) ? offsetY : canvas.top + Number(offsetY));
+        },
+
+
+        /**
+         * Move the canvas to an absolute point
+         *
+         * @param {Number} x
+         * @param {Number} y (optional)
+         */
+        moveTo: function moveTo(x, y) {
+            var self = this;
+            var canvas = self.canvas;
+            var changed = false;
+
+            // If "y" is not present, its default value is "x"
+            if (isUndefined(y)) {
+                y = x;
+            }
+
+            x = Number(x);
+            y = Number(y);
+
+            if (self.ready && !self.disabled && self.options.movable) {
+                if (isNumber(x)) {
+                    canvas.left = x;
+                    changed = true;
+                }
+
+                if (isNumber(y)) {
+                    canvas.top = y;
+                    changed = true;
+                }
+
+                if (changed) {
+                    self.renderCanvas(true);
+                }
+            }
+        },
+
+
+        /**
+         * Zoom the canvas with a relative ratio
+         *
+         * @param {Number} ratio
+         * @param {jQuery Event} _event (private)
+         */
+        zoom: function zoom(ratio, _event) {
+            var self = this;
+            var canvas = self.canvas;
+
+            ratio = Number(ratio);
+
+            if (ratio < 0) {
+                ratio = 1 / (1 - ratio);
+            } else {
+                ratio = 1 + ratio;
+            }
+
+            self.zoomTo(canvas.width * ratio / canvas.naturalWidth, _event);
+        },
+
+
+        /**
+         * Zoom the canvas to an absolute ratio
+         *
+         * @param {Number} ratio
+         * @param {jQuery Event} _event (private)
+         */
+        zoomTo: function zoomTo(ratio, _event) {
+            var self = this;
+            var options = self.options;
+            var pointers = self.pointers;
+            var canvas = self.canvas;
+            var width = canvas.width;
+            var height = canvas.height;
+            var naturalWidth = canvas.naturalWidth;
+            var naturalHeight = canvas.naturalHeight;
+
+            ratio = Number(ratio);
+
+            if (ratio >= 0 && self.ready && !self.disabled && options.zoomable) {
+                var newWidth = naturalWidth * ratio;
+                var newHeight = naturalHeight * ratio;
+                var originalEvent = void 0;
+
+                if (_event) {
+                    originalEvent = _event.originalEvent;
+                }
+
+                if (self.trigger('zoom', {
+                    originalEvent: originalEvent,
+                    oldRatio: width / naturalWidth,
+                    ratio: newWidth / naturalWidth
+                }).isDefaultPrevented()) {
+                    return;
+                }
+
+                if (originalEvent) {
+                    var offset = self.$cropper.offset();
+                    var center = pointers && objectKeys(pointers).length ? getPointersCenter(pointers) : {
+                        pageX: _event.pageX || originalEvent.pageX || 0,
+                        pageY: _event.pageY || originalEvent.pageY || 0
+                    };
+
+                    // Zoom from the triggering point of the event
+                    canvas.left -= (newWidth - width) * ((center.pageX - offset.left - canvas.left) / width);
+                    canvas.top -= (newHeight - height) * ((center.pageY - offset.top - canvas.top) / height);
+                } else {
+                    // Zoom from the center of the canvas
+                    canvas.left -= (newWidth - width) / 2;
+                    canvas.top -= (newHeight - height) / 2;
+                }
+
+                canvas.width = newWidth;
+                canvas.height = newHeight;
+                self.renderCanvas(true);
+            }
+        },
+
+
+        /**
+         * Rotate the canvas with a relative degree
+         *
+         * @param {Number} degree
+         */
+        rotate: function rotate(degree) {
+            var self = this;
+
+            self.rotateTo((self.image.rotate || 0) + Number(degree));
+        },
+
+
+        /**
+         * Rotate the canvas to an absolute degree
+         * https://developer.mozilla.org/en-US/docs/Web/CSS/transform-function#rotate()
+         *
+         * @param {Number} degree
+         */
+        rotateTo: function rotateTo(degree) {
+            var self = this;
+
+            degree = Number(degree);
+
+            if (isNumber(degree) && self.ready && !self.disabled && self.options.rotatable) {
+                self.image.rotate = degree % 360;
+                self.rotated = true;
+                self.renderCanvas(true);
+            }
+        },
+
+
+        /**
+         * Scale the image
+         * https://developer.mozilla.org/en-US/docs/Web/CSS/transform-function#scale()
+         *
+         * @param {Number} scaleX
+         * @param {Number} scaleY (optional)
+         */
+        scale: function scale(scaleX, scaleY) {
+            var self = this;
+            var image = self.image;
+            var changed = false;
+
+            // If "scaleY" is not present, its default value is "scaleX"
+            if (isUndefined(scaleY)) {
+                scaleY = scaleX;
+            }
+
+            scaleX = Number(scaleX);
+            scaleY = Number(scaleY);
+
+            if (self.ready && !self.disabled && self.options.scalable) {
+                if (isNumber(scaleX)) {
+                    image.scaleX = scaleX;
+                    changed = true;
+                }
+
+                if (isNumber(scaleY)) {
+                    image.scaleY = scaleY;
+                    changed = true;
+                }
+
+                if (changed) {
+                    self.renderImage(true);
+                }
+            }
+        },
+
+
+        /**
+         * Scale the abscissa of the image
+         *
+         * @param {Number} scaleX
+         */
+        scaleX: function scaleX(_scaleX) {
+            var self = this;
+            var scaleY = self.image.scaleY;
+
+            self.scale(_scaleX, isNumber(scaleY) ? scaleY : 1);
+        },
+
+
+        /**
+         * Scale the ordinate of the image
+         *
+         * @param {Number} scaleY
+         */
+        scaleY: function scaleY(_scaleY) {
+            var self = this;
+            var scaleX = self.image.scaleX;
+
+            self.scale(isNumber(scaleX) ? scaleX : 1, _scaleY);
+        },
+
+
+        /**
+         * Get the cropped area position and size data (base on the original image)
+         *
+         * @param {Boolean} isRounded (optional)
+         * @return {Object} data
+         */
+        getData: function getData(isRounded) {
+            var self = this;
+            var options = self.options;
+            var image = self.image;
+            var canvas = self.canvas;
+            var cropBox = self.cropBox;
+            var ratio = void 0;
+            var data = void 0;
+
+            if (self.ready && self.cropped) {
+                data = {
+                    x: cropBox.left - canvas.left,
+                    y: cropBox.top - canvas.top,
+                    width: cropBox.width,
+                    height: cropBox.height
+                };
+
+                ratio = image.width / image.naturalWidth;
+
+                $.each(data, function (i, n) {
+                    n /= ratio;
+                    data[i] = isRounded ? Math.round(n) : n;
+                });
+            } else {
+                data = {
+                    x: 0,
+                    y: 0,
+                    width: 0,
+                    height: 0
+                };
+            }
+
+            if (options.rotatable) {
+                data.rotate = image.rotate || 0;
+            }
+
+            if (options.scalable) {
+                data.scaleX = image.scaleX || 1;
+                data.scaleY = image.scaleY || 1;
+            }
+
+            return data;
+        },
+
+
+        /**
+         * Set the cropped area position and size with new data
+         *
+         * @param {Object} data
+         */
+        setData: function setData(data) {
+            var self = this;
+            var options = self.options;
+            var image = self.image;
+            var canvas = self.canvas;
+            var cropBoxData = {};
+            var rotated = void 0;
+            var isScaled = void 0;
+            var ratio = void 0;
+
+            if ($.isFunction(data)) {
+                data = data.call(self.element);
+            }
+
+            if (self.ready && !self.disabled && $.isPlainObject(data)) {
+                if (options.rotatable) {
+                    if (isNumber(data.rotate) && data.rotate !== image.rotate) {
+                        image.rotate = data.rotate;
+                        rotated = true;
+                        self.rotated = rotated;
+                    }
+                }
+
+                if (options.scalable) {
+                    if (isNumber(data.scaleX) && data.scaleX !== image.scaleX) {
+                        image.scaleX = data.scaleX;
+                        isScaled = true;
+                    }
+
+                    if (isNumber(data.scaleY) && data.scaleY !== image.scaleY) {
+                        image.scaleY = data.scaleY;
+                        isScaled = true;
+                    }
+                }
+
+                if (rotated) {
+                    self.renderCanvas();
+                } else if (isScaled) {
+                    self.renderImage();
+                }
+
+                ratio = image.width / image.naturalWidth;
+
+                if (isNumber(data.x)) {
+                    cropBoxData.left = data.x * ratio + canvas.left;
+                }
+
+                if (isNumber(data.y)) {
+                    cropBoxData.top = data.y * ratio + canvas.top;
+                }
+
+                if (isNumber(data.width)) {
+                    cropBoxData.width = data.width * ratio;
+                }
+
+                if (isNumber(data.height)) {
+                    cropBoxData.height = data.height * ratio;
+                }
+
+                self.setCropBoxData(cropBoxData);
+            }
+        },
+
+
+        /**
+         * Get the container size data
+         *
+         * @return {Object} data
+         */
+        getContainerData: function getContainerData() {
+            return this.ready ? this.container : {};
+        },
+
+
+        /**
+         * Get the image position and size data
+         *
+         * @return {Object} data
+         */
+        getImageData: function getImageData() {
+            return this.loaded ? this.image : {};
+        },
+
+
+        /**
+         * Get the canvas position and size data
+         *
+         * @return {Object} data
+         */
+        getCanvasData: function getCanvasData() {
+            var self = this;
+            var canvas = self.canvas;
+            var data = {};
+
+            if (self.ready) {
+                $.each(['left', 'top', 'width', 'height', 'naturalWidth', 'naturalHeight'], function (i, n) {
+                    data[n] = canvas[n];
+                });
+            }
+
+            return data;
+        },
+
+
+        /**
+         * Set the canvas position and size with new data
+         *
+         * @param {Object} data
+         */
+        setCanvasData: function setCanvasData(data) {
+            var self = this;
+            var canvas = self.canvas;
+            var aspectRatio = canvas.aspectRatio;
+
+            if ($.isFunction(data)) {
+                data = data.call(self.$element);
+            }
+
+            if (self.ready && !self.disabled && $.isPlainObject(data)) {
+                if (isNumber(data.left)) {
+                    canvas.left = data.left;
+                }
+
+                if (isNumber(data.top)) {
+                    canvas.top = data.top;
+                }
+
+                if (isNumber(data.width)) {
+                    canvas.width = data.width;
+                    canvas.height = data.width / aspectRatio;
+                } else if (isNumber(data.height)) {
+                    canvas.height = data.height;
+                    canvas.width = data.height * aspectRatio;
+                }
+
+                self.renderCanvas(true);
+            }
+        },
+
+
+        /**
+         * Get the crop box position and size data
+         *
+         * @return {Object} data
+         */
+        getCropBoxData: function getCropBoxData() {
+            var self = this;
+            var cropBox = self.cropBox;
+
+            return self.ready && self.cropped ? {
+                left: cropBox.left,
+                top: cropBox.top,
+                width: cropBox.width,
+                height: cropBox.height
+            } : {};
+        },
+
+
+        /**
+         * Set the crop box position and size with new data
+         *
+         * @param {Object} data
+         */
+        setCropBoxData: function setCropBoxData(data) {
+            var self = this;
+            var cropBox = self.cropBox;
+            var aspectRatio = self.options.aspectRatio;
+            var widthChanged = void 0;
+            var heightChanged = void 0;
+
+            if ($.isFunction(data)) {
+                data = data.call(self.$element);
+            }
+
+            if (self.ready && self.cropped && !self.disabled && $.isPlainObject(data)) {
+                if (isNumber(data.left)) {
+                    cropBox.left = data.left;
+                }
+
+                if (isNumber(data.top)) {
+                    cropBox.top = data.top;
+                }
+
+                if (isNumber(data.width) && data.width !== cropBox.width) {
+                    widthChanged = true;
+                    cropBox.width = data.width;
+                }
+
+                if (isNumber(data.height) && data.height !== cropBox.height) {
+                    heightChanged = true;
+                    cropBox.height = data.height;
+                }
+
+                if (aspectRatio) {
+                    if (widthChanged) {
+                        cropBox.height = cropBox.width / aspectRatio;
+                    } else if (heightChanged) {
+                        cropBox.width = cropBox.height * aspectRatio;
+                    }
+                }
+
+                self.renderCropBox();
+            }
+        },
+
+
+        /**
+         * Get a canvas drawn the cropped image
+         *
+         * @param {Object} options (optional)
+         * @return {HTMLCanvasElement} canvas
+         */
+        getCroppedCanvas: function getCroppedCanvas(options) {
+            var self = this;
+
+            if (!self.ready || !window.HTMLCanvasElement) {
+                return null;
+            }
+
+            if (!$.isPlainObject(options)) {
+                options = {};
+            }
+
+            if (!self.cropped) {
+                return getSourceCanvas(self.$clone[0], self.image, options);
+            }
+
+            var data = self.getData();
+            var originalWidth = data.width;
+            var originalHeight = data.height;
+            var aspectRatio = originalWidth / originalHeight;
+            var scaledWidth = void 0;
+            var scaledHeight = void 0;
+            var scaledRatio = void 0;
+
+            if ($.isPlainObject(options)) {
+                scaledWidth = options.width;
+                scaledHeight = options.height;
+
+                if (scaledWidth) {
+                    scaledHeight = scaledWidth / aspectRatio;
+                    scaledRatio = scaledWidth / originalWidth;
+                } else if (scaledHeight) {
+                    scaledWidth = scaledHeight * aspectRatio;
+                    scaledRatio = scaledHeight / originalHeight;
+                }
+            }
+
+            // The canvas element will use `Math.Math.floor` on a float number, so Math.floor first
+            var canvasWidth = Math.floor(scaledWidth || originalWidth);
+            var canvasHeight = Math.floor(scaledHeight || originalHeight);
+
+            var canvas = $('<canvas>')[0];
+            var context = canvas.getContext('2d');
+
+            canvas.width = canvasWidth;
+            canvas.height = canvasHeight;
+
+            if (options.fillColor) {
+                context.fillStyle = options.fillColor;
+                context.fillRect(0, 0, canvasWidth, canvasHeight);
+            }
+
+            // https://developer.mozilla.org/en-US/docs/Web/API/CanvasRenderingContext2D.drawImage
+            var parameters = function () {
+                var source = getSourceCanvas(self.$clone[0], self.image, options);
+                var sourceWidth = source.width;
+                var sourceHeight = source.height;
+                var canvasData = self.canvas;
+                var params = [source];
+
+                // Source canvas
+                var srcX = data.x + canvasData.naturalWidth * (Math.abs(data.scaleX || 1) - 1) / 2;
+                var srcY = data.y + canvasData.naturalHeight * (Math.abs(data.scaleY || 1) - 1) / 2;
+                var srcWidth = void 0;
+                var srcHeight = void 0;
+
+                // Destination canvas
+                var dstX = void 0;
+                var dstY = void 0;
+                var dstWidth = void 0;
+                var dstHeight = void 0;
+
+                if (srcX <= -originalWidth || srcX > sourceWidth) {
+                    srcX = 0;
+                    srcWidth = 0;
+                    dstX = 0;
+                    dstWidth = 0;
+                } else if (srcX <= 0) {
+                    dstX = -srcX;
+                    srcX = 0;
+                    dstWidth = Math.min(sourceWidth, originalWidth + srcX);
+                    srcWidth = dstWidth;
+                } else if (srcX <= sourceWidth) {
+                    dstX = 0;
+                    dstWidth = Math.min(originalWidth, sourceWidth - srcX);
+                    srcWidth = dstWidth;
+                }
+
+                if (srcWidth <= 0 || srcY <= -originalHeight || srcY > sourceHeight) {
+                    srcY = 0;
+                    srcHeight = 0;
+                    dstY = 0;
+                    dstHeight = 0;
+                } else if (srcY <= 0) {
+                    dstY = -srcY;
+                    srcY = 0;
+                    dstHeight = Math.min(sourceHeight, originalHeight + srcY);
+                    srcHeight = dstHeight;
+                } else if (srcY <= sourceHeight) {
+                    dstY = 0;
+                    dstHeight = Math.min(originalHeight, sourceHeight - srcY);
+                    srcHeight = dstHeight;
+                }
+
+                // All the numerical parameters should be integer for `drawImage` (#476)
+                params.push(Math.floor(srcX), Math.floor(srcY), Math.floor(srcWidth), Math.floor(srcHeight));
+
+                // Scale destination sizes
+                if (scaledRatio) {
+                    dstX *= scaledRatio;
+                    dstY *= scaledRatio;
+                    dstWidth *= scaledRatio;
+                    dstHeight *= scaledRatio;
+                }
+
+                // Avoid "IndexSizeError" in IE and Firefox
+                if (dstWidth > 0 && dstHeight > 0) {
+                    params.push(Math.floor(dstX), Math.floor(dstY), Math.floor(dstWidth), Math.floor(dstHeight));
+                }
+
+                return params;
+            }();
+
+            context.imageSmoothingEnabled = !!options.imageSmoothingEnabled;
+
+            if (options.imageSmoothingQuality) {
+                context.imageSmoothingQuality = options.imageSmoothingQuality;
+            }
+
+            context.drawImage.apply(context, _toConsumableArray(parameters));
+
+            return canvas;
+        },
+
+
+        /**
+         * Change the aspect ratio of the crop box
+         *
+         * @param {Number} aspectRatio
+         */
+        setAspectRatio: function setAspectRatio(aspectRatio) {
+            var self = this;
+            var options = self.options;
+
+            if (!self.disabled && !isUndefined(aspectRatio)) {
+                // 0 -> NaN
+                options.aspectRatio = Math.max(0, aspectRatio) || NaN;
+
+                if (self.ready) {
+                    self.initCropBox();
+
+                    if (self.cropped) {
+                        self.renderCropBox();
+                    }
+                }
+            }
+        },
+
+
+        /**
+         * Change the drag mode
+         *
+         * @param {String} mode (optional)
+         */
+        setDragMode: function setDragMode(mode) {
+            var self = this;
+            var options = self.options;
+            var croppable = void 0;
+            var movable = void 0;
+
+            if (self.loaded && !self.disabled) {
+                croppable = mode === 'crop';
+                movable = options.movable && mode === 'move';
+                mode = croppable || movable ? mode : 'none';
+
+                self.$dragBox.data('action', mode).toggleClass('cropper-crop', croppable).toggleClass('cropper-move', movable);
+
+                if (!options.cropBoxMovable) {
+                    // Sync drag mode to crop box when it is not movable(#300)
+                    self.$face.data('action', mode).toggleClass('cropper-crop', croppable).toggleClass('cropper-move', movable);
+                }
+            }
+        }
+    };
+
+    var _createClass = function () { function defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if ("value" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } } return function (Constructor, protoProps, staticProps) { if (protoProps) defineProperties(Constructor.prototype, protoProps); if (staticProps) defineProperties(Constructor, staticProps); return Constructor; }; }();
+
+    function _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError("Cannot call a class as a function"); } }
+
+    var CLASS_HIDDEN = 'cropper-hidden';
+    var REGEXP_DATA_URL = /^data:/;
+    var REGEXP_DATA_URL_JPEG = /^data:image\/jpeg;base64,/;
+
+    var Cropper = function () {
+        function Cropper(element, options) {
+            _classCallCheck(this, Cropper);
+
+            var self = this;
+
+            self.$element = $(element);
+            self.options = $.extend({}, DEFAULTS, $.isPlainObject(options) && options);
+            self.loaded = false;
+            self.ready = false;
+            self.completed = false;
+            self.rotated = false;
+            self.cropped = false;
+            self.disabled = false;
+            self.replaced = false;
+            self.limited = false;
+            self.wheeling = false;
+            self.isImg = false;
+            self.originalUrl = '';
+            self.canvas = null;
+            self.cropBox = null;
+            self.pointers = {};
+            self.init();
+        }
+
+        _createClass(Cropper, [{
+            key: 'init',
+            value: function init() {
+                var self = this;
+                var $this = self.$element;
+                var url = void 0;
+
+                if ($this.is('img')) {
+                    self.isImg = true;
+
+                    // Should use `$.fn.attr` here. e.g.: "img/picture.jpg"
+                    url = $this.attr('src');
+                    self.originalUrl = url;
+
+                    // Stop when it's a blank image
+                    if (!url) {
+                        return;
+                    }
+
+                    // Should use `$.fn.prop` here. e.g.: "http://example.com/img/picture.jpg"
+                    url = $this.prop('src');
+                } else if ($this.is('canvas') && window.HTMLCanvasElement) {
+                    url = $this[0].toDataURL();
+                }
+
+                self.load(url);
+            }
+
+            // A shortcut for triggering custom events
+
+        }, {
+            key: 'trigger',
+            value: function trigger(type, data) {
+                var e = $.Event(type, data);
+
+                this.$element.trigger(e);
+
+                return e;
+            }
+        }, {
+            key: 'load',
+            value: function load(url) {
+                var self = this;
+                var options = self.options;
+                var $this = self.$element;
+
+                if (!url) {
+                    return;
+                }
+
+                self.url = url;
+                self.image = {};
+
+                if (!options.checkOrientation || !window.ArrayBuffer) {
+                    self.clone();
+                    return;
+                }
+
+                // XMLHttpRequest disallows to open a Data URL in some browsers like IE11 and Safari
+                if (REGEXP_DATA_URL.test(url)) {
+                    if (REGEXP_DATA_URL_JPEG.test(url)) {
+                        self.read(dataURLToArrayBuffer(url));
+                    } else {
+                        self.clone();
+                    }
+                    return;
+                }
+
+                var xhr = new XMLHttpRequest();
+
+                xhr.onerror = $.proxy(function () {
+                    self.clone();
+                }, this);
+
+                xhr.onload = function load() {
+                    self.read(this.response);
+                };
+
+                if (options.checkCrossOrigin && isCrossOriginURL(url) && $this.prop('crossOrigin')) {
+                    url = addTimestamp(url);
+                }
+
+                xhr.open('get', url);
+                xhr.responseType = 'arraybuffer';
+                xhr.withCredentials = $this.prop('crossOrigin') === 'use-credentials';
+                xhr.send();
+            }
+        }, {
+            key: 'read',
+            value: function read(arrayBuffer) {
+                var self = this;
+                var options = self.options;
+                var orientation = getOrientation(arrayBuffer);
+                var image = self.image;
+                var rotate = 0;
+                var scaleX = 1;
+                var scaleY = 1;
+
+                if (orientation > 1) {
+                    self.url = arrayBufferToDataURL(arrayBuffer);
+
+                    switch (orientation) {
+                        // flip horizontal
+                        case 2:
+                            scaleX = -1;
+                            break;
+
+                            // rotate left 180°
+                        case 3:
+                            rotate = -180;
+                            break;
+
+                            // flip vertical
+                        case 4:
+                            scaleY = -1;
+                            break;
+
+                            // flip vertical + rotate right 90°
+                        case 5:
+                            rotate = 90;
+                            scaleY = -1;
+                            break;
+
+                            // rotate right 90°
+                        case 6:
+                            rotate = 90;
+                            break;
+
+                            // flip horizontal + rotate right 90°
+                        case 7:
+                            rotate = 90;
+                            scaleX = -1;
+                            break;
+
+                            // rotate left 90°
+                        case 8:
+                            rotate = -90;
+                            break;
+
+                        default:
+                    }
+                }
+
+                if (options.rotatable) {
+                    image.rotate = rotate;
+                }
+
+                if (options.scalable) {
+                    image.scaleX = scaleX;
+                    image.scaleY = scaleY;
+                }
+
+                self.clone();
+            }
+        }, {
+            key: 'clone',
+            value: function clone() {
+                var self = this;
+                var options = self.options;
+                var $this = self.$element;
+                var url = self.url;
+                var crossOrigin = '';
+                var crossOriginUrl = void 0;
+
+                if (options.checkCrossOrigin && isCrossOriginURL(url)) {
+                    crossOrigin = $this.prop('crossOrigin');
+
+                    if (crossOrigin) {
+                        crossOriginUrl = url;
+                    } else {
+                        crossOrigin = 'anonymous';
+
+                        // Bust cache (#148) when there is not a "crossOrigin" property
+                        crossOriginUrl = addTimestamp(url);
+                    }
+                }
+
+                self.crossOrigin = crossOrigin;
+                self.crossOriginUrl = crossOriginUrl;
+
+                var image = document.createElement('img');
+
+                if (crossOrigin) {
+                    image.crossOrigin = crossOrigin;
+                }
+
+                image.src = crossOriginUrl || url;
+
+                var $clone = $(image);
+
+                self.$clone = $clone;
+
+                if (self.isImg) {
+                    if ($this[0].complete) {
+                        self.start();
+                    } else {
+                        $this.one('load', $.proxy(self.start, this));
+                    }
+                } else {
+                    $clone.one('load', $.proxy(self.start, this)).one('error', $.proxy(self.stop, this)).addClass('cropper-hide').insertAfter($this);
+                }
+            }
+        }, {
+            key: 'start',
+            value: function start() {
+                var self = this;
+                var $clone = self.$clone;
+                var $image = self.$element;
+
+                if (!self.isImg) {
+                    $clone.off('error', self.stop);
+                    $image = $clone;
+                }
+
+                getImageSize($image[0], function (naturalWidth, naturalHeight) {
+                    $.extend(self.image, {
+                        naturalWidth: naturalWidth,
+                        naturalHeight: naturalHeight,
+                        aspectRatio: naturalWidth / naturalHeight
+                    });
+
+                    self.loaded = true;
+                    self.build();
+                });
+            }
+        }, {
+            key: 'stop',
+            value: function stop() {
+                var self = this;
+
+                self.$clone.remove();
+                self.$clone = null;
+            }
+        }, {
+            key: 'build',
+            value: function build() {
+                var self = this;
+                var options = self.options;
+                var $this = self.$element;
+                var $clone = self.$clone;
+
+                if (!self.loaded) {
+                    return;
+                }
+
+                // Unbuild first when replace
+                if (self.ready) {
+                    self.unbuild();
+                }
+
+                var $cropper = $(TEMPLATE);
+                var $cropBox = $cropper.find('.cropper-crop-box');
+                var $face = $cropBox.find('.cropper-face');
+
+                // Create cropper elements
+                self.$container = $this.parent();
+                self.$cropper = $cropper;
+                self.$canvas = $cropper.find('.cropper-canvas').append($clone);
+                self.$dragBox = $cropper.find('.cropper-drag-box');
+                self.$cropBox = $cropBox;
+                self.$viewBox = $cropper.find('.cropper-view-box');
+                self.$face = $face;
+
+                // Hide the original image
+                $this.addClass(CLASS_HIDDEN).after($cropper);
+
+                // Show the clone image if is hidden
+                if (!self.isImg) {
+                    $clone.removeClass('cropper-hide');
+                }
+
+                self.initPreview();
+                self.bind();
+
+                options.aspectRatio = Math.max(0, options.aspectRatio) || NaN;
+                options.viewMode = Math.max(0, Math.min(3, Math.round(options.viewMode))) || 0;
+
+                self.cropped = options.autoCrop;
+
+                if (options.autoCrop) {
+                    if (options.modal) {
+                        self.$dragBox.addClass('cropper-modal');
+                    }
+                } else {
+                    $cropBox.addClass(CLASS_HIDDEN);
+                }
+
+                if (!options.guides) {
+                    $cropBox.find('.cropper-dashed').addClass(CLASS_HIDDEN);
+                }
+
+                if (!options.center) {
+                    $cropBox.find('.cropper-center').addClass(CLASS_HIDDEN);
+                }
+
+                if (options.cropBoxMovable) {
+                    $face.addClass('cropper-move').data('action', 'all');
+                }
+
+                if (!options.highlight) {
+                    $face.addClass('cropper-invisible');
+                }
+
+                if (options.background) {
+                    $cropper.addClass('cropper-bg');
+                }
+
+                if (!options.cropBoxResizable) {
+                    $cropBox.find('.cropper-line, .cropper-point').addClass(CLASS_HIDDEN);
+                }
+
+                self.setDragMode(options.dragMode);
+                self.render();
+                self.ready = true;
+                self.setData(options.data);
+
+                // Trigger the ready event asynchronously to keep `data('cropper')` is defined
+                self.completing = setTimeout(function () {
+                    if ($.isFunction(options.ready)) {
+                        $this.one('ready', options.ready);
+                    }
+
+                    self.trigger('ready');
+                    self.trigger('crop', self.getData());
+                    self.completed = true;
+                }, 0);
+            }
+        }, {
+            key: 'unbuild',
+            value: function unbuild() {
+                var self = this;
+
+                if (!self.ready) {
+                    return;
+                }
+
+                if (!self.completed) {
+                    clearTimeout(self.completing);
+                }
+
+                self.ready = false;
+                self.completed = false;
+                self.initialImage = null;
+
+                // Clear `initialCanvas` is necessary when replace
+                self.initialCanvas = null;
+                self.initialCropBox = null;
+                self.container = null;
+                self.canvas = null;
+
+                // Clear `cropBox` is necessary when replace
+                self.cropBox = null;
+                self.unbind();
+
+                self.resetPreview();
+                self.$preview = null;
+
+                self.$viewBox = null;
+                self.$cropBox = null;
+                self.$dragBox = null;
+                self.$canvas = null;
+                self.$container = null;
+
+                self.$cropper.remove();
+                self.$cropper = null;
+            }
+        }], [{
+            key: 'setDefaults',
+            value: function setDefaults(options) {
+                $.extend(DEFAULTS, $.isPlainObject(options) && options);
+            }
+        }]);
+
+        return Cropper;
+    }();
+
+    $.extend(Cropper.prototype, render);
+    $.extend(Cropper.prototype, preview);
+    $.extend(Cropper.prototype, events);
+    $.extend(Cropper.prototype, handlers);
+    $.extend(Cropper.prototype, change);
+    $.extend(Cropper.prototype, methods);
+
+    var NAMESPACE = 'cropper';
+    var OtherCropper = $.fn.cropper;
+
+    $.fn.cropper = function jQueryCropper(option) {
+        for (var _len = arguments.length, args = Array(_len > 1 ? _len - 1 : 0), _key = 1; _key < _len; _key++) {
+            args[_key - 1] = arguments[_key];
+        }
+
+        var result = void 0;
+
+        this.each(function (i, element) {
+            var $this = $(element);
+            var data = $this.data(NAMESPACE);
+
+            if (!data) {
+                if (/destroy/.test(option)) {
+                    return;
+                }
+
+                var options = $.extend({}, $this.data(), $.isPlainObject(option) && option);
+                $this.data(NAMESPACE, data = new Cropper(element, options));
+            }
+
+            if (typeof option === 'string') {
+                var fn = data[option];
+
+                if ($.isFunction(fn)) {
+                    result = fn.apply(data, args);
+                }
+            }
+        });
+
+        return typeof result !== 'undefined' ? result : this;
+    };
+
+    $.fn.cropper.Constructor = Cropper;
+    $.fn.cropper.setDefaults = Cropper.setDefaults;
+
+    // No conflict
+    $.fn.cropper.noConflict = function noConflict() {
+        $.fn.cropper = OtherCropper;
+        return this;
+    };
+
+    exports('cropper', $.fn.cropper);
+});

+ 135 - 0
assets/lib/layui/admin/modules/plugin/cropper/croppers.js

@@ -0,0 +1,135 @@
+/*!
+ * Cropper v3.0.0
+ */
+
+layui.config({
+    base: '/static/cropper/' //layui自定义layui组件目录
+}).define(['jquery','layer','cropper'],function (exports) {
+    var $ = layui.jquery
+        ,layer = layui.layer;
+    var html = "<link rel=\"stylesheet\" href=\"/static/cropper/cropper.css\">\n" +
+        "<div class=\"layui-fluid showImgEdit\" style=\"display: none\">\n" +
+        "    <div class=\"layui-form-item\">\n" +
+        "        <div class=\"layui-input-inline layui-btn-container\" style=\"width: auto;\">\n" +
+        "            <label for=\"cropper_avatarImgUpload\" class=\"layui-btn layui-btn-primary\">\n" +
+        "                <i class=\"layui-icon\">&#xe67c;</i>选择图片\n" +
+        "            </label>\n" +
+        "            <input class=\"layui-upload-file\" id=\"cropper_avatarImgUpload\" type=\"file\" value=\"选择图片\" name=\"file\">\n" +
+        "        </div>\n" +
+        "        <div class=\"layui-form-mid layui-word-aux\">头像的尺寸限定150x150px,大小在50kb以内</div>\n" +
+        "    </div>\n" +
+        "    <div class=\"layui-row layui-col-space15\">\n" +
+        "        <div class=\"layui-col-xs9\">\n" +
+        "            <div class=\"readyimg\" style=\"height:450px;background-color: rgb(247, 247, 247);\">\n" +
+        "                <img src=\"\" >\n" +
+        "            </div>\n" +
+        "        </div>\n" +
+        "        <div class=\"layui-col-xs3\">\n" +
+        "            <div class=\"img-preview\" style=\"width:200px;height:200px;overflow:hidden\">\n" +
+        "            </div>\n" +
+        "        </div>\n" +
+        "    </div>\n" +
+        "    <div class=\"layui-row layui-col-space15\">\n" +
+        "        <div class=\"layui-col-xs9\">\n" +
+        "            <div class=\"layui-row\">\n" +
+        "                <div class=\"layui-col-xs6\">\n" +
+        "                    <button type=\"button\" class=\"layui-btn layui-icon layui-icon-left\" cropper-event=\"rotate\" data-option=\"-15\" title=\"Rotate -90 degrees\"> 向左旋转</button>\n" +
+        "                    <button type=\"button\" class=\"layui-btn layui-icon layui-icon-right\" cropper-event=\"rotate\" data-option=\"15\" title=\"Rotate 90 degrees\"> 向右旋转</button>\n" +
+        "                </div>\n" +
+        "                <div class=\"layui-col-xs5\" style=\"text-align: right;\">\n" +
+        "                    <button type=\"button\" class=\"layui-btn\" title=\"移动\"></button>\n" +
+        "                    <button type=\"button\" class=\"layui-btn\" title=\"放大图片\"></button>\n" +
+        "                    <button type=\"button\" class=\"layui-btn\" title=\"缩小图片\"></button>\n" +
+        "                    <button type=\"button\" class=\"layui-btn layui-icon layui-icon-refresh\" cropper-event=\"reset\" title=\"重置图片\"></button>\n" +
+        "                </div>\n" +
+        "            </div>\n" +
+        "        </div>\n" +
+        "        <div class=\"layui-col-xs3\">\n" +
+        "            <button class=\"layui-btn layui-btn-fluid\" cropper-event=\"confirmSave\" type=\"button\"> 保存修改</button>\n" +
+        "        </div>\n" +
+        "    </div>\n" +
+        "\n" +
+        "</div>";
+    var obj = {
+        render: function(e){
+            $('body').append(html);
+            var self = this,
+                elem = e.elem,
+                saveW = e.saveW,
+                saveH = e.saveH,
+                mark = e.mark,
+                area = e.area,
+                url = e.url,
+                done = e.done;
+
+            var content = $('.showImgEdit')
+                ,image = $(".showImgEdit .readyimg img")
+                ,preview = '.showImgEdit .img-preview'
+                ,file = $(".showImgEdit input[name='file']")
+                , options = {aspectRatio: mark,preview: preview,viewMode:1};
+
+            $(elem).on('click',function () {
+                layer.open({
+                    type: 1
+                    , content: content
+                    , area: area
+                    , success: function () {
+                        image.cropper(options);
+                    }
+                    , cancel: function (index) {
+                        layer.close(index);
+                        image.cropper('destroy');
+                    }
+                });
+            });
+            $(".layui-btn").on('click',function () {
+                var event = $(this).attr("cropper-event");
+                //监听确认保存图像
+                if(event === 'confirmSave'){
+                    image.cropper("getCroppedCanvas",{
+                        width: saveW,
+                        height: saveH
+                    }).toBlob(function(blob){
+                        var formData=new FormData();
+                        formData.append('file',blob,'head.jpg');
+                        $.ajax({
+                            method:"post",
+                            url: url, //用于文件上传的服务器端请求地址
+                            data: formData,
+                            processData: false,
+                            contentType: false,
+                            success:function(result){
+                                if(result.code == 0){
+                                    layer.msg(result.msg,{icon: 1});
+                                    layer.closeAll('page');
+                                    return done(result.data.src);
+                                }else if(result.code == -1){
+                                    layer.alert(result.msg,{icon: 2});
+                                }
+
+                            }
+                        });
+                    });
+                    //监听旋转
+                }else if(event === 'rotate'){
+                    var option = $(this).attr('data-option');
+                    image.cropper('rotate', option);
+                    //重设图片
+                }else if(event === 'reset'){
+                    image.cropper('reset');
+                }
+                //文件选择
+                file.change(function () {
+                    var r= new FileReader();
+                    var f=this.files[0];
+                    r.readAsDataURL(f);
+                    r.onload=function (e) {
+                        image.cropper('destroy').attr('src', this.result).cropper(options);
+                    };
+                });
+            });
+        }
+
+    };
+    exports('croppers', obj);
+});

File diff suppressed because it is too large
+ 158 - 0
assets/lib/layui/admin/modules/plugin/formselects/formselects.css


+ 1791 - 0
assets/lib/layui/admin/modules/plugin/formselects/formselects.js

@@ -0,0 +1,1791 @@
+'use strict';
+
+var _typeof = typeof Symbol === "function" && typeof Symbol.iterator === "symbol" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === "function" && obj.constructor === Symbol && obj !== Symbol.prototype ? "symbol" : typeof obj; };
+
+function _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
+
+/**
+ * name: formSelects
+ * 基于Layui Select多选
+ * version: 4.0.0.0910
+ * http://sun.faysunshine.com/layui/formSelects-v4/dist/formSelects-v4.js
+ */
+(function (layui, window, factory) {
+    if ((typeof exports === 'undefined' ? 'undefined' : _typeof(exports)) === 'object') {
+        // 支持 CommonJS
+        module.exports = factory();
+    } else if (typeof define === 'function' && define.amd) {
+        // 支持 AMD
+        define(factory);
+    } else if (window.layui && layui.define) {
+        //layui加载
+        layui.define(['jquery'], function (exports) {
+            exports('formSelects', factory());
+        });
+    } else {
+        window.formSelects = factory();
+    }
+})(typeof layui == 'undefined' ? null : layui, window, function () {
+    var v = '4.0.0.0910',
+        NAME = 'xm-select',
+        PNAME = 'xm-select-parent',
+        INPUT = 'xm-select-input',
+        TDIV = 'xm-select--suffix',
+        THIS = 'xm-select-this',
+        LABEL = 'xm-select-label',
+        SEARCH = 'xm-select-search',
+        SEARCH_TYPE = 'xm-select-search-type',
+        SHOW_COUNT = 'xm-select-show-count',
+        CREATE = 'xm-select-create',
+        CREATE_LONG = 'xm-select-create-long',
+        MAX = 'xm-select-max',
+        SKIN = 'xm-select-skin',
+        DIRECTION = "xm-select-direction",
+        HEIGHT = 'xm-select-height',
+        DISABLED = 'xm-dis-disabled',
+        DIS = 'xm-select-dis',
+        TEMP = 'xm-select-temp',
+        RADIO = 'xm-select-radio',
+        LINKAGE = 'xm-select-linkage',
+        DL = 'xm-select-dl',
+        DD_HIDE = 'xm-select-hide',
+        HIDE_INPUT = 'xm-hide-input',
+        SANJIAO = 'xm-select-sj',
+        ICON_CLOSE = 'xm-icon-close',
+        FORM_TITLE = 'xm-select-title',
+        FORM_SELECT = 'xm-form-select',
+        FORM_SELECTED = 'xm-form-selected',
+        FORM_NONE = 'xm-select-none',
+        FORM_EMPTY = 'xm-select-empty',
+        FORM_INPUT = 'xm-input',
+        FORM_DL_INPUT = 'xm-dl-input',
+        FORM_SELECT_TIPS = 'xm-select-tips',
+        CHECKBOX_YES = 'xm-iconfont',
+        FORM_TEAM_PID = 'XM_PID_VALUE',
+        CZ = 'xm-cz',
+        CZ_GROUP = 'xm-cz-group',
+        TIPS = '请选择',
+        data = {},
+        events = {
+        on: {},
+        endOn: {},
+        filter: {},
+        maxTips: {},
+        opened: {},
+        closed: {}
+    },
+        ajax = {
+        type: 'get',
+        header: {},
+        first: true,
+        data: {},
+        searchUrl: '',
+        searchName: 'keyword',
+        searchVal: null,
+        keyName: 'name',
+        keyVal: 'value',
+        keySel: 'selected',
+        keyDis: 'disabled',
+        keyChildren: 'children',
+        dataType: '',
+        delay: 500,
+        beforeSuccess: null,
+        success: null,
+        error: null,
+        beforeSearch: null,
+        response: {
+            statusCode: 0,
+            statusName: 'code',
+            msgName: 'msg',
+            dataName: 'data'
+        },
+        tree: {
+            nextClick: function nextClick(id, item, callback) {
+                callback([]);
+            },
+            folderChoose: true,
+            lazy: true
+        }
+    },
+        quickBtns = [{ icon: 'xm-iconfont icon-quanxuan', name: '全选', click: function click(id, cm) {
+            cm.selectAll(id, true, true);
+        } }, { icon: 'xm-iconfont icon-qingkong', name: '清空', click: function click(id, cm) {
+            cm.removeAll(id, true, true);
+        } }, { icon: 'xm-iconfont icon-fanxuan', name: '反选', click: function click(id, cm) {
+            cm.reverse(id, true, true);
+        } }, { icon: 'xm-iconfont icon-pifu', name: '换肤', click: function click(id, cm) {
+            cm.skin(id);
+        } }],
+        $ = window.$ || window.layui && window.layui.jquery,
+        $win = $(window),
+        ajaxs = {},
+        fsConfig = {},
+        fsConfigs = {},
+        FormSelects = function FormSelects(options) {
+        var _this = this;
+
+        this.config = {
+            name: null, //xm-select="xxx"
+            max: null,
+            maxTips: function maxTips(id, vals, val, max) {
+                var ipt = $('[xid="' + _this.config.name + '"]').prev().find('.' + NAME);
+                if (ipt.parents('.layui-form-item[pane]').length) {
+                    ipt = ipt.parents('.layui-form-item[pane]');
+                }
+                ipt.attr('style', 'border-color: red !important');
+                setTimeout(function () {
+                    ipt.removeAttr('style');
+                }, 300);
+            },
+            init: null, //初始化的选择值,
+            on: null, //select值发生变化
+            opened: null,
+            closed: null,
+            filter: function filter(id, inputVal, val, isDisabled) {
+                return val.name.indexOf(inputVal) == -1;
+            },
+            clearid: -1,
+            direction: 'auto',
+            height: null,
+            isEmpty: false,
+            btns: [quickBtns[0], quickBtns[1], quickBtns[2]],
+            searchType: 0,
+            create: function create(id, name) {
+                return Date.now();
+            },
+            template: function template(id, item) {
+                return item.name;
+            },
+            showCount: 0,
+            isCreate: false,
+            placeholder: TIPS,
+            clearInput: false
+        };
+        this.select = null;
+        this.values = [];
+        $.extend(this.config, options, {
+            searchUrl: options.isSearch ? options.searchUrl : null,
+            placeholder: options.optionsFirst ? options.optionsFirst.value ? TIPS : options.optionsFirst.innerHTML || TIPS : TIPS,
+            btns: options.radio ? [quickBtns[1]] : [quickBtns[0], quickBtns[1], quickBtns[2]]
+        }, fsConfigs[options.name] || fsConfig);
+        if (isNaN(this.config.showCount) || this.config.showCount <= 0) {
+            this.config.showCount = 19921012;
+        }
+    };
+
+    //一些简单的处理方法
+    var Common = function Common() {
+        this.appender();
+        this.on();
+        this.onreset();
+    };
+
+    Common.prototype.appender = function () {
+        //针对IE做的一些拓展
+        //拓展Array map方法
+        if (!Array.prototype.map) {
+            Array.prototype.map = function (i, h) {
+                var b,
+                    a,
+                    c,
+                    e = Object(this),
+                    f = e.length >>> 0;if (h) {
+                    b = h;
+                }a = new Array(f);c = 0;while (c < f) {
+                    var d, g;if (c in e) {
+                        d = e[c];g = i.call(b, d, c, e);a[c] = g;
+                    }c++;
+                }return a;
+            };
+        };
+
+        //拓展Array foreach方法
+        if (!Array.prototype.forEach) {
+            Array.prototype.forEach = function forEach(g, b) {
+                var d, c;if (this == null) {
+                    throw new TypeError("this is null or not defined");
+                }var f = Object(this);var a = f.length >>> 0;if (typeof g !== "function") {
+                    throw new TypeError(g + " is not a function");
+                }if (arguments.length > 1) {
+                    d = b;
+                }c = 0;while (c < a) {
+                    var e;if (c in f) {
+                        e = f[c];g.call(d, e, c, f);
+                    }c++;
+                }
+            };
+        };
+
+        //拓展Array filter方法
+        if (!Array.prototype.filter) {
+            Array.prototype.filter = function (b) {
+                if (this === void 0 || this === null) {
+                    throw new TypeError();
+                }var f = Object(this);var a = f.length >>> 0;if (typeof b !== "function") {
+                    throw new TypeError();
+                }var e = [];var d = arguments[1];for (var c = 0; c < a; c++) {
+                    if (c in f) {
+                        var g = f[c];if (b.call(d, g, c, f)) {
+                            e.push(g);
+                        }
+                    }
+                }return e;
+            };
+        };
+    };
+
+    Common.prototype.init = function (target) {
+        var _this2 = this;
+
+        //初始化页面上已有的select
+        $(target ? target : 'select[' + NAME + ']').each(function (index, select) {
+            var othis = $(select),
+                id = othis.attr(NAME),
+                hasLayuiRender = othis.next('.layui-form-select'),
+                hasRender = othis.next('.' + PNAME),
+                options = {
+                name: id,
+                disabled: select.disabled,
+                max: othis.attr(MAX) - 0,
+                isSearch: othis.attr(SEARCH) != undefined,
+                searchUrl: othis.attr(SEARCH),
+                isCreate: othis.attr(CREATE) != undefined,
+                radio: othis.attr(RADIO) != undefined,
+                skin: othis.attr(SKIN),
+                direction: othis.attr(DIRECTION),
+                optionsFirst: select.options[0],
+                height: othis.attr(HEIGHT),
+                formname: othis.attr('name') || othis.attr('_name'),
+                layverify: othis.attr('lay-verify') || othis.attr('_lay-verify'),
+                layverType: othis.attr('lay-verType'),
+                searchType: othis.attr(SEARCH_TYPE) == 'dl' ? 1 : 0,
+                showCount: othis.attr(SHOW_COUNT) - 0
+            },
+                value = othis.find('option[selected]').toArray().map(function (option) {
+                //获取已选中的数据
+                return {
+                    name: option.innerHTML,
+                    value: option.value
+                };
+            }),
+                fs = new FormSelects(options);
+
+            fs.values = value;
+
+            if (fs.config.init) {
+                fs.values = fs.config.init.map(function (item) {
+                    if ((typeof item === 'undefined' ? 'undefined' : _typeof(item)) == 'object') {
+                        return item;
+                    }
+                    return {
+                        name: othis.find('option[value="' + item + '"]').text(),
+                        value: item
+                    };
+                }).filter(function (item) {
+                    return item.name;
+                });
+                fs.config.init = fs.values.concat([]);
+            } else {
+                fs.config.init = value.concat([]);
+            }
+
+            !fs.values && (fs.values = []);
+
+            data[id] = fs;
+
+            //先取消layui对select的渲染
+            hasLayuiRender[0] && hasLayuiRender.remove();
+            hasRender[0] && hasRender.remove();
+
+            //构造渲染div
+            var dinfo = _this2.renderSelect(id, fs.config.placeholder, select);
+            var heightStyle = !fs.config.height || fs.config.height == 'auto' ? '' : 'xm-hg style="height: 34px;"';
+            var inputHtml = ['<div class="' + LABEL + '">', '<input type="text" fsw class="' + FORM_INPUT + ' ' + INPUT + '" ' + (fs.config.isSearch ? '' : 'style="display: none;"') + ' autocomplete="off" debounce="0" />', '</div>'];
+            var reElem = $('<div class="' + FORM_SELECT + '" ' + SKIN + '="' + fs.config.skin + '">\n\t\t\t\t\t<input class="' + HIDE_INPUT + '" value="" name="' + fs.config.formname + '" lay-verify="' + fs.config.layverify + '" lay-verType="' + fs.config.layverType + '" type="text" style="position: absolute;bottom: 0; z-index: -1;width: 100%; height: 100%; border: none; opacity: 0;"/>\n\t\t\t\t\t<div class="' + FORM_TITLE + ' ' + (fs.config.disabled ? DIS : '') + '">\n\t\t\t\t\t\t<div class="' + FORM_INPUT + ' ' + NAME + '" ' + heightStyle + '>\n\t\t\t\t\t\t\t' + inputHtml.join('') + '\n\t\t\t\t\t\t\t<i class="' + SANJIAO + '"></i>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div class="' + TDIV + '">\n\t\t\t\t\t\t\t<input type="text" autocomplete="off" placeholder="' + fs.config.placeholder + '" readonly="readonly" unselectable="on" class="' + FORM_INPUT + '">\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<div></div>\n\t\t\t\t\t</div>\n\t\t\t\t\t<dl xid="' + id + '" class="' + DL + ' ' + (fs.config.radio ? RADIO : '') + '">' + dinfo + '</dl>\n\t\t\t\t</div>');
+
+            var $parent = $('<div class="' + PNAME + '" FS_ID="' + id + '"></div>');
+            $parent.append(reElem);
+            othis.after($parent);
+            othis.attr('lay-ignore', '');
+            othis.removeAttr('name') && othis.attr('_name', fs.config.formname);
+            othis.removeAttr('lay-verify') && othis.attr('_lay-verify', fs.config.layverify);
+
+            //如果可搜索, 加上事件
+            if (fs.config.isSearch) {
+                ajaxs[id] = $.extend({}, ajax, { searchUrl: fs.config.searchUrl }, ajaxs[id]);
+                $(document).on('input', 'div.' + PNAME + '[FS_ID="' + id + '"] .' + INPUT, function (e) {
+                    _this2.search(id, e, fs.config.searchUrl);
+                });
+                if (fs.config.searchUrl) {
+                    //触发第一次请求事件
+                    _this2.triggerSearch(reElem, true);
+                }
+            } else {
+                //隐藏第二个dl
+                reElem.find('dl dd.' + FORM_DL_INPUT).css('display', 'none');
+            }
+        });
+    };
+
+    Common.prototype.search = function (id, e, searchUrl, call) {
+        var _this3 = this;
+
+        var input = void 0;
+        if (call) {
+            input = call;
+        } else {
+            input = e.target;
+            var keyCode = e.keyCode;
+            if (keyCode === 9 || keyCode === 13 || keyCode === 37 || keyCode === 38 || keyCode === 39 || keyCode === 40) {
+                return false;
+            }
+        }
+        var inputValue = $.trim(input.value);
+        //过滤一下tips
+        this.changePlaceHolder($(input));
+
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        searchUrl = ajaxConfig.searchUrl || searchUrl;
+        var fs = data[id],
+            isCreate = fs.config.isCreate,
+            reElem = $('dl[xid="' + id + '"]').parents('.' + FORM_SELECT);
+        //如果开启了远程搜索
+        if (searchUrl) {
+            if (ajaxConfig.searchVal) {
+                inputValue = ajaxConfig.searchVal;
+                ajaxConfig.searchVal = '';
+            }
+            if (!ajaxConfig.beforeSearch || ajaxConfig.beforeSearch && ajaxConfig.beforeSearch instanceof Function && ajaxConfig.beforeSearch(id, searchUrl, inputValue)) {
+                var delay = ajaxConfig.delay;
+                if (ajaxConfig.first) {
+                    ajaxConfig.first = false;
+                    delay = 10;
+                }
+                clearTimeout(fs.clearid);
+                fs.clearid = setTimeout(function () {
+                    reElem.find('dl > *:not(.' + FORM_SELECT_TIPS + ')').remove();
+                    reElem.find('dd.' + FORM_NONE).addClass(FORM_EMPTY).text('请求中');
+                    _this3.ajax(id, searchUrl, inputValue, false, null, true);
+                }, delay);
+            }
+        } else {
+            reElem.find('dl .' + DD_HIDE).removeClass(DD_HIDE);
+            //遍历选项, 选择可以显示的值
+            reElem.find('dl dd:not(.' + FORM_SELECT_TIPS + ')').each(function (idx, item) {
+                var _item = $(item);
+                var searchFun = events.filter[id] || data[id].config.filter;
+                if (searchFun && searchFun(id, inputValue, _this3.getItem(id, _item), _item.hasClass(DISABLED)) == true) {
+                    _item.addClass(DD_HIDE);
+                }
+            });
+            //控制分组名称
+            reElem.find('dl dt').each(function (index, item) {
+                if (!$(item).nextUntil('dt', ':not(.' + DD_HIDE + ')').length) {
+                    $(item).addClass(DD_HIDE);
+                }
+            });
+            //动态创建
+            this.create(id, isCreate, inputValue);
+            var shows = reElem.find('dl dd:not(.' + FORM_SELECT_TIPS + '):not(.' + DD_HIDE + ')');
+            if (!shows.length) {
+                reElem.find('dd.' + FORM_NONE).addClass(FORM_EMPTY).text('无匹配项');
+            } else {
+                reElem.find('dd.' + FORM_NONE).removeClass(FORM_EMPTY);
+            }
+        }
+    };
+
+    Common.prototype.isArray = function (obj) {
+        return Object.prototype.toString.call(obj) == "[object Array]";
+    };
+
+    Common.prototype.triggerSearch = function (div, isCall) {
+        var _this4 = this;
+
+        (div ? [div] : $('.' + FORM_SELECT).toArray()).forEach(function (reElem, index) {
+            reElem = $(reElem);
+            var id = reElem.find('dl').attr('xid');
+            if (id && data[id] && data[id].config.isEmpty || isCall) {
+                _this4.search(id, null, null, data[id].config.searchType == 0 ? reElem.find('.' + LABEL + ' .' + INPUT) : reElem.find('dl .' + FORM_DL_INPUT + ' .' + INPUT));
+            }
+        });
+    };
+
+    Common.prototype.clearInput = function (id) {
+        var div = $('.' + PNAME + '[fs_id="' + id + '"]');
+        var input = data[id].config.searchType == 0 ? div.find('.' + LABEL + ' .' + INPUT) : div.find('dl .' + FORM_DL_INPUT + ' .' + INPUT);
+        input.val('');
+    };
+
+    Common.prototype.ajax = function (id, searchUrl, inputValue, isLinkage, linkageWidth, isSearch, successCallback, isReplace) {
+        var _this5 = this;
+
+        var reElem = $('.' + PNAME + ' dl[xid="' + id + '"]').parents('.' + FORM_SELECT);
+        if (!reElem[0] || !searchUrl) {
+            return;
+        }
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        var ajaxData = $.extend(true, {}, ajaxConfig.data);
+        ajaxData[ajaxConfig.searchName] = inputValue;
+        //是否需要对ajax添加随机时间
+        //ajaxData['_'] = Date.now();
+        $.ajax({
+            type: ajaxConfig.type,
+            headers: ajaxConfig.header,
+            url: searchUrl,
+            data: ajaxConfig.dataType == 'json' ? JSON.stringify(ajaxData) : ajaxData,
+            success: function success(res) {
+                if (typeof res == 'string') {
+                    res = JSON.parse(res);
+                }
+                ajaxConfig.beforeSuccess && ajaxConfig.beforeSuccess instanceof Function && (res = ajaxConfig.beforeSuccess(id, searchUrl, inputValue, res));
+                if (_this5.isArray(res)) {
+                    var newRes = {};
+                    newRes[ajaxConfig.response.statusName] = ajaxConfig.response.statusCode;
+                    newRes[ajaxConfig.response.msgName] = "";
+                    newRes[ajaxConfig.response.dataName] = res;
+                    res = newRes;
+                }
+                if (res[ajaxConfig.response.statusName] != ajaxConfig.response.statusCode) {
+                    reElem.find('dd.' + FORM_NONE).addClass(FORM_EMPTY).text(res[ajaxConfig.response.msgName]);
+                } else {
+                    reElem.find('dd.' + FORM_NONE).removeClass(FORM_EMPTY);
+                    _this5.renderData(id, res[ajaxConfig.response.dataName], isLinkage, linkageWidth, isSearch, isReplace);
+                    data[id].config.isEmpty = res[ajaxConfig.response.dataName].length == 0;
+                }
+                successCallback && successCallback(id);
+                ajaxConfig.success && ajaxConfig.success instanceof Function && ajaxConfig.success(id, searchUrl, inputValue, res);
+            },
+            error: function error(err) {
+                reElem.find('dd[lay-value]:not(.' + FORM_SELECT_TIPS + ')').remove();
+                reElem.find('dd.' + FORM_NONE).addClass(FORM_EMPTY).text('服务异常');
+                ajaxConfig.error && ajaxConfig.error instanceof Function && ajaxConfig.error(id, searchUrl, inputValue, err);
+            }
+        });
+    };
+
+    Common.prototype.renderData = function (id, dataArr, linkage, linkageWidth, isSearch, isReplace) {
+        var _this6 = this;
+
+        if (linkage) {
+            //渲染多级联动
+            this.renderLinkage(id, dataArr, linkageWidth);
+            return;
+        }
+        if (isReplace) {
+            this.renderReplace(id, dataArr);
+            return;
+        }
+
+        var reElem = $('.' + PNAME + ' dl[xid="' + id + '"]').parents('.' + FORM_SELECT);
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        var pcInput = reElem.find('.' + TDIV + ' input');
+
+        dataArr = this.exchangeData(id, dataArr);
+        var values = [];
+        reElem.find('dl').html(this.renderSelect(id, pcInput.attr('placeholder') || pcInput.attr('back'), dataArr.map(function (item) {
+            var itemVal = $.extend({}, item, {
+                innerHTML: item[ajaxConfig.keyName],
+                value: item[ajaxConfig.keyVal],
+                sel: item[ajaxConfig.keySel],
+                disabled: item[ajaxConfig.keyDis],
+                type: item.type,
+                name: item[ajaxConfig.keyName]
+            });
+            if (itemVal.sel) {
+                values.push(itemVal);
+            }
+            return itemVal;
+        })));
+
+        var label = reElem.find('.' + LABEL);
+        var dl = reElem.find('dl[xid]');
+        if (isSearch) {
+            //如果是远程搜索, 这里需要判重
+            var oldVal = data[id].values;
+            oldVal.forEach(function (item, index) {
+                dl.find('dd[lay-value="' + item.value + '"]').addClass(THIS);
+            });
+            values.forEach(function (item, index) {
+                if (_this6.indexOf(oldVal, item) == -1) {
+                    _this6.addLabel(id, label, item);
+                    dl.find('dd[lay-value="' + item.value + '"]').addClass(THIS);
+                    oldVal.push(item);
+                }
+            });
+        } else {
+            values.forEach(function (item, index) {
+                _this6.addLabel(id, label, item);
+                dl.find('dd[lay-value="' + item.value + '"]').addClass(THIS);
+            });
+            data[id].values = values;
+        }
+        this.commonHandler(id, label);
+    };
+
+    Common.prototype.renderLinkage = function (id, dataArr, linkageWidth) {
+        var result = [],
+            index = 0,
+            temp = { "0": dataArr },
+            ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        db[id] = {};
+
+        var _loop = function _loop() {
+            var group = result[index++] = [],
+                _temp = temp;
+            temp = {};
+            $.each(_temp, function (pid, arr) {
+                $.each(arr, function (idx, item) {
+                    var val = {
+                        pid: pid,
+                        name: item[ajaxConfig.keyName],
+                        value: item[ajaxConfig.keyVal]
+                    };
+                    db[id][val.value] = $.extend(item, val);
+                    group.push(val);
+                    var children = item[ajaxConfig.keyChildren];
+                    if (children && children.length) {
+                        temp[val.value] = children;
+                    }
+                });
+            });
+        };
+
+        do {
+            _loop();
+        } while (Object.getOwnPropertyNames(temp).length);
+
+        var reElem = $('.' + PNAME + ' dl[xid="' + id + '"]').parents('.' + FORM_SELECT);
+        var html = ['<div class="xm-select-linkage">'];
+
+        $.each(result, function (idx, arr) {
+            var groupDiv = ['<div style="left: ' + (linkageWidth - 0) * idx + 'px;" class="xm-select-linkage-group xm-select-linkage-group' + (idx + 1) + ' ' + (idx != 0 ? 'xm-select-linkage-hide' : '') + '">'];
+            $.each(arr, function (idx2, item) {
+                var span = '<li title="' + item.name + '" pid="' + item.pid + '" xm-value="' + item.value + '"><span>' + item.name + '</span></li>';
+                groupDiv.push(span);
+            });
+            groupDiv.push('</div>');
+            html = html.concat(groupDiv);
+        });
+        html.push('<div style="clear: both; height: 288px;"></div>');
+        html.push('</div>');
+        reElem.find('dl').html(html.join(''));
+        reElem.find('.' + INPUT).css('display', 'none'); //联动暂时不支持搜索
+    };
+
+    Common.prototype.renderReplace = function (id, dataArr) {
+        var _this7 = this;
+
+        var dl = $('.' + PNAME + ' dl[xid="' + id + '"]');
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+
+        dataArr = this.exchangeData(id, dataArr);
+        db[id] = dataArr;
+
+        var html = dataArr.map(function (item) {
+            var itemVal = $.extend({}, item, {
+                innerHTML: item[ajaxConfig.keyName],
+                value: item[ajaxConfig.keyVal],
+                sel: item[ajaxConfig.keySel],
+                disabled: item[ajaxConfig.keyDis],
+                type: item.type,
+                name: item[ajaxConfig.keyName]
+            });
+            return _this7.createDD(id, itemVal);
+        }).join('');
+
+        dl.find('dd:not(.' + FORM_SELECT_TIPS + '),dt:not([style])').remove();
+        dl.find('dt[style]').after($(html));
+    };
+
+    Common.prototype.exchangeData = function (id, arr) {
+        //这里处理树形结构
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        var childrenName = ajaxConfig['keyChildren'];
+        var disabledName = ajaxConfig['keyDis'];
+        db[id] = {};
+        var result = this.getChildrenList(arr, childrenName, disabledName, [], false);
+        return result;
+    };
+
+    Common.prototype.getChildrenList = function (arr, childrenName, disabledName, pid, disabled) {
+        var result = [],
+            offset = 0;
+        for (var a = 0; a < arr.length; a++) {
+            var item = arr[a];
+            if (item.type && item.type == 'optgroup') {
+                result.push(item);
+                continue;
+            } else {
+                offset++;
+            }
+            var parentIds = pid.concat([]);
+            parentIds.push(offset - 1 + '_E');
+            item[FORM_TEAM_PID] = JSON.stringify(parentIds);
+            item[disabledName] = item[disabledName] || disabled;
+            result.push(item);
+            var child = item[childrenName];
+            if (child && common.isArray(child) && child.length) {
+                item['XM_TREE_FOLDER'] = true;
+                var pidArr = parentIds.concat([]);
+                var childResult = this.getChildrenList(child, childrenName, disabledName, pidArr, item[disabledName]);
+                result = result.concat(childResult);
+            }
+        }
+        return result;
+    };
+
+    Common.prototype.create = function (id, isCreate, inputValue) {
+        if (isCreate && inputValue) {
+            var fs = data[id],
+                dl = $('[xid="' + id + '"]'),
+                tips = dl.find('dd.' + FORM_SELECT_TIPS + '.' + FORM_DL_INPUT),
+                tdd = null,
+                temp = dl.find('dd.' + TEMP);
+            dl.find('dd:not(.' + FORM_SELECT_TIPS + '):not(.' + TEMP + ')').each(function (index, item) {
+                if (inputValue == $(item).find('span').attr('name')) {
+                    tdd = item;
+                }
+            });
+            if (!tdd) {
+                //如果不存在, 则创建
+                var val = fs.config.create(id, inputValue);
+                if (temp[0]) {
+                    temp.attr('lay-value', val);
+                    temp.find('span').text(inputValue);
+                    temp.find('span').attr("name", inputValue);
+                    temp.removeClass(DD_HIDE);
+                } else {
+                    tips.after($(this.createDD(id, {
+                        name: inputValue,
+                        innerHTML: inputValue,
+                        value: val
+                    }, TEMP + ' ' + CREATE_LONG)));
+                }
+            }
+        } else {
+            $('[xid=' + id + '] dd.' + TEMP).remove();
+        }
+    };
+
+    Common.prototype.createDD = function (id, item, clz) {
+        var ajaxConfig = ajaxs[id] ? ajaxs[id] : ajax;
+        var name = $.trim(item.innerHTML);
+        db[id][item.value] = $(item).is('option') ? item = function () {
+            var resultItem = {};
+            resultItem[ajaxConfig.keyName] = name;
+            resultItem[ajaxConfig.keyVal] = item.value;
+            resultItem[ajaxConfig.keyDis] = item.disabled;
+            return resultItem;
+        }() : item;
+        var template = data[id].config.template(id, item);
+        var pid = item[FORM_TEAM_PID];
+        pid ? pid = JSON.parse(pid) : pid = [-1];
+        var attr = pid[0] == -1 ? '' : 'tree-id="' + pid.join('-') + '" tree-folder="' + !!item['XM_TREE_FOLDER'] + '"';
+        return '<dd lay-value="' + item.value + '" class="' + (item.disabled ? DISABLED : '') + ' ' + (clz ? clz : '') + '" ' + attr + '>\n\t\t\t\t\t<div class="xm-unselect xm-form-checkbox ' + (item.disabled ? DISABLED : '') + '"  style="margin-left: ' + (pid.length - 1) * 30 + 'px">\n\t\t\t\t\t\t<i class="' + CHECKBOX_YES + '"></i>\n\t\t\t\t\t\t<span name="' + name + '">' + template + '</span>\n\t\t\t\t\t</div>\n\t\t\t\t</dd>';
+    };
+
+    Common.prototype.createQuickBtn = function (obj, right) {
+        return '<div class="' + CZ + '" method="' + obj.name + '" title="' + obj.name + '" ' + (right ? 'style="margin-right: ' + right + '"' : '') + '><i class="' + obj.icon + '"></i><span>' + obj.name + '</span></div>';
+    };
+
+    Common.prototype.renderBtns = function (id, show, right) {
+        var _this8 = this;
+
+        var quickBtn = [];
+        var dl = $('dl[xid="' + id + '"]');
+        quickBtn.push('<div class="' + CZ_GROUP + '" show="' + show + '" style="max-width: ' + (dl.prev().width() - 54) + 'px;">');
+        $.each(data[id].config.btns, function (index, item) {
+            quickBtn.push(_this8.createQuickBtn(item, right));
+        });
+        quickBtn.push('</div>');
+        quickBtn.push(this.createQuickBtn({ icon: 'xm-iconfont icon-caidan', name: '' }));
+        return quickBtn.join('');
+    };
+
+    Common.prototype.renderSelect = function (id, tips, select) {
+        var _this9 = this;
+
+        db[id] = {};
+        var arr = [];
+        if (data[id].config.btns.length) {
+            setTimeout(function () {
+                var dl = $('dl[xid="' + id + '"]');
+                dl.parents('.' + FORM_SELECT).attr(SEARCH_TYPE, data[id].config.searchType);
+                dl.find('.' + CZ_GROUP).css('max-width', dl.prev().width() - 54 + 'px');
+            }, 10);
+            arr.push(['<dd lay-value="" class="' + FORM_SELECT_TIPS + '" style="background-color: #FFF!important;">', this.renderBtns(id, null, '30px'), '</dd>', '<dd lay-value="" class="' + FORM_SELECT_TIPS + ' ' + FORM_DL_INPUT + '" style="background-color: #FFF!important;">', '<i class="xm-iconfont icon-sousuo"></i>', '<input type="text" class="' + FORM_INPUT + ' ' + INPUT + '" placeholder="\u8BF7\u641C\u7D22"/>', '</dd>'].join(''));
+        } else {
+            arr.push('<dd lay-value="" class="' + FORM_SELECT_TIPS + '">' + tips + '</dd>');
+        }
+        if (this.isArray(select)) {
+            $(select).each(function (index, item) {
+                if (item) {
+                    if (item.type && item.type === 'optgroup') {
+                        arr.push('<dt>' + item.name + '</dt>');
+                    } else {
+                        arr.push(_this9.createDD(id, item));
+                    }
+                }
+            });
+        } else {
+            $(select).find('*').each(function (index, item) {
+                if (item.tagName.toLowerCase() == 'option' && index == 0 && !item.value) {
+                    return;
+                }
+                if (item.tagName.toLowerCase() === 'optgroup') {
+                    arr.push('<dt>' + item.label + '</dt>');
+                } else {
+                    arr.push(_this9.createDD(id, item));
+                }
+            });
+        }
+        arr.push('<dt style="display:none;"> </dt>');
+        arr.push('<dd class="' + FORM_SELECT_TIPS + ' ' + FORM_NONE + ' ' + (arr.length === 2 ? FORM_EMPTY : '') + '">\u6CA1\u6709\u9009\u9879</dd>');
+        return arr.join('');
+    };
+
+    Common.prototype.on = function () {
+        var _this10 = this;
+
+        //事件绑定
+        this.one();
+
+        $(document).on('click', function (e) {
+            if (!$(e.target).parents('.' + FORM_TITLE)[0]) {
+                //清空input中的值
+                $('.' + PNAME + ' dl .' + DD_HIDE).removeClass(DD_HIDE);
+                $('.' + PNAME + ' dl dd.' + FORM_EMPTY).removeClass(FORM_EMPTY);
+                $('.' + PNAME + ' dl dd.' + TEMP).remove();
+                $.each(data, function (key, fs) {
+                    _this10.clearInput(key);
+                    if (!fs.values.length) {
+                        _this10.changePlaceHolder($('div[FS_ID="' + key + '"] .' + LABEL));
+                    }
+                });
+            }
+            $('.' + PNAME + ' .' + FORM_SELECTED).each(function (index, item) {
+                _this10.changeShow($(item).find('.' + FORM_TITLE), false);
+            });
+        });
+    };
+
+    Common.prototype.calcLabelLeft = function (label, w, call) {
+        var pos = this.getPosition(label[0]);
+        pos.y = pos.x + label[0].clientWidth;
+        var left = label[0].offsetLeft;
+        if (!label.find('span').length) {
+            left = 0;
+        } else if (call) {
+            //校正归位
+            var span = label.find('span:last');
+            span.css('display') == 'none' ? span = span.prev()[0] : span = span[0];
+            var spos = this.getPosition(span);
+            spos.y = spos.x + span.clientWidth;
+
+            if (spos.y > pos.y) {
+                left = left - (spos.y - pos.y) - 5;
+            } else {
+                left = 0;
+            }
+        } else {
+            if (w < 0) {
+                var _span = label.find(':last');
+                _span.css('display') == 'none' ? _span = _span.prev()[0] : _span = _span[0];
+                var _spos = this.getPosition(_span);
+                _spos.y = _spos.x + _span.clientWidth;
+                if (_spos.y > pos.y) {
+                    left -= 10;
+                }
+            } else {
+                if (left < 0) {
+                    left += 10;
+                }
+                if (left > 0) {
+                    left = 0;
+                }
+            }
+        }
+        label.css('left', left + 'px');
+    };
+
+    Common.prototype.one = function (target) {
+        var _this11 = this;
+
+        //一次性事件绑定
+        $(target ? target : document).off('click', '.' + FORM_TITLE).on('click', '.' + FORM_TITLE, function (e) {
+            var othis = $(e.target),
+                title = othis.is(FORM_TITLE) ? othis : othis.parents('.' + FORM_TITLE),
+                dl = title.next(),
+                id = dl.attr('xid');
+
+            //清空非本select的input val
+            $('dl[xid]').not(dl).each(function (index, item) {
+                _this11.clearInput($(item).attr('xid'));
+            });
+            $('dl[xid]').not(dl).find('dd.' + DD_HIDE).removeClass(DD_HIDE);
+
+            //如果是disabled select
+            if (title.hasClass(DIS)) {
+                return false;
+            }
+            //如果点击的是右边的三角或者只读的input
+            if (othis.is('.' + SANJIAO) || othis.is('.' + INPUT + '[readonly]')) {
+                _this11.changeShow(title, !title.parents('.' + FORM_SELECT).hasClass(FORM_SELECTED));
+                return false;
+            }
+            //如果点击的是input的右边, focus一下
+            if (title.find('.' + INPUT + ':not(readonly)')[0]) {
+                var input = title.find('.' + INPUT),
+                    epos = { x: e.pageX, y: e.pageY },
+                    pos = _this11.getPosition(title[0]),
+                    width = title.width();
+                while (epos.x > pos.x) {
+                    if ($(document.elementFromPoint(epos.x, epos.y)).is(input)) {
+                        input.focus();
+                        _this11.changeShow(title, true);
+                        return false;
+                    }
+                    epos.x -= 50;
+                }
+            }
+
+            //如果点击的是可搜索的input
+            if (othis.is('.' + INPUT)) {
+                _this11.changeShow(title, true);
+                return false;
+            }
+            //如果点击的是x按钮
+            if (othis.is('i[fsw="' + NAME + '"]')) {
+                var val = _this11.getItem(id, othis),
+                    dd = dl.find('dd[lay-value=\'' + val.value + '\']');
+                if (dd.hasClass(DISABLED)) {
+                    //如果是disabled状态, 不可选, 不可删
+                    return false;
+                }
+                _this11.handlerLabel(id, dd, false, val);
+                return false;
+            }
+
+            _this11.changeShow(title, !title.parents('.' + FORM_SELECT).hasClass(FORM_SELECTED));
+            return false;
+        });
+        $(target ? target : document).off('click', 'dl.' + DL).on('click', 'dl.' + DL, function (e) {
+            var othis = $(e.target);
+            if (othis.is('.' + LINKAGE) || othis.parents('.' + LINKAGE)[0]) {
+                //linkage的处理
+                othis = othis.is('li') ? othis : othis.parents('li[xm-value]');
+                var _group = othis.parents('.xm-select-linkage-group'),
+                    _id = othis.parents('dl').attr('xid');
+                if (!_id) {
+                    return false;
+                }
+                //激活li
+                _group.find('.xm-select-active').removeClass('xm-select-active');
+                othis.addClass('xm-select-active');
+                //激活下一个group, 激活前显示对应数据
+                _group.nextAll('.xm-select-linkage-group').addClass('xm-select-linkage-hide');
+                var nextGroup = _group.next('.xm-select-linkage-group');
+                nextGroup.find('li').addClass('xm-select-linkage-hide');
+                nextGroup.find('li[pid="' + othis.attr('xm-value') + '"]').removeClass('xm-select-linkage-hide');
+                //如果没有下一个group, 或没有对应的值
+                if (!nextGroup[0] || nextGroup.find('li:not(.xm-select-linkage-hide)').length == 0) {
+                    var vals = [],
+                        index = 0,
+                        isAdd = !othis.hasClass('xm-select-this');
+                    if (data[_id].config.radio) {
+                        othis.parents('.xm-select-linkage').find('.xm-select-this').removeClass('xm-select-this');
+                    }
+                    do {
+                        vals[index++] = {
+                            name: othis.find('span').text(),
+                            value: othis.attr('xm-value')
+                        };
+                        othis = othis.parents('.xm-select-linkage-group').prev().find('li[xm-value="' + othis.attr('pid') + '"]');
+                    } while (othis.length);
+                    vals.reverse();
+                    var val = {
+                        name: vals.map(function (item) {
+                            return item.name;
+                        }).join('/'),
+                        value: vals.map(function (item) {
+                            return item.value;
+                        }).join('/')
+                    };
+                    _this11.handlerLabel(_id, null, isAdd, val);
+                } else {
+                    nextGroup.removeClass('xm-select-linkage-hide');
+                }
+                return false;
+            }
+
+            if (othis.is('dl')) {
+                return false;
+            }
+
+            if (othis.is('dt')) {
+                othis.nextUntil('dt').each(function (index, item) {
+                    item = $(item);
+                    if (item.hasClass(DISABLED) || item.hasClass(THIS)) {} else {
+                        item.find('i:not(.icon-expand)').click();
+                    }
+                });
+                return false;
+            }
+            var dd = othis.is('dd') ? othis : othis.parents('dd');
+            var id = dd.parent('dl').attr('xid');
+
+            if (dd.hasClass(DISABLED)) {
+                //被禁用选项的处理
+                return false;
+            }
+
+            //菜单功效
+            if (othis.is('i.icon-caidan')) {
+                var opens = [],
+                    closes = [];
+                othis.parents('dl').find('dd[tree-folder="true"]').each(function (index, item) {
+                    $(item).attr('xm-tree-hidn') == undefined ? opens.push(item) : closes.push(item);
+                });
+                var arr = closes.length ? closes : opens;
+                arr.forEach(function (item) {
+                    return item.click();
+                });
+                return false;
+            }
+            //树状结构的选择
+            var treeId = dd.attr('tree-id');
+            if (treeId) {
+                //忽略右边的图标
+                if (othis.is('i:not(.icon-expand)')) {
+                    _this11.handlerLabel(id, dd, !dd.hasClass(THIS));
+                    return false;
+                }
+                var ajaxConfig = ajaxs[id] || ajax;
+                var treeConfig = ajaxConfig.tree;
+                var childrens = dd.nextAll('dd[tree-id^="' + treeId + '"]');
+                if (childrens && childrens.length) {
+                    var len = childrens[0].clientHeight;
+                    len ? (_this11.addTreeHeight(dd, len), len = 0) : (len = dd.attr('xm-tree-hidn') || 36, dd.removeAttr('xm-tree-hidn'), dd.find('>i').remove(), childrens = childrens.filter(function (index, item) {
+                        return $(item).attr('tree-id').split('-').length - 1 == treeId.split('-').length;
+                    }));
+                    childrens.animate({
+                        height: len
+                    }, 150);
+                    return false;
+                } else {
+                    if (treeConfig.nextClick && treeConfig.nextClick instanceof Function) {
+                        treeConfig.nextClick(id, _this11.getItem(id, dd), function (res) {
+                            if (!res || !res.length) {
+                                _this11.handlerLabel(id, dd, !dd.hasClass(THIS));
+                            } else {
+                                dd.attr('tree-folder', 'true');
+                                var ddChilds = [];
+                                res.forEach(function (item, idx) {
+                                    item.innerHTML = item[ajaxConfig.keyName];
+                                    item[FORM_TEAM_PID] = JSON.stringify(treeId.split('-').concat([idx]));
+                                    ddChilds.push(_this11.createDD(id, item));
+                                    db[id][item[ajaxConfig.keyVal]] = item;
+                                });
+                                dd.after(ddChilds.join(''));
+                            }
+                        });
+                        return false;
+                    }
+                }
+            }
+
+            if (dd.hasClass(FORM_SELECT_TIPS)) {
+                //tips的处理
+                var btn = othis.is('.' + CZ) ? othis : othis.parents('.' + CZ);
+                if (!btn[0]) {
+                    return false;
+                }
+                var method = btn.attr('method');
+                var obj = data[id].config.btns.filter(function (bean) {
+                    return bean.name == method;
+                })[0];
+                obj && obj.click && obj.click instanceof Function && obj.click(id, _this11);
+                return false;
+            }
+            _this11.handlerLabel(id, dd, !dd.hasClass(THIS));
+            return false;
+        });
+    };
+
+    Common.prototype.addTreeHeight = function (dd, len) {
+        var _this12 = this;
+
+        var treeId = dd.attr('tree-id');
+        var childrens = dd.nextAll('dd[tree-id^="' + treeId + '"]');
+        if (childrens.length) {
+            dd.append('<i class="xm-iconfont icon-expand"></i>');
+            dd.attr('xm-tree-hidn', len);
+            childrens.each(function (index, item) {
+                var that = $(item);
+                _this12.addTreeHeight(that, len);
+            });
+        }
+    };
+
+    var db = {};
+    Common.prototype.getItem = function (id, value) {
+        if (value instanceof $) {
+            if (value.is('i[fsw="' + NAME + '"]')) {
+                var span = value.parent();
+                return db[id][value] || {
+                    name: span.find('font').text(),
+                    value: span.attr('value')
+                };
+            }
+            var val = value.attr('lay-value');
+            return !db[id][val] ? db[id][val] = {
+                name: value.find('span[name]').attr('name'),
+                value: val
+            } : db[id][val];
+        } else if (typeof value == 'string' && value.indexOf('/') != -1) {
+            return db[id][value] || {
+                name: this.valToName(id, value),
+                value: value
+            };
+        }
+        return db[id][value];
+    };
+
+    Common.prototype.linkageAdd = function (id, val) {
+        var dl = $('dl[xid="' + id + '"]');
+        dl.find('.xm-select-active').removeClass('xm-select-active');
+        var vs = val.value.split('/');
+        var pid = void 0,
+            li = void 0,
+            index = 0;
+        var lis = [];
+        do {
+            pid = vs[index];
+            li = dl.find('.xm-select-linkage-group' + (index + 1) + ' li[xm-value="' + pid + '"]');
+            li[0] && lis.push(li);
+            index++;
+        } while (li.length && pid != undefined);
+        if (lis.length == vs.length) {
+            $.each(lis, function (idx, item) {
+                item.addClass('xm-select-this');
+            });
+        }
+    };
+
+    Common.prototype.linkageDel = function (id, val) {
+        var dl = $('dl[xid="' + id + '"]');
+        var vs = val.value.split('/');
+        var pid = void 0,
+            li = void 0,
+            index = vs.length - 1;
+        do {
+            pid = vs[index];
+            li = dl.find('.xm-select-linkage-group' + (index + 1) + ' li[xm-value="' + pid + '"]');
+            if (!li.parent().next().find('li[pid=' + pid + '].xm-select-this').length) {
+                li.removeClass('xm-select-this');
+            }
+            index--;
+        } while (li.length && pid != undefined);
+    };
+
+    Common.prototype.valToName = function (id, val) {
+        var dl = $('dl[xid="' + id + '"]');
+        var vs = (val + "").split('/');
+        if (!vs.length) {
+            return null;
+        }
+        var names = [];
+        $.each(vs, function (idx, item) {
+            var name = dl.find('.xm-select-linkage-group' + (idx + 1) + ' li[xm-value="' + item + '"] span').text();
+            names.push(name);
+        });
+        return names.length == vs.length ? names.join('/') : null;
+    };
+
+    Common.prototype.commonHandler = function (key, label) {
+        if (!label || !label[0]) {
+            return;
+        }
+        this.checkHideSpan(key, label);
+        //计算input的提示语
+        this.changePlaceHolder(label);
+        //计算高度
+        this.retop(label.parents('.' + FORM_SELECT));
+        this.calcLabelLeft(label, 0, true);
+        //表单默认值
+        this.setHidnVal(key, label);
+        //title值
+        label.parents('.' + FORM_TITLE + ' .' + NAME).attr('title', data[key].values.map(function (val) {
+            return val.name;
+        }).join(','));
+    };
+
+    Common.prototype.initVal = function (id) {
+        var _this13 = this;
+
+        var target = {};
+        if (id) {
+            target[id] = data[id];
+        } else {
+            target = data;
+        }
+        $.each(target, function (key, val) {
+            var values = val.values,
+                div = $('dl[xid="' + key + '"]').parent(),
+                label = div.find('.' + LABEL),
+                dl = div.find('dl');
+            dl.find('dd.' + THIS).removeClass(THIS);
+
+            var _vals = values.concat([]);
+            _vals.concat([]).forEach(function (item, index) {
+                _this13.addLabel(key, label, item);
+                dl.find('dd[lay-value="' + item.value + '"]').addClass(THIS);
+            });
+            if (val.config.radio) {
+                _vals.length && values.push(_vals[_vals.length - 1]);
+            }
+            _this13.commonHandler(key, label);
+        });
+    };
+
+    Common.prototype.setHidnVal = function (key, label) {
+        if (!label || !label[0]) {
+            return;
+        }
+        label.parents('.' + PNAME).find('.' + HIDE_INPUT).val(data[key].values.map(function (val) {
+            return val.value;
+        }).join(','));
+    };
+
+    Common.prototype.handlerLabel = function (id, dd, isAdd, oval, notOn) {
+        var div = $('[xid="' + id + '"]').prev().find('.' + LABEL),
+            val = dd && this.getItem(id, dd),
+            vals = data[id].values,
+            on = data[id].config.on || events.on[id],
+            endOn = data[id].config.endOn || events.endOn[id];
+        if (oval) {
+            val = oval;
+        }
+        var fs = data[id];
+        if (isAdd && fs.config.max && fs.values.length >= fs.config.max) {
+            var maxTipsFun = events.maxTips[id] || data[id].config.maxTips;
+            maxTipsFun && maxTipsFun(id, vals.concat([]), val, fs.config.max);
+            return;
+        }
+        if (!notOn) {
+            if (on && on instanceof Function && on(id, vals.concat([]), val, isAdd, dd && dd.hasClass(DISABLED)) == false) {
+                return;
+            }
+        }
+        var dl = $('dl[xid="' + id + '"]');
+        isAdd ? (dd && dd[0] ? (dd.addClass(THIS), dd.removeClass(TEMP)) : dl.find('.xm-select-linkage')[0] && this.linkageAdd(id, val), this.addLabel(id, div, val), vals.push(val)) : (dd && dd[0] ? dd.removeClass(THIS) : dl.find('.xm-select-linkage')[0] && this.linkageDel(id, val), this.delLabel(id, div, val), this.remove(vals, val));
+        if (!div[0]) return;
+        //单选选完后直接关闭选择域
+        if (fs.config.radio) {
+            this.changeShow(div, false);
+        }
+        //移除表单验证的红色边框
+        div.parents('.' + FORM_TITLE).prev().removeClass('layui-form-danger');
+
+        //清空搜索值
+        fs.config.clearInput && this.clearInput(id);
+
+        this.commonHandler(id, div);
+
+        !notOn && endOn && endOn instanceof Function && endOn(id, vals.concat([]), val, isAdd, dd && dd.hasClass(DISABLED));
+    };
+
+    Common.prototype.addLabel = function (id, div, val) {
+        if (!val) return;
+        var tips = 'fsw="' + NAME + '"';
+        var _ref = [$('<span ' + tips + ' value="' + val.value + '"><font ' + tips + '>' + val.name + '</font></span>'), $('<i ' + tips + ' class="xm-iconfont icon-close"></i>')],
+            $label = _ref[0],
+            $close = _ref[1];
+
+        $label.append($close);
+        //如果是radio模式
+        var fs = data[id];
+        if (fs.config.radio) {
+            fs.values.length = 0;
+            $('dl[xid="' + id + '"]').find('dd.' + THIS + ':not([lay-value="' + val.value + '"])').removeClass(THIS);
+            div.find('span').remove();
+        }
+        //如果是固定高度
+        div.find('input').css('width', '50px');
+        div.find('input').before($label);
+    };
+
+    Common.prototype.delLabel = function (id, div, val) {
+        if (!val) return;
+        div.find('span[value="' + val.value + '"]:first').remove();
+    };
+
+    Common.prototype.checkHideSpan = function (id, div) {
+        var parentHeight = div.parents('.' + NAME)[0].offsetHeight + 5;
+        div.find('span.xm-span-hide').removeClass('xm-span-hide');
+        div.find('span[style]').remove();
+
+        var count = data[id].config.showCount;
+        div.find('span').each(function (index, item) {
+            if (index >= count) {
+                $(item).addClass('xm-span-hide');
+            }
+        });
+
+        var prefix = div.find('span:eq(' + count + ')');
+        prefix[0] && prefix.before($('<span style="padding-right: 6px;" fsw="' + NAME + '"> + ' + (div.find('span').length - count) + '</span>'));
+    };
+
+    Common.prototype.retop = function (div) {
+        //计算dl显示的位置
+        var dl = div.find('dl'),
+            top = div.offset().top + div.outerHeight() + 5 - $win.scrollTop(),
+            dlHeight = dl.outerHeight();
+        var up = div.hasClass('layui-form-selectup') || dl.css('top').indexOf('-') != -1 || top + dlHeight > $win.height() && top >= dlHeight;
+        div = div.find('.' + NAME);
+
+        var fs = data[dl.attr('xid')];
+        var base = dl.parents('.layui-form-pane')[0] && dl.prev()[0].clientHeight > 38 ? 14 : 10;
+        if (fs && fs.config.direction == 'up' || up) {
+            up = true;
+            if (fs && fs.config.direction == 'down') {
+                up = false;
+            }
+        }
+        var reHeight = div[0].offsetTop + div.height() + base;
+        if (up) {
+            dl.css({
+                top: 'auto',
+                bottom: reHeight + 3 + 'px'
+            });
+        } else {
+            dl.css({
+                top: reHeight + 'px',
+                bottom: 'auto'
+            });
+        }
+    };
+
+    Common.prototype.changeShow = function (children, isShow) {
+        //显示于隐藏
+        $('.layui-form-selected').removeClass('layui-form-selected');
+        var top = children.parents('.' + FORM_SELECT),
+            realShow = top.hasClass(FORM_SELECTED),
+            id = top.find('dl').attr('xid');
+        $('.' + PNAME + ' .' + FORM_SELECT).not(top).removeClass(FORM_SELECTED);
+        if (isShow) {
+            this.retop(top);
+            top.addClass(FORM_SELECTED);
+            top.find('.' + INPUT).focus();
+            if (!top.find('dl dd[lay-value]:not(.' + FORM_SELECT_TIPS + ')').length) {
+                top.find('dl .' + FORM_NONE).addClass(FORM_EMPTY);
+            }
+        } else {
+            top.removeClass(FORM_SELECTED);
+            this.clearInput(id);
+            top.find('dl .' + FORM_EMPTY).removeClass(FORM_EMPTY);
+            top.find('dl dd.' + DD_HIDE).removeClass(DD_HIDE);
+            top.find('dl dd.' + TEMP).remove();
+            //计算ajax数据是否为空, 然后重新请求数据
+            if (id && data[id] && data[id].config.isEmpty) {
+                this.triggerSearch(top);
+            }
+            this.changePlaceHolder(top.find('.' + LABEL));
+        }
+        if (isShow != realShow) {
+            var openFun = data[id].config.opened || events.opened[id];
+            isShow && openFun && openFun instanceof Function && openFun(id);
+            var closeFun = data[id].config.closed || events.closed[id];
+            !isShow && closeFun && closeFun instanceof Function && closeFun(id);
+        }
+    };
+
+    Common.prototype.changePlaceHolder = function (div) {
+        //显示于隐藏提示语
+        //调整pane模式下的高度
+        var title = div.parents('.' + FORM_TITLE);
+        title[0] || (title = div.parents('dl').prev());
+        if (!title[0]) {
+            return;
+        }
+
+        var id = div.parents('.' + PNAME).find('dl[xid]').attr('xid');
+        if (data[id] && data[id].config.height) {//既然固定高度了, 那就看着办吧
+
+        } else {
+            var height = title.find('.' + NAME)[0].clientHeight;
+            title.css('height', (height > 36 ? height + 4 : height) + 'px');
+            //如果是layui pane模式, 处理label的高度
+            var label = title.parents('.' + PNAME).parent().prev();
+            if (label.is('.layui-form-label') && title.parents('.layui-form-pane')[0]) {
+                height = height > 36 ? height + 4 : height;
+                title.css('height', height + 'px');
+                label.css({
+                    height: height + 2 + 'px',
+                    lineHeight: height - 18 + 'px'
+                });
+            }
+        }
+
+        var input = title.find('.' + TDIV + ' input'),
+            isShow = !div.find('span:last')[0] && !title.find('.' + INPUT).val();
+        if (isShow) {
+            var ph = input.attr('back');
+            input.removeAttr('back');
+            input.attr('placeholder', ph);
+        } else {
+            var _ph = input.attr('placeholder');
+            input.removeAttr('placeholder');
+            input.attr('back', _ph);
+        }
+    };
+
+    Common.prototype.indexOf = function (arr, val) {
+        for (var i = 0; i < arr.length; i++) {
+            if (arr[i].value == val || arr[i].value == (val ? val.value : val) || arr[i] == val || JSON.stringify(arr[i]) == JSON.stringify(val)) {
+                return i;
+            }
+        }
+        return -1;
+    };
+
+    Common.prototype.remove = function (arr, val) {
+        var idx = this.indexOf(arr, val ? val.value : val);
+        if (idx > -1) {
+            arr.splice(idx, 1);
+            return true;
+        }
+        return false;
+    };
+
+    Common.prototype.selectAll = function (id, isOn, skipDis) {
+        var _this14 = this;
+
+        var dl = $('[xid="' + id + '"]');
+        if (!dl[0]) {
+            return;
+        }
+        if (dl.find('.xm-select-linkage')[0]) {
+            return;
+        }
+        dl.find('dd[lay-value]:not(.' + FORM_SELECT_TIPS + '):not(.' + THIS + ')' + (skipDis ? ':not(.' + DISABLED + ')' : '')).each(function (index, item) {
+            item = $(item);
+            var val = _this14.getItem(id, item);
+            _this14.handlerLabel(id, dl.find('dd[lay-value="' + val.value + '"]'), true, val, !isOn);
+        });
+    };
+
+    Common.prototype.removeAll = function (id, isOn, skipDis) {
+        var _this15 = this;
+
+        var dl = $('[xid="' + id + '"]');
+        if (!dl[0]) {
+            return;
+        }
+        if (dl.find('.xm-select-linkage')[0]) {
+            //针对多级联动的处理
+            data[id].values.concat([]).forEach(function (item, idx) {
+                var vs = item.value.split('/');
+                var pid = void 0,
+                    li = void 0,
+                    index = 0;
+                do {
+                    pid = vs[index++];
+                    li = dl.find('.xm-select-linkage-group' + index + ':not(.xm-select-linkage-hide) li[xm-value="' + pid + '"]');
+                    li.click();
+                } while (li.length && pid != undefined);
+            });
+            return;
+        }
+        data[id].values.concat([]).forEach(function (item, index) {
+            if (skipDis && dl.find('dd[lay-value="' + item.value + '"]').hasClass(DISABLED)) {} else {
+                _this15.handlerLabel(id, dl.find('dd[lay-value="' + item.value + '"]'), false, item, !isOn);
+            }
+        });
+    };
+
+    Common.prototype.reverse = function (id, isOn, skipDis) {
+        var _this16 = this;
+
+        var dl = $('[xid="' + id + '"]');
+        if (!dl[0]) {
+            return;
+        }
+        if (dl.find('.xm-select-linkage')[0]) {
+            return;
+        }
+        dl.find('dd[lay-value]:not(.' + FORM_SELECT_TIPS + ')' + (skipDis ? ':not(.' + DISABLED + ')' : '')).each(function (index, item) {
+            item = $(item);
+            var val = _this16.getItem(id, item);
+            _this16.handlerLabel(id, dl.find('dd[lay-value="' + val.value + '"]'), !item.hasClass(THIS), val, !isOn);
+        });
+    };
+
+    Common.prototype.skin = function (id) {
+        var skins = ['default', 'primary', 'normal', 'warm', 'danger'];
+        var skin = skins[Math.floor(Math.random() * skins.length)];
+        $('dl[xid="' + id + '"]').parents('.' + PNAME).find('.' + FORM_SELECT).attr('xm-select-skin', skin);
+        this.check(id) && this.commonHandler(id, $('dl[xid="' + id + '"]').parents('.' + PNAME).find('.' + LABEL));
+    };
+
+    Common.prototype.getPosition = function (e) {
+        var x = 0,
+            y = 0;
+        while (e != null) {
+            x += e.offsetLeft;
+            y += e.offsetTop;
+            e = e.offsetParent;
+        }
+        return { x: x, y: y };
+    };
+
+    Common.prototype.onreset = function () {
+        //监听reset按钮, 然后重置多选
+        $(document).on('click', '[type=reset]', function (e) {
+            $(e.target).parents('form').find('.' + PNAME + ' dl[xid]').each(function (index, item) {
+                var id = item.getAttribute('xid'),
+                    dl = $(item),
+                    dd = void 0,
+                    temp = {};
+                common.removeAll(id);
+                data[id].config.init.forEach(function (val, idx) {
+                    if (val && (!temp[val] || data[id].config.repeat) && (dd = dl.find('dd[lay-value="' + val.value + '"]'))[0]) {
+                        common.handlerLabel(id, dd, true);
+                        temp[val] = 1;
+                    }
+                });
+            });
+        });
+    };
+
+    Common.prototype.bindEvent = function (name, id, fun) {
+        if (id && id instanceof Function) {
+            fun = id;
+            id = null;
+        }
+        if (fun && fun instanceof Function) {
+            if (!id) {
+                $.each(data, function (id, val) {
+                    data[id] ? data[id].config[name] = fun : events[name][id] = fun;
+                });
+            } else {
+                data[id] ? (data[id].config[name] = fun, delete events[name][id]) : events[name][id] = fun;
+            }
+        }
+    };
+
+    Common.prototype.check = function (id, notAutoRender) {
+        if ($('dl[xid="' + id + '"]').length) {
+            return true;
+        } else if ($('select[xm-select="' + id + '"]').length) {
+            if (!notAutoRender) {
+                this.render(id, $('select[xm-select="' + id + '"]'));
+                return true;
+            }
+        } else {
+            delete data[id];
+            return false;
+        }
+    };
+
+    Common.prototype.render = function (id, select) {
+        common.init(select);
+        common.one($('dl[xid="' + id + '"]').parents('.' + PNAME));
+        common.initVal(id);
+    };
+
+    Common.prototype.log = function (obj) {
+        console.log(obj);
+    };
+
+    var Select4 = function Select4() {
+        this.v = v;
+        this.render();
+    };
+    var common = new Common();
+
+    Select4.prototype.value = function (id, type, isAppend) {
+        if (typeof id != 'string') {
+            return [];
+        }
+        var fs = data[id];
+        if (!common.check(id)) {
+            return [];
+        }
+        if (typeof type == 'string' || type == undefined) {
+            var arr = fs.values.concat([]) || [];
+            if (type == 'val') {
+                return arr.map(function (val) {
+                    return val.value;
+                });
+            }
+            if (type == 'valStr') {
+                return arr.map(function (val) {
+                    return val.value;
+                }).join(',');
+            }
+            if (type == 'name') {
+                return arr.map(function (val) {
+                    return val.name;
+                });
+            }
+            if (type == 'nameStr') {
+                return arr.map(function (val) {
+                    return val.name;
+                }).join(',');
+            }
+            return arr;
+        }
+        if (common.isArray(type)) {
+            var dl = $('[xid="' + id + '"]'),
+                temp = {},
+                dd = void 0,
+                isAdd = true;
+            if (isAppend == false) {
+                //删除传入的数组
+                isAdd = false;
+            } else if (isAppend == true) {
+                //追加模式
+                isAdd = true;
+            } else {
+                //删除原有的数据
+                common.removeAll(id);
+            }
+            if (isAdd) {
+                fs.values.forEach(function (val, index) {
+                    temp[val.value] = 1;
+                });
+            }
+            type.forEach(function (val, index) {
+                if (val && (!temp[val] || fs.config.repeat)) {
+                    if ((dd = dl.find('dd[lay-value="' + val + '"]'))[0]) {
+                        common.handlerLabel(id, dd, isAdd, null, true);
+                        temp[val] = 1;
+                    } else {
+                        var name = common.valToName(id, val);
+                        if (name) {
+                            common.handlerLabel(id, dd, isAdd, common.getItem(id, val), true);
+                            temp[val] = 1;
+                        }
+                    }
+                }
+            });
+        }
+    };
+
+    Select4.prototype.on = function (id, fun, isEnd) {
+        common.bindEvent(isEnd ? 'endOn' : 'on', id, fun);
+        return this;
+    };
+
+    Select4.prototype.filter = function (id, fun) {
+        common.bindEvent('filter', id, fun);
+        return this;
+    };
+
+    Select4.prototype.maxTips = function (id, fun) {
+        common.bindEvent('maxTips', id, fun);
+        return this;
+    };
+
+    Select4.prototype.opened = function (id, fun) {
+        common.bindEvent('opened', id, fun);
+        return this;
+    };
+
+    Select4.prototype.closed = function (id, fun) {
+        common.bindEvent('closed', id, fun);
+        return this;
+    };
+
+    Select4.prototype.config = function (id, config, isJson) {
+        if (id && (typeof id === 'undefined' ? 'undefined' : _typeof(id)) == 'object') {
+            isJson = config == true;
+            config = id;
+            id = null;
+        }
+        if (config && (typeof config === 'undefined' ? 'undefined' : _typeof(config)) == 'object') {
+            if (isJson) {
+                config.header || (config.header = {});
+                config.header['Content-Type'] = 'application/json; charset=UTF-8';
+                config.dataType = 'json';
+            }
+            id ? (ajaxs[id] = $.extend(true, {}, ajaxs[id] || ajax, config), !common.check(id) && this.render(id), data[id] && config.direction && (data[id].config.direction = config.direction), data[id] && config.clearInput && (data[id].config.clearInput = true), config.searchUrl && data[id] && common.triggerSearch($('.' + PNAME + ' dl[xid="' + id + '"]').parents('.' + FORM_SELECT), true)) : ($.extend(true, ajax, config), $.each(ajaxs, function (key, item) {
+                $.extend(true, item, config);
+            }));
+        }
+        return this;
+    };
+
+    Select4.prototype.render = function (id, options) {
+        var _ref2;
+
+        if (id && (typeof id === 'undefined' ? 'undefined' : _typeof(id)) == 'object') {
+            options = id;
+            id = null;
+        }
+        var config = options ? (_ref2 = {
+            init: options.init,
+            disabled: options.disabled,
+            max: options.max,
+            isSearch: options.isSearch,
+            searchUrl: options.searchUrl,
+            isCreate: options.isCreate,
+            radio: options.radio,
+            skin: options.skin,
+            direction: options.direction,
+            height: options.height,
+            formname: options.formname,
+            layverify: options.layverify,
+            layverType: options.layverType,
+            showCount: options.showCount,
+            placeholder: options.placeholder,
+            create: options.create,
+            filter: options.filter,
+            maxTips: options.maxTips,
+            on: options.on
+        }, _defineProperty(_ref2, 'on', options.on), _defineProperty(_ref2, 'opened', options.opened), _defineProperty(_ref2, 'closed', options.closed), _defineProperty(_ref2, 'template', options.template), _defineProperty(_ref2, 'clearInput', options.clearInput), _ref2) : {};
+
+        options && options.searchType != undefined && (config.searchType = options.searchType == 'dl' ? 1 : 0);
+
+        if (id) {
+            fsConfigs[id] = {};
+            $.extend(fsConfigs[id], data[id] ? data[id].config : {}, config);
+        } else {
+            $.extend(fsConfig, config);
+        }
+
+        ($('select[' + NAME + '="' + id + '"]')[0] ? $('select[' + NAME + '="' + id + '"]') : $('select[' + NAME + ']')).each(function (index, select) {
+            var sid = select.getAttribute(NAME);
+            common.render(sid, select);
+            setTimeout(function () {
+                return common.setHidnVal(sid, $('select[xm-select="' + sid + '"] + div.' + PNAME + ' .' + LABEL));
+            }, 10);
+        });
+        return this;
+    };
+
+    Select4.prototype.disabled = function (id) {
+        var target = {};
+        id ? common.check(id) && (target[id] = data[id]) : target = data;
+
+        $.each(target, function (key, val) {
+            $('dl[xid="' + key + '"]').prev().addClass(DIS);
+        });
+        return this;
+    };
+
+    Select4.prototype.undisabled = function (id) {
+        var target = {};
+        id ? common.check(id) && (target[id] = data[id]) : target = data;
+
+        $.each(target, function (key, val) {
+            $('dl[xid="' + key + '"]').prev().removeClass(DIS);
+        });
+        return this;
+    };
+
+    Select4.prototype.data = function (id, type, config) {
+        if (!id || !type || !config) {
+            common.log('id: ' + id + ' param error !!!');
+            return this;
+        }
+        if (!common.check(id)) {
+            common.log('id: ' + id + ' not render !!!');
+            return this;
+        }
+        this.value(id, []);
+        this.config(id, config);
+        if (type == 'local') {
+            common.renderData(id, config.arr, config.linkage == true, config.linkageWidth ? config.linkageWidth : '100');
+        } else if (type == 'server') {
+            common.ajax(id, config.url, config.keyword, config.linkage == true, config.linkageWidth ? config.linkageWidth : '100');
+        }
+        return this;
+    };
+
+    Select4.prototype.btns = function (id, btns, config) {
+        if (id && common.isArray(id)) {
+            btns = id;
+            id = null;
+        }
+        if (!btns || !common.isArray(btns)) {
+            return this;
+        };
+        var target = {};
+        id ? common.check(id) && (target[id] = data[id]) : target = data;
+
+        btns = btns.map(function (obj) {
+            if (typeof obj == 'string') {
+                if (obj == 'select') {
+                    return quickBtns[0];
+                }
+                if (obj == 'remove') {
+                    return quickBtns[1];
+                }
+                if (obj == 'reverse') {
+                    return quickBtns[2];
+                }
+                if (obj == 'skin') {
+                    return quickBtns[3];
+                }
+            }
+            return obj;
+        });
+
+        $.each(target, function (key, val) {
+            val.config.btns = btns;
+            var dd = $('dl[xid="' + key + '"]').find('.' + FORM_SELECT_TIPS + ':first');
+            if (btns.length) {
+                var show = config && config.show && (config.show == 'name' || config.show == 'icon') ? config.show : '';
+                var html = common.renderBtns(key, show, config && config.space ? config.space : '30px');
+                dd.html(html);
+            } else {
+                var pcInput = dd.parents('.' + FORM_SELECT).find('.' + TDIV + ' input');
+                var _html = pcInput.attr('placeholder') || pcInput.attr('back');
+                dd.html(_html);
+                dd.removeAttr('style');
+            }
+        });
+
+        return this;
+    };
+
+    Select4.prototype.search = function (id, val) {
+        if (id && common.check(id)) {
+            ajaxs[id] = $.extend(true, {}, ajaxs[id] || ajax, {
+                first: true,
+                searchVal: val
+            });
+            common.triggerSearch($('dl[xid="' + id + '"]').parents('.' + FORM_SELECT), true);
+        }
+        return this;
+    };
+
+    Select4.prototype.replace = function (id, type, config) {
+        var _this17 = this;
+
+        if (!id || !type || !config) {
+            common.log('id: ' + id + ' param error !!!');
+            return this;
+        }
+        if (!common.check(id, true)) {
+            common.log('id: ' + id + ' not render !!!');
+            return this;
+        }
+        var oldVals = this.value(id, 'val');
+        this.value(id, []);
+        this.config(id, config);
+        if (type == 'local') {
+            common.renderData(id, config.arr, config.linkage == true, config.linkageWidth ? config.linkageWidth : '100', false, true);
+            this.value(id, oldVals, true);
+        } else if (type == 'server') {
+            common.ajax(id, config.url, config.keyword, config.linkage == true, config.linkageWidth ? config.linkageWidth : '100', false, function (id) {
+                _this17.value(id, oldVals, true);
+            }, true);
+        }
+    };
+
+    return new Select4();
+});

+ 2 - 2
assets/lib/layui/upload.js

@@ -31,7 +31,7 @@ function showUpload(i,e,key,url,assets)
 		var config = {
 		  type: 0,
 		  title: e.attr('value'),
-		  shade: 0,
+		  shade: 0.1,
 		  shadeClose : true,
 		  content: html,
 		  area: '800px'
@@ -63,7 +63,7 @@ function editorShowUpload(cur, url, title, func, self)
 		var config = {
 		  type: 0,
 		  title: title,
-		  shade: 0,
+		  shade: 0.1,
 		  shadeClose : true,
 		  content: html,
 		  area: '800px'

+ 9 - 1
assets/lib/manage/main.js

@@ -1748,7 +1748,15 @@ function fastEdit(e, url, title, col)
 		html = html + valid;
 		showAlert(html, func, title);
 		$('#update_button').hide();
-		init();
+		layui.use(['formSelects'], function(){
+			$("select").each(function()
+			{
+				if ($(this).attr('xm-select')) {
+					layui.formSelects.render($(this).attr('xm-select'));
+				}
+			})
+		  });
+		
 	});
 }
 

Some files were not shown because too many files changed in this diff