使用jQuery和HTML导出为CSV

78

我有一个表格数据需要导出为csv格式,但不能使用任何外部插件或API。我使用了window.open方法传递MIME类型,但遇到以下问题:

如何使用jQuery确定系统上安装的是Microsoft Excel还是OpenOffice

代码应该独立于系统上安装的软件,即OpenOffice还是Microsoft Excel。 我相信CSV是可以在两个编辑器中显示的格式。

代码

    <html>

<head>
<script src="http://ajax.googleapis.com/ajax/libs/jquery/1.9.1/jquery.min.js"></script>  

<script type="text/JavaScript">
$(document).ready(function(){
    $("#btnExport").click(function(e) {
        var msg = GetMimeTypes();
        //OpenOffice
        window.open('data:application/vnd.oasis.opendocument.spreadsheet,' + $('#dvData').html());
        //MS-Excel
        window.open('data:application/vnd.ms-excel,' + $('#dvData').html());
        //CSV
        window.open('data:application/csv,charset=utf-8,' + $('#dvData').html());
        e.preventDefault();
    });
});

function GetMimeTypes () {
    var message = "";
        // Internet Explorer supports the mimeTypes collection, but it is always empty
    if (navigator.mimeTypes && navigator.mimeTypes.length > 0) {
        var mimes = navigator.mimeTypes;
        for (var i=0; i < mimes.length; i++) {
            message += "<b>" + mimes[i].type + "</b> : " + mimes[i].description + "<br />";
        }
    }
    else {
        message = "Your browser does not support this ";
       //sorry!
    }

    return ( message);
}
</script>

</head>
<body>
<div id="dvData">
<table>
    <tr>
        <th>Column One </th>
        <th>Column Two</th>
        <th>Column Three</th>
    </tr>
    <tr>
        <td>row1 Col1</td>
        <td>row1 Col2</td>
        <td>row1 Col3</td>
   </tr>
   <tr>
        <td>row2 Col1</td>
        <td>row2 Col2</td>
        <td>row2 Col3</td>
   </tr>
      <tr>
        <td>row3 Col1</td>
        <td>row3 Col2</td>
        <td>row3 Col3</td>  
   </tr>
</table>
</div>
<br/>
<input type="button" id="btnExport" value=" Export Table data into Excel " />

</body>

错误

CSV:在浏览器中无法识别。

ODS和Excel:可用,但当系统安装有Excel或OpenOffice时,我无法确定要生成哪个文件?

IE 8版本:完全不工作,会在新窗口中打开,如下图所示。

enter image description here


你具体遇到了什么问题?我不明白。 - Pekka
我无法使用jQuery和HTML将其导出为CSV...我不想使用任何插件...我使用了MIME类型,但似乎没有起作用。 - GOK
然后展示一些代码并描述具体哪些部分无法正常工作?你收到了什么错误信息?等等。 - Pekka
我已经发布了一个问题,可能是你期望的答案。希望这可以帮到你。 http://stackoverflow.com/questions/19539274/export-html-table-to-excel-in-java-script - Prabakaran G
可能是在jQuery中导出为CSV的重复问题。 - Italo Borssatto
显示剩余6条评论
6个回答

133

演示

请参见以下说明。

$(document).ready(function() {

  function exportTableToCSV($table, filename) {

    var $rows = $table.find('tr:has(td)'),

      // Temporary delimiter characters unlikely to be typed by keyboard
      // This is to avoid accidentally splitting the actual contents
      tmpColDelim = String.fromCharCode(11), // vertical tab character
      tmpRowDelim = String.fromCharCode(0), // null character

      // actual delimiter characters for CSV format
      colDelim = '","',
      rowDelim = '"\r\n"',

      // Grab text from table into CSV formatted string
      csv = '"' + $rows.map(function(i, row) {
        var $row = $(row),
          $cols = $row.find('td');

        return $cols.map(function(j, col) {
          var $col = $(col),
            text = $col.text();

          return text.replace(/"/g, '""'); // escape double quotes

        }).get().join(tmpColDelim);

      }).get().join(tmpRowDelim)
      .split(tmpRowDelim).join(rowDelim)
      .split(tmpColDelim).join(colDelim) + '"';

    // Deliberate 'false', see comment below
    if (false && window.navigator.msSaveBlob) {

      var blob = new Blob([decodeURIComponent(csv)], {
        type: 'text/csv;charset=utf8'
      });

      // Crashes in IE 10, IE 11 and Microsoft Edge
      // See MS Edge Issue #10396033
      // Hence, the deliberate 'false'
      // This is here just for completeness
      // Remove the 'false' at your own risk
      window.navigator.msSaveBlob(blob, filename);

    } else if (window.Blob && window.URL) {
      // HTML5 Blob        
      var blob = new Blob([csv], {
        type: 'text/csv;charset=utf-8'
      });
      var csvUrl = URL.createObjectURL(blob);

      $(this)
        .attr({
          'download': filename,
          'href': csvUrl
        });
    } else {
      // Data URI
      var csvData = 'data:application/csv;charset=utf-8,' + encodeURIComponent(csv);

      $(this)
        .attr({
          'download': filename,
          'href': csvData,
          'target': '_blank'
        });
    }
  }

  // This must be a hyperlink
  $(".export").on('click', function(event) {
    // CSV
    var args = [$('#dvData>table'), 'export.csv'];

    exportTableToCSV.apply(this, args);

    // If CSV, don't do event.preventDefault() or return false
    // We actually need this to be a typical hyperlink
  });
});
a.export,
a.export:visited {
  display: inline-block;
  text-decoration: none;
  color: #000;
  background-color: #ddd;
  border: 1px solid #ccc;
  padding: 8px;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<a href="#" class="export">Export Table data into Excel</a>
<div id="dvData">
  <table>
    <tr>
      <th>Column One</th>
      <th>Column Two</th>
      <th>Column Three</th>
    </tr>
    <tr>
      <td>row1 Col1</td>
      <td>row1 Col2</td>
      <td>row1 Col3</td>
    </tr>
    <tr>
      <td>row2 Col1</td>
      <td>row2 Col2</td>
      <td>row2 Col3</td>
    </tr>
    <tr>
      <td>row3 Col1</td>
      <td>row3 Col2</td>
      <td>row3 Col3</td>
    </tr>
    <tr>
      <td>row4 'Col1'</td>
      <td>row4 'Col2'</td>
      <td>row4 'Col3'</td>
    </tr>
    <tr>
      <td>row5 &quot;Col1&quot;</td>
      <td>row5 &quot;Col2&quot;</td>
      <td>row5 &quot;Col3&quot;</td>
    </tr>
    <tr>
      <td>row6 "Col1"</td>
      <td>row6 "Col2"</td>
      <td>row6 "Col3"</td>
    </tr>
  </table>
</div>


截至2017年

现在使用HTML5 BlobURL 作为首选方法,Data URI 作为备用方法。

在Internet Explorer上

其他答案建议使用window.navigator.msSaveBlob;然而,已知它会导致IE10 / Window 7和IE11 / Windows 10崩溃。它是否在Microsoft Edge中可行是值得怀疑的(参见Microsoft Edge问题票号#10396033)。

仅仅在Microsoft自己的开发人员工具/控制台中调用此功能就会导致浏览器崩溃:

navigator.msSaveBlob(new Blob(["hello"], {type: "text/plain"}), "test.txt");

距离我第一次回答已经过去了四年,新的IE版本包括IE10、IE11和Edge。它们都会在一个由Microsoft发明的函数上崩溃(慢速鼓掌)。

自行决定是否添加navigator.msSaveBlob支持,风险自负。


截至2013年

通常情况下,这将使用服务器端解决方案执行,但这是我尝试的客户端解决方案。简单地将HTML作为Data URI转储将无法工作,但这是一步有用的。所以:

  1. 将表格内容转换为有效的CSV格式字符串。(这是容易的部分。)
  2. 强制浏览器下载它。 window.open方法在Firefox中不起作用,因此我使用了<a href="{Data URI here}">
  3. 使用<a>标记的download属性分配默认文件名,该属性仅在Firefox和Google Chrome中起作用。由于它只是一个属性,它会优雅地降级。

笔记

兼容性

测试浏览器包括:

  • Firefox 20+,Win/Mac(适用
  • Google Chrome 26+,Win/Mac(适用
  • Safari 6,Mac(适用,但文件名会被忽略)
  • IE 9+(失败

内容编码

CSV导出正确,但在Excel导入时,字符ü打印为ä。Excel将该值解释错误。

引入var csv = '\ufeff';,然后Excel 2013+正确解释值。

如果需要与Excel 2007兼容,请在每个数据值处添加UTF-8前缀。也请参见:


它对我的要求非常有效,只是for循环不考虑表格中的<th>或表头。这是故意为之的吗?如果有两个带有colspan的表头,我们能使用这个功能来处理吗? - GOK
哎呀..但是有没有办法修复它?我正在IE8上检查它。 - GOK
19
快速说明:如需包括<th>,请将'tr:has(td)'替换为'tr:has(td),tr:has(th)',并将稍后的'td'替换为'td,th'。 - Wouter
4
包括以下表头(根据@Wouter的建议实现):
  1. var $rows = $table.find('tr:has(td),tr:has(th)'),
  2. $cols = $row.find('td,th');
- Bala
关于 text = $col.text(); 的问题,你如何获取使用 css:before 生成的文本?我得到了空值。请参考:http://stackoverflow.com/questions/35774973/get-css-generated-text-inside-a-td-with-jquery - Mario
显示剩余6条评论

7

我不确定上面的CSV生成代码是否如此出色,因为它似乎跳过了th单元格,并且似乎没有考虑到值中的逗号。所以这里是我的CSV生成代码,它可能会有用。

它假定你有$table变量,这是一个jQuery对象(例如:var $table = $('#yourtable');

$rows = $table.find('tr');

var csvData = "";

for(var i=0;i<$rows.length;i++){
                var $cells = $($rows[i]).children('th,td'); //header or content cells

                for(var y=0;y<$cells.length;y++){
                    if(y>0){
                      csvData += ",";
                    }
                    var txt = ($($cells[y]).text()).toString().trim();
                    if(txt.indexOf(',')>=0 || txt.indexOf('\"')>=0 || txt.indexOf('\n')>=0){
                        txt = "\"" + txt.replace(/\"/g, "\"\"") + "\"";
                    }
                    csvData += txt;
                }
                csvData += '\n';
 }

2

对于@Terry Young的答案进行微小更新,即添加IE 10+支持。

if (window.navigator.msSaveOrOpenBlob) {
  // IE 10+
  var blob = new Blob([decodeURIComponent(encodeURI(csvString))], {
    type: 'text/csv;charset=' + document.characterSet
  });
  window.navigator.msSaveBlob(blob, filename);
} else {
  // actual real browsers
  //Data URI
  csvData = 'data:application/csv;charset=utf-8,' + encodeURIComponent(csvData);

    $(this).attr({
      'download': filename,
      'href': csvData,
      'target': '_blank'
    });
}

这个方法适用于IE9将HTML表格导出到Excel,例如 type: 'data:application/vnd.ms-excel;' 吗?谢谢。 - Disera

2
 <a id="export" role='button'>
        Click Here To Download Below Report
    </a>
    <table id="testbed_results" style="table-layout:fixed">
        <thead>
            <tr width="100%" style="color:white" bgcolor="#3195A9" id="tblHeader">
                <th>Name</th>
                <th>Date</th>
                <th>Speed</th>
                <th>Column2</th>
                <th>Interface</th>
                <th>Interface2</th>
                <th>Sub</th>
                <th>COmpany result</th>
                <th>company2</th>
                <th>Gen</th>
            </tr>
        </thead>
        <tbody>
            <tr id="samplerow">
                <td>hello</td>
                <td>100</td>
                <td>200</td>
                <td>300</td>
                <td>html2svc</td>
                <td>ajax</td>
                <td>200</td>
                <td>7</td>
                <td>8</td>
                <td>9</td>
            </tr>
            <tr>
                <td>hello</td>
                <td>100</td>
                <td>200</td>
                <td>300</td>
                <td>html2svc</td>
                <td>ajax</td>
                <td>200</td>
                <td>7</td>
                <td>8</td>
                <td>9</td>
            </tr>
        </tbody>
    </table>

    $(document).ready(function () {
        Html2CSV('testbed_results', 'myfilename','export');
    });



    function Html2CSV(tableId, filename,alinkButtonId) {
        var array = [];
        var headers = [];
        var arrayItem = [];
        var csvData = new Array();
        $('#' + tableId + ' th').each(function (index, item) {
            headers[index] = '"' + $(item).html() + '"';
        });
        csvData.push(headers);
        $('#' + tableId + ' tr').has('td').each(function () {

            $('td', $(this)).each(function (index, item) {
                arrayItem[index] = '"' + $(item).html() + '"';
            });
            array.push(arrayItem);
            csvData.push(arrayItem);
        });




        var fileName = filename + '.csv';
        var buffer = csvData.join("\n");
        var blob = new Blob([buffer], {
            "type": "text/csv;charset=utf8;"
        });
        var link = document.getElementById(alinkButton);

        if (link.download !== undefined) { // feature detection
            // Browsers that support HTML5 download attribute
            link.setAttribute("href", window.URL.createObjectURL(blob));
            link.setAttribute("download", fileName);
        }
        else if (navigator.msSaveBlob) { // IE 10+
            link.setAttribute("href", "#");
            link.addEventListener("click", function (event) {
                navigator.msSaveBlob(blob, fileName);
            }, false);
        }
        else {
            // it needs to implement server side export
            link.setAttribute("href", "http://www.example.com/export");
        }
    }

</script>

2
你应该添加一些文本,说明你的代码是如何解决这个问题的,以及为什么会这样做。 - Reeno

1
如果您的数据是以CSV格式存在的,并且想将其转换为HTML以在网页上显示,您可以使用http://code.google.com/p/js-tables/插件。请查看这个示例http://code.google.com/p/js-tables/wiki/Table。由于您已经在使用jQuery库,我假设您能够添加其他JavaScript工具库。
如果数据是以CSV格式存在的,您应该可以使用通用的“application/octetstream”MIME类型。所有三种您尝试过的MIME类型都依赖于客户端计算机上安装的软件。

1
据我所了解,您的数据存储在一个表格中,您想从这些数据创建CSV文件。然而,您在创建CSV文件时遇到了问题。
我的想法是迭代并解析表格的内容,并从解析后的数据生成字符串。您可以查看如何将JSON转换为CSV格式并存储在变量中以获取示例。您在示例中使用jQuery,因此不算作外部插件。然后,您只需使用window.open提供结果字符串,并使用建议的application/octetstream即可。

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接