ztreecustomer

Ace文件输入插件是一款用于增强HTML文件输入元素的功能和样式的jQuery插件。它提供了多种样式选项,如显示已选文件的缩略图、拖放上传等功能,并允许用户通过设置回调函数来控制文件的有效性。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


$('#id-input-file-1 , #id-input-file-2').ace_file_input({
                no_file:'没有选择文件',
                btn_choose:'选择',
                btn_change:'改变',
                droppable:false,
                onchange:null,
                thumbnail:false //| true | large
                //whitelist:'gif|png|jpg|jpeg'
                //blacklist:'exe|php'
                //onchange:''
                //
            });

            $('#id-input-file-3').ace_file_input({
                style:'well',
                btn_choose:'Drop files here or click to choose',
                btn_change:null,
                no_icon:'icon-cloud-upload',
                droppable:true,
                onchange:null,
                thumbnail:'small',
                before_change:function(files, dropped) {
                    /**
                    if(files instanceof Array || (!!window.FileList && files instanceof FileList)) {
                        //check each file and see if it is valid, if not return false or make a new array, add the valid files to it and return the array
                        //note: if files have not been dropped, this does not change the internal value of the file input element, as it is set by the browser, and further file uploading and handling should be done via ajax, etc, otherwise all selected files will be sent to server
                        //example:
                        var result = []
                        for(var i = 0; i < files.length; i++) {
                            var file = files[i];
                            if((/^image\//i).test(file.type) && file.size < 102400)
                                result.push(file);
                        }
                        return result;
                    }
                    */
                    return true;
                }
                /*,
                before_remove : function() {
                    return true;
                }*/
            }).on('change', function(){
                //console.log($(this).data('ace_input_files'));
                //console.log($(this).data('ace_input_method'));
            });

(function($ , undefined) {
    var multiplible = 'multiple' in document.createElement('INPUT');
    var hasFileList = 'FileList' in window;//file list enabled in modern browsers
    var hasFileReader = 'FileReader' in window;

    var Ace_File_Input = function(element , settings) {
        var self = this;
        this.settings = $.extend({}, $.fn.ace_file_input.defaults, settings);

        this.$element = $(element);
        this.element = element;
        this.disabled = false;
        this.can_reset = true;

        this.$element.on('change.ace_inner_call', function(e , ace_inner_call){
            if(ace_inner_call === true) return;//this change event is called from above drop event
            return handle_on_change.call(self);
        });

        this.$element.wrap('<div class="ace-file-input" />');

        this.apply_settings();
    }
    Ace_File_Input.error = {
        'FILE_LOAD_FAILED' : 1,
        'IMAGE_LOAD_FAILED' : 2,
        'THUMBNAIL_FAILED' : 3
    };


    Ace_File_Input.prototype.apply_settings = function() {
        var self = this;
        var remove_btn = !!this.settings.icon_remove;

        this.multi = this.$element.attr('multiple') && multiplible;
        this.well_style = this.settings.style == 'well';

        if(this.well_style) this.$element.parent().addClass('ace-file-multiple');
         else this.$element.parent().removeClass('ace-file-multiple');

        this.$element.parent().find(':not(input[type=file])').remove();//remove all except our input, good for when changing settings
        this.$element.after('<label class="file-label" data-title="'+this.settings.btn_choose+'"><span class="file-name" data-title="'+this.settings.no_file+'">'+(this.settings.no_icon ? '<i class="'+this.settings.no_icon+'"></i>' : '')+'</span></label>'+(remove_btn ? '<a class="remove" href="#"><i class="'+this.settings.icon_remove+'"></i></a>' : ''));
        this.$label = this.$element.next();

        this.$label.on('click', function(){//firefox mobile doesn't allow 'tap'!
            if(!this.disabled && !self.element.disabled && !self.$element.attr('readonly')) 
                self.$element.click();
        })

        if(remove_btn) this.$label.next('a').on(ace.click_event, function(){
            if(! self.can_reset ) return false;

            var ret = true;
            if(self.settings.before_remove) ret = self.settings.before_remove.call(self.element);
            if(!ret) return false;
            return self.reset_input();
        });


        if(this.settings.droppable && hasFileList) {
            enable_drop_functionality.call(this);
        }
    }

    Ace_File_Input.prototype.show_file_list = function($files) {
        var files = typeof $files === "undefined" ? this.$element.data('ace_input_files') : $files;
        if(!files || files.length == 0) return;

        //////////////////////////////////////////////////////////////////

        if(this.well_style) {
            this.$label.find('.file-name').remove();
            if(!this.settings.btn_change) this.$label.addClass('hide-placeholder');
        }
        this.$label.attr('data-title', this.settings.btn_change).addClass('selected');

        for (var i = 0; i < files.length; i++) {
            var filename = typeof files[i] === "string" ? files[i] : $.trim( files[i].name );
            var index = filename.lastIndexOf("\\") + 1;
            if(index == 0)index = filename.lastIndexOf("/") + 1;
            filename = filename.substr(index);

            var fileIcon = 'icon-file';
            if((/\.(jpe?g|png|gif|svg|bmp|tiff?)$/i).test(filename)) {
                fileIcon = 'icon-picture';
            }
            else if((/\.(mpe?g|flv|mov|avi|swf|mp4|mkv|webm|wmv|3gp)$/i).test(filename)) fileIcon = 'icon-film';
            else if((/\.(mp3|ogg|wav|wma|amr|aac)$/i).test(filename)) fileIcon = 'icon-music';


            if(!this.well_style) this.$label.find('.file-name').attr({'data-title':filename}).find('[class*="icon-"]').attr('class', fileIcon);
            else {
                this.$label.append('<span class="file-name" data-title="'+filename+'"><i class="'+fileIcon+'"></i></span>');
                var type = $.trim(files[i].type);
                var can_preview = hasFileReader && this.settings.thumbnail 
                        &&
                        ( (type.length > 0 && type.match('image')) || (type.length == 0 && fileIcon == 'icon-picture') )//the second one is for Android's default browser which gives an empty text for file.type
                if(can_preview) {
                    var self = this;
                    $.when(preview_image.call(this, files[i])).fail(function(result){
                        //called on failure to load preview
                        if(self.settings.preview_error) self.settings.preview_error.call(self, filename, result.code);
                    });
                }
            }

        }

        return true;
    }

    Ace_File_Input.prototype.reset_input = function() {
      this.$label.attr({'data-title':this.settings.btn_choose, 'class':'file-label'})
            .find('.file-name:first').attr({'data-title':this.settings.no_file , 'class':'file-name'})
            .find('[class*="icon-"]').attr('class', this.settings.no_icon)
            .prev('img').remove();
            if(!this.settings.no_icon) this.$label.find('[class*="icon-"]').remove();

        this.$label.find('.file-name').not(':first').remove();

        if(this.$element.data('ace_input_files')) {
            this.$element.removeData('ace_input_files');
            this.$element.removeData('ace_input_method');
        }

        this.reset_input_field();

        return false;
    }

    Ace_File_Input.prototype.reset_input_field = function() {
        //http://stackoverflow.com/questions/1043957/clearing-input-type-file-using-jquery/13351234#13351234
        this.$element.wrap('<form>').closest('form').get(0).reset();
        this.$element.unwrap();
    }

    Ace_File_Input.prototype.enable_reset = function(can_reset) {
        this.can_reset = can_reset;
    }

    Ace_File_Input.prototype.disable = function() {
        this.disabled = true;
        this.$element.attr('disabled', 'disabled').addClass('disabled');
    }
    Ace_File_Input.prototype.enable = function() {
        this.disabled = false;
        this.$element.removeAttr('disabled').removeClass('disabled');
    }

    Ace_File_Input.prototype.files = function() {
        return $(this).data('ace_input_files') || null;
    }
    Ace_File_Input.prototype.method = function() {
        return $(this).data('ace_input_method') || '';
    }

    Ace_File_Input.prototype.update_settings = function(new_settings) {
        this.settings = $.extend({}, this.settings, new_settings);
        this.apply_settings();
    }



    var enable_drop_functionality = function() {
        var self = this;
        var dropbox = this.element.parentNode;      
        $(dropbox).on('dragenter', function(e){
            e.preventDefault();
            e.stopPropagation();
        }).on('dragover', function(e){
            e.preventDefault();
            e.stopPropagation();
        }).on('drop', function(e){
            e.preventDefault();
            e.stopPropagation();

            var dt = e.originalEvent.dataTransfer;
            var files = dt.files;
            if(!self.multi && files.length > 1) {//single file upload, but dragged multiple files
                var tmpfiles = [];
                tmpfiles.push(files[0]);
                files = tmpfiles;//keep only first file
            }

            var ret = true;
            if(self.settings.before_change) ret = self.settings.before_change.call(self.element, files, true);//true means files have been dropped
            if(!ret || ret.length == 0) {
                return false;
            }

            //user can return a modified File Array as result
            if(ret instanceof Array || (hasFileList && ret instanceof FileList)) files = ret;


            self.$element.data('ace_input_files', files);//save files data to be used later by user
            self.$element.data('ace_input_method', 'drop');


            self.show_file_list(files);


            self.$element.triggerHandler('change' , [true]);//true means inner_call
            return true;
        });
    }


    var handle_on_change = function() {
        var ret = true;
        if(this.settings.before_change) ret = this.settings.before_change.call(this.element, this.element.files || [this.element.value]/*make it an array*/, false);//false means files have been selected, not dropped
        if(!ret || ret.length == 0) {
            if(!this.$element.data('ace_input_files')) this.reset_input_field();//if nothing selected before, reset because of the newly unacceptable (ret=false||length=0) selection
            return false;
        }


        //user can return a modified File Array as result
        var files = !hasFileList ? null ://for old IE, etc
                    ((ret instanceof Array || ret instanceof FileList) ? ret : this.element.files);
        this.$element.data('ace_input_method', 'select');


        if(files && files.length > 0) {//html5
            this.$element.data('ace_input_files', files);
        }
        else {
            var name = $.trim( this.element.value );
            if(name && name.length > 0) {
                files = []
                files.push(name);
                this.$element.data('ace_input_files', files);
            }
        }

        if(!files || files.length == 0) return false;
        this.show_file_list(files);

        return true;
    }




    var preview_image = function(file) {
        var self = this;
        var $span = self.$label.find('.file-name:last');//it should be out of onload, otherwise all onloads may target the same span because of delays

        var deferred = new $.Deferred
        var reader = new FileReader();
        reader.onload = function (e) {
            $span.prepend("<img class='middle' style='display:none;' />");
            var img = $span.find('img:last').get(0);

            $(img).one('load', function() {
                //if image loaded successfully
                var size = 50;
                if(self.settings.thumbnail == 'large') size = 150;
                else if(self.settings.thumbnail == 'fit') size = $span.width();
                $span.addClass(size > 50 ? 'large' : '');

                var thumb = get_thumbnail(img, size, file.type);
                if(thumb == null) {
                    //if making thumbnail fails
                    $(this).remove();
                    deferred.reject({code:Ace_File_Input.error['THUMBNAIL_FAILED']});
                    return;
                }

                var w = thumb.w, h = thumb.h;
                if(self.settings.thumbnail == 'small') {w=h=size;};
                $(img).css({'background-image':'url('+thumb.src+')' , width:w, height:h})                                   
                        .data('thumb', thumb.src)
                        .attr({src:''})
                        .show()

                ///////////////////
                deferred.resolve();
            }).one('error', function() {
                //for example when a file has image extenstion, but format is something else
                $span.find('img').remove();
                deferred.reject({code:Ace_File_Input.error['IMAGE_LOAD_FAILED']});
            });

            img.src = e.target.result;
        }
        reader.onerror = function (e) {
            deferred.reject({code:Ace_File_Input.error['FILE_LOAD_FAILED']});
        }
        reader.readAsDataURL(file);

        return deferred.promise();
    }

    var get_thumbnail = function(img, size, type) {

        var w = img.width, h = img.height;
        if(w > size || h > size) {
          if(w > h) {
            h = parseInt(size/w * h);
            w = size;
          } else {
            w = parseInt(size/h * w);
            h = size;
          }
        }

        var dataURL
        try {
            var canvas = document.createElement('canvas');
            canvas.width = w; canvas.height = h;
            var context = canvas.getContext('2d');
            context.drawImage(img, 0, 0, img.width, img.height, 0, 0, w, h);
            dataURL = canvas.toDataURL(/*type == 'image/jpeg' ? type : 'image/png', 10*/)
        } catch(e) {
            dataURL = null;
        }

        //there was only one image that failed in firefox completely randomly! so let's double check it
        if(!( /^data\:image\/(png|jpe?g|gif);base64,[0-9A-Za-z\+\/\=]+$/.test(dataURL)) ) dataURL = null;
        if(! dataURL) return null;

        return {src: dataURL, w:w, h:h};
    }



    ///////////////////////////////////////////
    $.fn.ace_file_input = function (option,value) {
        var retval;

        var $set = this.each(function () {
            var $this = $(this);
            var data = $this.data('ace_file_input');
            var options = typeof option === 'object' && option;

            if (!data) $this.data('ace_file_input', (data = new Ace_File_Input(this, options)));
            if (typeof option === 'string') retval = data[option](value);
        });

        return (retval === undefined) ? $set : retval;
    };


    $.fn.ace_file_input.defaults = {
        style:false,
        no_file:'No File ...',
        no_icon:'icon-upload-alt',
        btn_choose:'Choose',
        btn_change:'Change',
        icon_remove:'icon-remove',
        droppable:false,
        thumbnail:false,//large, fit, small

        //callbacks
        before_change:null,
        before_remove:null,
        preview_error:null
     }


})(window.jQuery);
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值