31

背景:

我想制作一个使用 JavaScript/HTML并且可以由浏览器直接从文件系统打开的“应用程序”。此应用程序必须能够从另一个文件中读取数据。然后,我将使用 JS 对其进行解析并呈现页面。作为一个简化的示例,假设我有一个 CSV 文件(在此处下载)

Mark Rodgers,mark.rodgers@company.com,Accounting
[...]
Melissa Jones,melissa@company.com,CEO

我希望能够使用 JS 读取文件并使用其中的数据来生成我的页面。

到目前为止我已经完成了什么:

演示(右键单击->“另存为”将 HTML 保存到您的计算机)。它也可以在jsfiddle上以半损坏的方式使用(布局已损坏,但在功能上应该仍然正确)。

只需将 CSV 文本文件拖放到拖放框中,或使用文件菜单选择文本文件,JavaScript 就会读取、解析文件并填充表格。

这依赖于 FileReader API;大部分繁重的工作都是由这个函数完成的:

function handleFileSelect(evt) {
    evt.stopPropagation();
    evt.preventDefault();

    var files = evt.target.files || evt.dataTransfer.files; // FileList object.
    var file = files[0];

    // this creates the FileReader and reads stuff as text
    var fr = new FileReader();
    fr.onload = parse;
    fr.readAsText(file);

    // this is the function that actually parses the file
    // and populates the table
    function parse()
    {
        var table = document.getElementById('emps');
        var employees = fr.result.split('\n'); var c = 0;
        for (var i in employees)
        {
            var employee = employees[i].split(',');
            if (employee.length == 3)
            {
                var row = document.createElement('tr');
                row.innerHTML = "<td>" + employee.join("</td><td>") + "</td>";
                table.appendChild(row);
                c++;
            }
        }
        document.getElementById('result').innerHTML = '<span>Added ' + c + ' employees from file: ' + file.name + '</span>';
    }
}

几乎没问题,但它给用户手动加载文件带来了不便。理想情况下,它应该能够自动加载它,但出于安全原因,没有浏览器允许这样做......但是。

解决方案要求:

  • 必须离线工作;即:它不能依赖任何在线服务。这也包括在本地机器上运行的 HTTP 服务器。我们的想法是让它在任何只安装了浏览器的计算机上运行。

  • 必须在使用file:///协议打开页面时工作(即:硬盘驱动器上的 HTML 页面)。

  • 不应依赖第三方插件(例如:Flash、Java、颤抖的ActiveX)。我很确定如果页面在,这些可能无论如何都不起作用file:///

  • 它必须能够接受任意数据。这排除了以行为良好的格式加载文件,该格式可以像 JSON 一样可供使用。

  • 如果它适用于 Firefox 或 Chrome 中的任何一个(理想情况下两者都适用),那就没问题了。依赖实验性 API 也是可以的

我事先知道文件名是什么,所以它可以在 HTML 本身中编码。任何使我能够从磁盘读取文件的解决方案都可以,它不必使用 FileReader API。

因此,如果有一个聪明的技巧可以将文件加载到一个页面中也很好(也许将它加载到一个不可见的 iframe 并让 JS 检索内容);那也没关系。

4

6 回答 6

4

这是我用于 Firefox 的代码,它不可移植,但作品

正如 OP 评论的那样,enablePrivilege()已被弃用,这应该被认为是可用的。但是由于我的 Firefox 使用以前的配置文件仍然可以使用我的代码,所以我深入研究了prefs.jsabout:config隐藏这些设置),这里是你需要的设置,你可以让它工作。

user_pref("capability.principal.codebase.p0.granted", "UniversalXPConnect");
user_pref("capability.principal.codebase.p0.id", "file://");  // path to the html file.
user_pref("capability.principal.codebase.p0.subjectName", "");

代码如下:

var File = function(file) {
  netscape.security.PrivilegeManager.enablePrivilege('UniversalXPConnect');
  var ios = Components.classes["@mozilla.org/network/io-service;1"]
                            .getService(Components.interfaces.nsIIOService);
  if (!File.baseURI) {
    File.baseURI = ios.newURI(location.href.substring(0, location.href.lastIndexOf('/')+1), null, null);
    File.baseFolder = File.baseURI.QueryInterface(Components.interfaces.nsIFileURL).file.path;
  }
  var URL = ios.newURI(file, null, File.baseURI);
  this.fptr = URL.QueryInterface(Components.interfaces.nsIFileURL).file;
}

File.prototype = {
  write: function(data) {
    netscape.security.PrivilegeManager.enablePrivilege('UniversalXPConnect');
    var foStream = Components.classes["@mozilla.org/network/file-output-stream;1"]
                             .createInstance(Components.interfaces.nsIFileOutputStream);
    foStream.init(this.fptr, 0x02 | 0x08 | 0x20, 0666, 0);
    var converter = Components.classes["@mozilla.org/intl/converter-output-stream;1"]
                              .createInstance(Components.interfaces.nsIConverterOutputStream);
    converter.init(foStream, null, 0, 0);
    converter.writeString(data);
    converter.close();
  },
  read: function() {
    netscape.security.PrivilegeManager.enablePrivilege('UniversalXPConnect');
    var fstream = Components.classes["@mozilla.org/network/file-input-stream;1"]
                            .createInstance(Components.interfaces.nsIFileInputStream);
    var cstream = Components.classes["@mozilla.org/intl/converter-input-stream;1"]
                            .createInstance(Components.interfaces.nsIConverterInputStream);
    fstream.init(this.fptr, -1, 0, 0);
    cstream.init(fstream, null, 0, 0);
    var data = "";
    // let (str = {}) { // use this only when using javascript 1.8
    var str = {};
      cstream.readString(0xffffffff, str);
      data = str.value;
    // }
    cstream.close();
    return data;
  }
};
于 2012-11-17T07:16:58.153 回答
4

这是一个在本地或服务器上工作的外部文件中使用 JSON 数据的示例。此示例仅使用浏览器的语言设置加载具有本地化 html 的 <script>,然后处理其 json 对象以使用本地化内容重置指示标记中的数据

<html><meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<head>
<script>
    function setLang(){
        for (var i=0;i<items.length;i++){
            term=document.getElementById(items[i].id)
            if (term) term.innerHTML=items[i].value
        }
    }
    var lang=navigator.userLanguage || navigator.language;
    var script=document.createElement("script");
    script.src=document.URL+"-"+lang.substring(0,2)+".js"
    var head = document.getElementsByTagName('head')[0]
    head.insertBefore(script,head.firstChild)
</script>
</head>
<body onload='setLang()'>
<div id="string1" class="txt">This is the default text of string1.</div> 
<div id="string2" class="txt">This is the default text of string2.</div>
</body></html>

用于此的数据文件如下所示:

items=[
{"id":"string1","value":"Localized text of string1."},
{"id":"string2", "value":"Localized text of string2."}
];

但是您可以使用任何参数来有条件地加载适当的文件(它将作为 <head> 中的第一个标签插入,因此它可以在任何地方使用)并且 JSON 格式能够处理大量数据。您可能希望将函数 setLang 重命名为更合适的名称并对其进行修改以满足您的需求,例如...对于每个我添加一行,然后添加带有数据的字段(看起来您已经掌握了该部分的句柄)你的 JSON 看起来像:

items=[
{"fname":"john","lname":"smith","address":"1 1st St","phone":"555-1212"},
{"fname":"jane","lname":"smith","address":"1 1st St","phone":"555-1212"}
];

如果您需要预处理您的数据,awk 非常方便 - 它类似于:(未经测试的猜测)

awk 'BEGIN{FS=",";print "items=[\n"}
{printf "{\"fname\":\"%s\",\"lname\":\"smith\",\"address\":\"1 1st St\",\"phone\":\"555-1212\"},\n", $1, $2, $3, $4}
END{print "];"}' file.csv > file.js

编辑:现在 OP 更清楚了,只有 mozilla 浏览器允许 XMLHttpRequest on file:// 开箱即用,并且可以配置 chrome(可能是其他基于 webkit 的浏览器)允许它。知道它可能不适用于 IE<10,您可以:

var filePath = "your_file.txt";
xmlhttp = new XMLHttpRequest();
xmlhttp.open("GET",filePath,false);
xmlhttp.overrideMimeType('text/plain');
xmlhttp.send(null);
//maybe check status !=404 here
var fileContent = xmlhttp.responseText;
var fileArray = fileContent.split('\n')
var n = fileArray.length;
//process your data from here probably using split again for ','

我将最初的 json-p 变体留给可能有类似问题但对其数据格式有一定控制权的其他人,因为它适用于所有支持 javascript 的浏览器。但是,如果有人知道使它适用于 IE 的方法(除了运行小型 Web 服务器),请编辑。

编辑2:

使用 mozilla 浏览器,您还可以使用 iframe

<html>
<meta http-equiv="Content-Type" content="text/html; charset=utf-8">
<head>
<script>
function showContents(frameObject){
    alert(frameObject.contentDocument.body.innerHTML);
    //replace with your code
}
</script>
</head>
<body onload='showContents()'>
<iframe id="frametest" src="data.txt" onload="showContents(this);" 
    style="visibility:hidden;display:none"></iframe>
</body></html>
于 2012-11-27T04:51:48.610 回答
3

假设 csv 文件与应用程序位于同一目录中,我将使用 AJAX 加载文件。据我所知,可以获取文本格式的文件,然后对其进行解析。这应该适用于 IE 和 Firefox,但不适用于 Chrome(除非使用--allow-file-access-from-files命令行设置运行 chrome)。

于 2012-12-03T09:10:12.810 回答
2

这可以很容易地使用 javascript XMLHttpRequest() 类完成:

function FileHelper()
{}
{
    FileHelper.readStringFromFileAtPath = function(pathOfFileToReadFrom)
    {
        var request = new XMLHttpRequest();
        request.open("GET", pathOfFileToReadFrom, false);
        request.send(null);
        var returnValue = request.responseText;

        return returnValue;
    }
}

...

var text = FileHelper.readStringFromFileAtPath ( "mytext.txt" );
于 2014-03-03T16:22:02.850 回答
1

据我了解,文件的内容完全在您的控制之下,它不一定是特定的格式?你只需要一种阅读方式吗?

您可以声明一个全局函数“handleFile”。在您的外部文件中,内容必须是这样的:

handleFile('Mark Rodgers,mark.rodgers@company.com,Accounting');

要“读取”文件,只需添加一个具有相应 src 属性的脚本元素。在你的函数“handleFile”中,你得到你的内容。

文件的位置可能最初必须由用户设置,但之后您可以将位置保存在 localStorage 或类似的东西中。

于 2012-11-26T14:15:12.157 回答
1

确保文件在同一目录或子目录中,使用 AJAX 加载文件。

与脚本标签不同,您将可以访问内容。

于 2012-11-27T06:51:56.297 回答