上传按钮<input type="file" />智能多效美化

原文链接:Styling & Customizing File Inputs the Smart Way(作者:Osvaldas Valutis)

这份教程主要介绍如何将<input type="file" />与<label>配合、以保持语义化+可访问性为目标辅以少量Javascript来美化。

实例

网上流传过一些“自定义”<input type="file" />元素的方法,试验过但都实现不了像Readerrr上那种上传文件即时显示到内容流里的效果。兴许最烂的做法是:把input直接放进模拟按钮的容器、100%大填满容器、点击容器的任意区域即等于点击input。听起来可行但实际又比较奇怪对吧?并且实际上这样做也会带来一些不可接受的缺失,比如可用性、比如对touch事件的支持等等。

后来我在StackOverflow网站上发现一条隐藏得很深的灵感,有些人顶它但是被淹没在该主题的回复里。这条回复的重点只有一个——<label>!好用在于,它可以跟input的click事件关联上,这就实现了语义化解决方案。

<input type="file" name="file" id="file" class="inputfile" />
<label for="file">Choose a file</label>

因此点击这两个元素的任何一个都能得到相同的结果——弹出文件上传选择对话框。这意味着最难的部分解决了。没有JavaScript,不需要其他比如捕捉鼠标位置之类的麻烦方案,只需要两行代码。如图:

上传效果

现在剩下的工作就是将它当成一个普通按钮来尽情美化。

隐藏<input>

首先,部分CSS属性对隐藏<input type="file" />无效

display: none;
visibility: hidden;

原因是:这样做之后input将不能提交数据,也会被tab顺序跳过(键盘可用性一下子就完蛋了)。为了完美实现视觉不可见但浏览器可辨认,我测试出了这样的CSS组合:

.inputfile {
    width: 0.1px; 
    height: 0.1px; 
    opacity: 0; 
    overflow: hidden; 
    position: absolute; 
    z-index: -1;
}

你可能会好奇为什么宽和高会设成0.1px而不是0px,因为在某些浏览器下0宽高将会让<input>元素被tab键忽略。而position: absolute的目的是不干扰随后元素的位置。

美化<label>

对<label>的修饰就随意了,先来几行简单的效果:

.inputfile + label {
    font-size: 1.25em;
    font-weight: 700;
    color: white;
    background-color: black;
    display: inline-block;
}
.inputfile:focus + label,
.inputfile + label:hover {
    background-color: red;
}
效果

可用性

一个用户要如何知道网页上的某个元素是“可点”?首先它要传达出一种可摁或者可点的暗示;其次hover时光标应该直接体现。而<label>默认没有变小手的效果,那么加上:

键盘导航

如果用户不能使用键盘访问到这个按钮,设计就失败了。除了做到优雅隐藏外,还应该在focus状态时有所反馈:

.inputfile:focus + label {
    outline: 1px dotted #000; 
    outline: -webkit-focus-ring-color auto 5px;
}

-webkit-focus-ring-color auto 5px是Chrome/Opera/Safari上的一个默认支持的小技巧。

Touch事件上可能存在的问题

如果你正在使用FastClick(一个消除可触屏上的300毫秒反馈延迟的库)之类的东西,并且想对label增加一些额外属性,按钮有可能出现问题,这时候就需要加上:

pointer-events: none;

JavaScript辅助

最后还不能忘记一件事:判断文件是否已选中<input type="file" />既然已经被隐藏,你就看不到选上了的文件名。这个时候就只能再依赖一小段JavaScript,使<label>的文本显示为选中的文件名。如果有多个文件被选中,文本还应该显示出文件总数。
<input type="file" name="file" id="file" class="inputfile" data-multiple-caption="{count} files selected" multiple />

var inputs = document.querySelectorAll( '.inputfile' );
Array.prototype.forEach.call( inputs, function( input ) {
  var label = input.nextElementSibling,     
  labelVal = label.innerHTML;   
  input.addEventListener( 'change', function( e ) { 
    var fileName = '';
    if( this.files && this.files.length > 1 )           
      fileName = ( this.getAttribute( 'data-multiple-caption' ) || '' ).replace( '{count}', this.files.length ); 
    else
      fileName = e.target.value.split( '\\' ).pop();

    if( fileName )
      label.querySelector( 'span' ).innerHTML = fileName;
      else label.innerHTML = labelVal;  
    });
});

jQuery的话:

$( '.inputfile' ).each( function() {
    var $input = $( this ),
        $label = $input.next( 'label' ),
        labelVal = $label.html();

    $input.on( 'change', function( e ) {
        var fileName = '';

        if( this.files && this.files.length > 1 )
            fileName = ( this.getAttribute( 'data-multiple-caption' ) || '' ).replace( '{count}', this.files.length );
        else if( e.target.value )
            fileName = e.target.value.split( '\\' ).pop();

        if( fileName )
            $label.find( 'span' ).html( fileName );
        else
            $label.html( labelVal );
    });
});

部分解释:

  • 保留原生[multiple]特性允许一次上传多个文件,而[data-multiple-caption]是自定义特性,用于显示被选中的文件数,同时也可以设置自定义文本。{count}则是可选项,文件数将显示在内部。没有使用JS变量而是直接增加HTML特性的目的是,后者在同一页内多次使用时比较便利。
  • [multiple]在IE9及以下不被支持,也不属于JavaScript的原生files属性。对于后者,我们可以简单地使用value(值),比如c:\path\filename.jpg这样的路径,split( '\\' ).pop()就能一步到位获取到文件名。
  • 有个有趣的细节:在弹出文件选择对话框后你可以摁ESC键重设/清空input的值,虽然有可能限于Chrome/Opera。因此设计出labelVal这个变量来装这个值,并且随时回头利用。

效果如图:

如果不支持JavaScript怎么办?

鉴于没有JavaScript以外的方式能获取文件是否选中的状态,最好的办法就是依赖<input type="file" />的原生外观。只需要对<html>增加一个.no-js的class名,再用JS去改变为.js——以辨识出JS是否得到支持。

<html class="no-js">
    <head>
        <!-- remove this if you use Modernizr -->
        <script>
            (function(e,t,n) {
                var r=e.querySelectorAll("html")[0];
                r.className=r.className.replace(/(^|\s)no-js(\s|$)/,"$1js$2")
            })(document,window,0);
        </script>
    </head>
</html>

CSS相应为:

.js .inputfile {
    width: 0.1px;
    height: 0.1px;
    opacity: 0;
    overflow: hidden;
    position: absolute;
    z-index: -1;
}
.no-js .inputfile + label {
    display: none;
}

Firefox bug

可能大家都没想到,Firefox完全忽略input[type="file"]:focus这样的定义,却完美支持:hover:active!更意外的是,Firefox竟然支持在JavaScript里获取焦点,因此解决方法就是多加两行:

input.addEventListener( 'focus', function() {
    input.classList.add( 'has-focus' );
});
input.addEventListener( 'blur', function() {
    input.classList.remove( 'has-focus' );
});
.inputfile:focus + label,
.inputfile.has-focus + label {
    outline: 1px dotted #000;
    outline: -webkit-focus-ring-color auto 5px;
}

DEMO:http://tympanus.net/Tutorials/CustomFileInputs/
打包下载:http://tympanus.net/Tutorials/CustomFileInputs/CustomFileInputs.zip

题外话

有用户指出这样实现的上传按钮无法通过拖拽的方式来上传——毕竟这是<input type="file" />的原生功能。有人建议可以辅用Plupload或者Dropzone.js之类的插件,但此人反驳道本来就支持的功能现在失去了,再找其他东西来修补是一种脱裤放屁的行为。楼主回复认为,上传按钮的拖拽支持本来就非常隐蔽,并不是明面上可见的功能,除了一些习惯泡在网上的技术男,绝大部分普通用户看到一个形象的上传按钮时肯定是点击它而不是往上面拖文件。


除了设宽高为0,有人给出隐藏input但不让它失效的其他方法:

input {
    position: absolute;
    opacity: 0.01;
}

也就是把透明度设为一个极小的值。目前未测试不确定适用范围(该人声称已在其公司的项目里广泛应用)。


还有一些人提出一个比较常见的理解:

input {
    position: absolute;
    width: 容器宽;
    height: 容器高;
    opacity: 0;
    user-select: none;
}

透明度为0,同时不允许被选中,绝对定位覆盖填满容器,然后点击的时候就肯定点击在input上。看似很好理解并且实现简单,但作者指出在IE10及以下这种做法可能导致双击才能生效:在线测试


进而有人提出需求:如果还能清除选中的文件名就好了。作者链接到另一个StackOverflow上的帖子,那边有人问如何用jQuery清空<input type="file" />的值,因为这样做无效:

$('#control').attr({ value: '' }); 

那边帖子有一个顶得最高的解决方法,颇为巧妙:

function resetFormElement(e) {
  e.wrap('<form>').closest('form').get(0).reset();
  e.unwrap();

  // Prevent form submission
  e.stopPropagation();
  e.preventDefault();
}

具体做法就是产生一个外包<form>,然后原生reset掉其内的<input>,再去掉这个外壳。最后添加preventDefault的目的是,假如表单里存在一个触发reset的button,这样做可以避免上面的JS触发它。


也有人提出把<input>放在<label>里,显得更天然更合理:

<label class="upload">
    Choose a file...
    <input type="file" />
</label>

但作者指出如果需要捕捉<input>是否有:focus就只能依赖JS了,因此在楼例里作者把<label>放在<input>后面,利用CSSinput + label去选择到它。


最后有人提出如果动态添加多个<input type="file" />,以上的方法就不够全面,希望作者能给出一个思路,到目前为止作者没有回复。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,242评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,769评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,484评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,133评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,007评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,080评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,496评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,190评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,464评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,549评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,330评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,205评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,567评论 3 298
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,889评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,160评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,475评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,650评论 2 335

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,049评论 25 707
  • 发现 关注 消息 iOS 第三方库、插件、知名博客总结 作者大灰狼的小绵羊哥哥关注 2017.06.26 09:4...
    肇东周阅读 11,943评论 4 60
  • JavaScript 资源大全中文版很多程序员应该记得 GitHub 上有一个 Awesome - XXX 系列的...
    wwmin_阅读 3,391评论 1 92
  • 成家立业后也很想买一只小狗养养,可是不久怀孕了,生娃了,一娃生好又怀了二娃,老公本来说等大娃上幼儿园了就一起去挑一...
    风吹海树阅读 317评论 0 3
  • 文/沵花 不知道从什么时候开始,我们不是为了自己而活。 世界那么大,每天都有变化。快节奏的生活催促着我们应该不断的...
    沵花阅读 634评论 0 0