我正在尝试提供一个仅限脚本的解决方案,用于通过浏览器读取客户端计算机上文件的内容。
我有一个适用于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,但我愿意接受任何其他浏览器的建议。
修改:回答问题“你为什么要这样做?”:
基本上,我想在客户端将文件内容与一次性密码一起散列,以便我可以将此信息作为验证发回。
答案 0 :(得分:115)
已编辑以添加有关文件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对象的唯一键是fileName
和fileSize
。根据文件和文件列表支持的commit message,这些受Mozilla's File object的启发,但它们似乎只支持一部分功能。
如果您想更改此设置,您可以始终send a patch到WebKit项目。另一种可能性是建议将Mozilla API包含在HTML 5中; WHATWG邮件列表可能是最好的地方。如果你这样做,那么很可能会有一种跨浏览器的方式来做到这一点,至少在几年后。当然,提交补丁或提案以包含在HTML 5中确实意味着一些工作可以捍卫这个想法,但Firefox已经实现它的事实为您提供了一些开始。
答案 1 :(得分: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 :(得分:0)
有一个现代的本地替代方法: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。
答案 3 :(得分:0)
这很好用
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);
}