Emscripten:提供下载/保存生成的MEMFS文件

Emscripten: Offer to download/save a generated MEMFS file

本文关键字:MEMFS 文件 保存 下载 Emscripten      更新时间:2023-10-16

我玩emscripten是为了好玩,并编译了一个旧的模拟器程序在浏览器中运行。

该程序将输出保存为数据文件,由于我没有做任何特别的事情,它应该驻留在MEMFS(内存文件系统)上。

如何允许用户从 MEMFS 下载生成的文件?我可以触发浏览器的常规保存文件对话框吗?

知道了:

  • 编译期间将 FS 添加到EXTRA_EXPORTED_RUNTIME_METHODS。
$ emcc --bind -std=c++11 
       -s EXTRA_EXPORTED_RUNTIME_METHODS=FS 
       -o main.js *.cpp 
  • 将以下函数添加到.html或.js文件中。
function offerFileAsDownload(filename, mime) {
  mime = mime || "application/octet-stream";
  let content = Module.FS.readFile(filename);
  console.log(`Offering download of "${filename}", with ${content.length} bytes...`);
  var a = document.createElement('a');
  a.download = filename;
  a.href = URL.createObjectURL(new Blob([content], {type: mime}));
  a.style.display = 'none';
  document.body.appendChild(a);
  a.click();
  setTimeout(() => {
    document.body.removeChild(a);
    URL.revokeObjectURL(a.href);
  }, 2000);
}
  • 在 postRun 上调用函数(选项 1):
<script>
// This must go BEFORE sourcing main.js:
var Module = {
  onRuntimeInitialized: () => {
    Module.postRun.push(() => offerFileAsDownload("filename.ext", "mime/type"));
  }
};
</script>
<script src="main.js"></script>
  • 在C++代码末尾调用函数(选项 2):
int main() {
  // ...
  emscripten::val::global("window").call<void>(
    "offerFileAsDownload",
    string("filename.ext"),
    string("mime/type")
  );
  // ...
}

我发布了一个小的仅标头C++库,允许在Emscripten中下载和上传文件:

https://github.com/Armchair-Software/emscripten-browser-file

这比其他答案简单得多,只需使用 EM_JS - 以下示例来自上面:

EM_JS(void, download, (char const *filename, char const *mime_type, void const *buffer, size_t buffer_size), {
  /// Offer a buffer in memory as a file to download, specifying download filename and mime type
  var a = document.createElement('a');
  a.download = UTF8ToString(filename);
  a.href = URL.createObjectURL(new Blob([new Uint8Array(Module.HEAPU8.buffer, buffer, buffer_size)], {type: UTF8ToString(mime_type)}));
  a.click();
});

这适用于任何内存缓冲区 - 如果要专门引用 MEMFS,可以使用 Module.FS.readFile(filename)