在各种浏览器中使用javascript在客户端读取文件内容

问题描述 投票:81回答:3

我正在尝试提供一个仅限脚本的解决方案,用于通过浏览器读取客户端计算机上的文件内容。

我有一个适用于Firefox和Internet Explorer的解决方案。它不漂亮,但我现在只是在尝试:

function getFileContents() {
    var fileForUpload = document.forms[0].fileForUpload;
    var fileName = fileForUpload.value;

    if (fileForUpload.files) {
        var fileContents = fileForUpload.files.item(0).getAsBinary();
        document.forms[0].fileContents.innerHTML = fileContents;
    } else {
        // try the IE method
        var fileContents = ieReadFile(fileName);
        document.forms[0].fileContents.innerHTML = fileContents;
    }
}       

function ieReadFile(filename) 
{
    try
    {
        var fso  = new ActiveXObject("Scripting.FileSystemObject"); 
        var fh = fso.OpenTextFile(filename, 1); 
        var contents = fh.ReadAll(); 
        fh.Close();
        return contents;
    }
    catch (Exception)
    {
        return "Cannot open file :(";
    }
}

我可以调用getFileContents(),它会将内容写入fileContents文本区域。

有没有办法在其他浏览器中执行此操作?

我目前最关心的是Safari和Chrome,但我愿意接受任何其他浏览器的建议。

编辑:回答问题,“你为什么要这样做?”:

基本上,我想在客户端将文件内容与一次性密码一起散列,以便我可以将此信息作为验证发回。

javascript html file-io sandbox
3个回答
122
投票

已编辑以添加有关File API的信息

自从我最初写这个答案以来,File API已被提议作为标准和implemented in most browsers(从IE 10开始,它增加了对这里描述的FileReader API的支持,虽然还不是File API)。 API比旧的Mozilla API复杂一点,因为它旨在支持异步读取文件,更好地支持二进制文件和解码不同的文本编码。有some documentation available on the Mozilla Developer Network以及各种在线示例。您可以按如下方式使用它:

var file = document.getElementById("fileForUpload").files[0];
if (file) {
    var reader = new FileReader();
    reader.readAsText(file, "UTF-8");
    reader.onload = function (evt) {
        document.getElementById("fileContents").innerHTML = evt.target.result;
    }
    reader.onerror = function (evt) {
        document.getElementById("fileContents").innerHTML = "error reading file";
    }
}

原始答案

似乎没有办法在WebKit中执行此操作(因此,Safari和Chrome)。 File对象唯一的关键是fileNamefileSize。根据文件和文件列表支持的commit message,这些受到Mozilla's File object的启发,但它们似乎只支持一部分功能。

如果你想改变它,你可以随时send a patch到WebKit项目。另一种可能性是建议将Mozilla API包含在HTML 5中; WHATWG邮件列表可能是最好的地方。如果你这样做,那么很可能会有一种跨浏览器的方式来做到这一点,至少在几年后。当然,提交补丁或提案以包含在HTML 5中确实意味着一些工作可以捍卫这个想法,但Firefox已经实现它的事实为您提供了一些开始。


17
投票

要使用文件打开对话框读取用户选择的文件,可以使用<input type="file">标记。你可以找到information on it from MSDN。选择文件后,您可以使用FileReader API读取内容。

function onFileLoad(elementId, event) {
    document.getElementById(elementId).innerText = event.target.result;
}

function onChooseFile(event, onLoadFileHandler) {
    if (typeof window.FileReader !== 'function')
        throw ("The file API isn't supported on this browser.");
    let input = event.target;
    if (!input)
        throw ("The browser does not properly implement the event object");
    if (!input.files)
        throw ("This browser does not support the `files` property of the file input.");
    if (!input.files[0])
        return undefined;
    let file = input.files[0];
    let fr = new FileReader();
    fr.onload = onLoadFileHandler;
    fr.readAsText(file);
}
<input type='file' onchange='onChooseFile(event, onFileLoad.bind(this, "contents"))' />
<p id="contents"></p>

2
投票

快乐的编码! 如果在Internet Explorer上出现错误,请更改安全设置以允许ActiveX

var CallBackFunction = function(content)
{
    alert(content);
}
ReadFileAllBrowsers(document.getElementById("file_upload"), CallBackFunction); 

//Tested in Mozilla Firefox browser, Chrome
function ReadFileAllBrowsers(FileElement, CallBackFunction)
{
try
{
    var file = FileElement.files[0];
    var contents_ = "";

     if (file) {
        var reader = new FileReader();
        reader.readAsText(file, "UTF-8");
        reader.onload = function(evt)
        {
            CallBackFunction(evt.target.result);
        }
        reader.onerror = function (evt) {
            alert("Error reading file");
        }
    }
}
catch (Exception)
 {
    var fall_back =  ieReadFile(FileElement.value);
    if(fall_back != false)
    {
        CallBackFunction(fall_back);
    }
 }
}

///Reading files with Internet Explorer
function ieReadFile(filename)
{
 try
 {
    var fso  = new ActiveXObject("Scripting.FileSystemObject");
    var fh = fso.OpenTextFile(filename, 1);
    var contents = fh.ReadAll();
    fh.Close();
    return contents;
 }
 catch (Exception)
  {
    alert(Exception);
    return false;
  }
 }
© www.soinside.com 2019 - 2024. All rights reserved.