拖拽异步上传实现,自定义标签在IE6

H5页检测手机是否安装app 相关流程

2017/04/21 · HTML5 · 2 评论 · H5

原文出处: sunsmeill   

近期公司需要针对分享流程进行优化,其中一点就是前端H5检测是否安装应用,来进行不同的判断(下载或直接跳转到app中)。原理很简单:创建一个iframe去打开uri。如果打开app成功网页进入后台,再切换回来时间会超过2.5s。利用时间去检测。下面来看具体实现过程:

自定义标签在IE6-8的困境

2015/07/20 · HTML5 · IE, 自定义标签

原文出处: 司徒正美   

或许未来前端组件化之路都是自定义标签,但这东西早在20年前,JSTL已在搞了。现在Web Component还只有webkit支持。但一个组件库,还需要一个特殊的标识它们是一块的。不过这个XML已经帮我们搞定了,使用scopeName,如”<xxx:dialog>”。在我继续往下想如何处理如何为这个标签绑定数据,与其他组件通信,管理生命周期,等等大事之前,我还有一个不得不面对的问题,就是如何兼容IE6-8!

比如以下一个页面:

图片 1

在chrome, firefox, IE11, IE11的IE6兼容模式分别如下:

图片 2
图片 3
图片 4
图片 5

我们会发现IE6下实际是多出许多标签,它是把闭标签也变成一个独立的元素节点

图片 6

这个AA:DIV标签被开膛破肚,里面子节点全部暴出来,成为其兄弟节点了。因此想兼容它,就要费点劲。有个两个情况需要考虑,1是用户已经将它写在页面上,情况同上;2是用户是将它放在字符串模版中,这个用正则搞定。不过正则要是碰上复杂的属性名,还是会晕掉。因此我还是打算使用原生的HTML parser。换言之,字符串,我还是会将它变成节点。这么办呢?!我想了许多办法,后来还是使用VML的命名空间法搞定!

我们将上面的页面改复杂点,再看看效果!

图片 7
图片 8

可以看到其套嵌关系现在完全正确,并且标签名不会大写化,也不会生成多余节点!

好了,我们再判定一下是否为自定义标签,或者准确地说,这个节点是否我们组件库中定义的自定义标签。某些情况下,一个页面可以存在多套组件库,包括avalon的,ploymer的,或者是直接用Web Component写的。

avalon的组件库将使用命名空间,这样就好区别开。在IE6-9中,判定element.scopeName是否为aa(这是组件库的命名空间,你可以改个更高大上的名字),在其他浏览器判定此元素的localName是否以aa:开头就行了!

JavaScript

function isWidget(el, uiName){ return el.scopeName ? el.scopeName === uiName: el.localName.indexOf(uiName+":") === 0 }

1
2
3
function isWidget(el, uiName){
  return   el.scopeName ? el.scopeName === uiName: el.localName.indexOf(uiName+":") === 0
}

这个难题解决后,我们就可以开搞基于自定义标签的UI库了!

1 赞 1 收藏 评论

图片 9

File杂谈——拖拽异步上传实现

2015/07/25 · HTML5 · 异步上传

原文出处: 百码山庄   

在前一篇文章《File杂谈——拖拽上传前传》中我制作了一个静态的拖拽上传界面,拖拽文件到显示区域释放,可以显示拖入文件的基本信息。本文将在此基础上进一步加工,打造一个完整的拖拽上传示例。

你可能会遇到的问题

  • 什么是uri,获取uri需要哪些帮助?
  • 安卓中应用切换到后台, 计时器仍会不断运行有什么解决方法?
  • 微信中不支持第三方uri,下载应用。怎么解决来完成跳转到自身app。

都会在文中找到答案。

示例说明

点击区域选择文件或直接将文件拖入区域,触发文件上传功能,文件将异步发送到服务器。待服务端处理完成后返回基本信息,在页面中显示。由于服务器容量问题,本示例未做文件保存处理,只是简单的将文件基本信息返回,文件上传的后端具体处理逻辑需要自行补充。

uri获取

这里的uri,指得就是通过 Url scheme 来实现的H5与安卓、苹果应用之间的跳转链接。

我们需要找到客户端的同事,来获取如下格式的链接。

xx://'跳转页面'/'携带参数'

1
xx://'跳转页面'/'携带参数'

这里给大家简单解释下url scheme。

url 就是我们平常理解的链接。
scheme 是指url链接中的最初位置,就是上边链接中 ‘xx’的位置。
详细介绍可以看这里:使用url scheme详解

用这个链接我们可以跳转到 应用中的某个页面,并可以携带一定的参数。这个是我们实现这个功能的前提哟。

新的小伙伴FormData

我们知道,传统的文件上传如果要实现异步的效果,我们会使用iframe去模拟,或使用flash上传插件。但是今天,我们又认识了一位新成员——FromData,它可以通过js创建表单对象,并可以向该对象中添加表单数据(字符串、数字、文件等)。再结合我们熟悉的XMLHttpRequest对象,将表单数据异步提交到服务端,这样我们的问题就解决了。

下面,我们来看下核心代码:

JavaScript

function uploadFile(fs) { var len = fs.length; for (var i = 0; i < len; i++) { sendFile(fs[i]); } } function sendFile(file) { var xhr = new XMLHttpRequest(), fd = new FormData(); fd.append('file', file); xhr.onreadystatechange = function() { if (xhr.readyState == 4 && xhr.status == 200) { // 将服务端返回信息输出到日志区(考虑多文件情况) consoleDiv.innerHTML += '<br>' + xhr.responseText; } }; xhr.open('POST', './upload.php'); xhr.send(fd); } // 文件控件发生变化时,调用uploadFile函数,触发上传功能 file.onchange = function() { uploadFile(this.files); }; // 在区域内释放拖入文件时,调用文件上传函数 area.ondrop = function(ev) { ev.preventDefault(); var dt = ev.dataTransfer; uploadFile(dt.files); };

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
function uploadFile(fs) {
    var len = fs.length;
    for (var i = 0; i < len; i++) {
        sendFile(fs[i]);
    }
}
function sendFile(file) {
    var xhr = new XMLHttpRequest(),
        fd = new FormData();
    fd.append('file', file);
    xhr.onreadystatechange = function() {
        if (xhr.readyState == 4 && xhr.status == 200) {
            // 将服务端返回信息输出到日志区(考虑多文件情况)
            consoleDiv.innerHTML += '<br>' + xhr.responseText;
        }
    };
    xhr.open('POST', './upload.php');
    xhr.send(fd);
}
// 文件控件发生变化时,调用uploadFile函数,触发上传功能
file.onchange = function() {
    uploadFile(this.files);
};
// 在区域内释放拖入文件时,调用文件上传函数
area.ondrop = function(ev) {
    ev.preventDefault();
    var dt = ev.dataTransfer;
    uploadFile(dt.files);
};

代码很简单,不再做过多阐述。但是这里我想发表一点个人意见:根据示例我们不难发现有这么一个问题——如果用户都使用拖拽上传功能,而不使用点击触发File控件选择文件上传,那么File控件完全没有存在的必要。联系上文中我提到的File控件的地位受到威胁这一观点,我大胆的设想,假如未来的某一项标准中给每个HTMLElement暴露一个选择文件的功能接口,那么拖拽和点选功能将可以集于一个元素之上,到那时File控件的地位恐怕不仅仅是受到威胁,很有可能退出历史舞台!出于File控件视觉效果和交互不统一的角度去考虑,我觉得以上推测还是有可能的,哈哈~~

虽然示例并未在后端做太多工作,我这里还是以PHP为例,说明一下后端该如何工作。单从示例而言,我的代码是这样的:

PHP

$file = $_FILES['file']; echo json_encode($file);

1
2
$file = $_FILES['file'];
echo json_encode($file);

可以说是非常简单了。而我们在实际应用中往往还会涉及更多更复杂的处理逻辑。最起码的我们应该要将tmp_name对应的临时文件移动到我们指定的上传目录吧。当然,这一过程我们就会对文件类型进行判断,大小限制,重命名,数据保存,等等。基本代码:

PHP

$file = $_FILES['file']; $path = './upload'; if ($file['size'] > 2000000) { echo '{"error": "1000", "message": "上传文件大小超限,不能超过xxM"}'; } $path .= '/file_' . time() . '.png'; // 这里还可能会存在文件数据保存,新旧名称关联等逻辑 move_uploaded_file($file['tmp_name'], $path);

1
2
3
4
5
6
7
8
$file = $_FILES['file'];
$path = './upload';
if ($file['size'] > 2000000) {
    echo '{"error": "1000", "message": "上传文件大小超限,不能超过xxM"}';
}
$path .= '/file_' . time() . '.png';
// 这里还可能会存在文件数据保存,新旧名称关联等逻辑
move_uploaded_file($file['tmp_name'], $path);

具体实现

一个神奇的方法sendAsBinary

前面我们聊到的使用FormData来实现文件异步上传,在高级浏览器中都能正常运作,没有太大问题。接下来我们另外一个在Firefox实现异步上传的方法。这个方法,我们又会交到一个新的朋友——FireReader。FileReader是HTML5新增的一个对象,它可以访问用户本地文件,并且可以以不同格式读取文件内容。

第一步:通过iframe打开App

Android平台则各个app厂商差异很大,比如Chrome从25及以后就不再支持通过js触发(非用户点击),所以这里使用iframe src地址等来触发scheme。

//在iframe 中打开APP var ifr = document.createElement('iframe'); ifr.src = openUrl; ifr.style.display = 'none';

1
2
3
4
    //在iframe 中打开APP
    var ifr = document.createElement('iframe');
    ifr.src = openUrl;
    ifr.style.display = 'none';

FileReader基本使用

首先我们来看一下如何创建一个FileReader实例对象,以及它拥有哪些实例方法。在js中创建一个FileReader对象很简单:

JavaScript

var reader = new FileReader();

1
var reader = new FileReader();

我们可以通过reader对象访问本地文件,那么reader对象拥有哪些我们常用的属性、事件和方法呢?请看以下列表:

本文由澳门新葡亰平台官网发布于web前端,转载请注明出处:拖拽异步上传实现,自定义标签在IE6

TAG标签:
Ctrl+D 将本页面保存为书签,全面了解最新资讯,方便快捷。