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

IT技术 javascript html file-io sandbox
2021-01-12 22:55:10

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

我有一个适用于 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,但我愿意接受任何其他浏览器的建议。

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

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

5个回答

编辑以添加有关文件 API 的信息

自从我最初写这个答案以来,文件 API已被提议作为标准并在大多数浏览器中实现(从 IE 10 开始,它增加了FileReader对此处描述API 的支持,尽管还不是FileAPI)。该 API 比旧的 Mozilla API 稍微复杂一些,因为它旨在支持文件的异步读取、更好地支持二进制文件和不同文本编码的解码。Mozilla 开发人员网络上一些文档以及各种在线示例。您可以按如下方式使用它:

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根据File 和 FileList 支持提交消息,这些受到Mozilla 的 File object 的启发,但它们似乎只支持功能的一个子集。

如果您想更改此设置,您可以随时WebKit 项目发送补丁另一种可能性是提议将 Mozilla API 包含在HTML 5 中WHATWG的邮件列表是可能做到这一点的最好的地方。如果你这样做了,那么至少在几年内会有一种跨浏览器的方式来做到这一点的可能性更大。当然,提交一个补丁或一个包含到 HTML 5 的提案确实意味着需要为这个想法辩护,但事实上 Firefox 已经实现了它给你一些开始。

嗯,实际上这是一个公平的观点。有用户交互来选择该文件。谢谢。
2021-03-13 22:55:10
@Damovisa 我不知道你是否仍然关心这个,但我想我会更新我的答案,提到新的 File API,它可以满足你的需求,并在 Firefox、Chrome 和夜间构建中实现苹果浏览器。
2021-03-13 22:55:10
感谢您 - 我认为此时我还不够专注于提交补丁。无论如何,这是您可能不希望在您不知情的情况下发生的事情。它有点打破了浏览器沙箱......
2021-03-17 22:55:10
它不会破坏浏览器沙箱,因为您故意选择上传该文件;如果它可以到达服务器,它可以返回到浏览器,只需额外的往返。鉴于使网络应用程序脱机模式工作的工作,这将是一个合理的功能。
2021-03-31 22:55:10
不应该在调用之前附加事件处理程序readAsText吗?
2021-04-08 22:55:10

为了读取用户选择的文件,使用文件打开对话框,您可以使用<input type="file">标签。您可以从 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>

有一个现代的本地替代方案:File实现Blob,所以我们可以调用Blob.text()

async function readText(event) {
  const file = event.target.files.item(0)
  const text = await file.text();
  
  document.getElementById("output").innerText = text
}
<input type="file" onchange="readText(event)" />
<pre id="output"></pre>

目前(2020 年 9 月)Chrome 和 Firefox 支持此功能,对于其他浏览器,您需要加载 polyfill,例如blob-polyfill

快乐编码!
如果您在 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;
  }
}

这工作正常

function onClick(event) {
    filecontent = "";
    var myFile = event.files[0];
    var reader = new FileReader();

    reader.addEventListener('load', function (e) {
      filecontent = e.target.result;

    });
    reader.readAsBinaryString(myFile);
  }