From 0a733139e3d1f706db35bc6e1cf5ac2e914a89ab Mon Sep 17 00:00:00 2001 From: Felix Chantrel Date: Thu, 21 Nov 2019 11:25:01 +1030 Subject: [PATCH 1/4] Added minimal browser support. --- FileOpener2.js | 125 +++++++++++++++++++++++++++++++++++++++++++++++ FileSaver.min.js | 3 ++ isChrome.js | 26 ++++++++++ plugin.xml | 23 +++++++++ 4 files changed, 177 insertions(+) create mode 100644 FileOpener2.js create mode 100644 FileSaver.min.js create mode 100644 isChrome.js diff --git a/FileOpener2.js b/FileOpener2.js new file mode 100644 index 00000000..c73096e5 --- /dev/null +++ b/FileOpener2.js @@ -0,0 +1,125 @@ +/* +The MIT License (MIT) + +Copyright (c) 2019 fefc - fefc.dev@gmail.com + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. +*/ + +const cacheDirectory = (require('./isChrome')()) ? 'filesystem:' + window.location.origin + '/temporary/' : 'file:///temporary/'; +const dataDirectory = (require('./isChrome')()) ? 'filesystem:' + window.location.origin + '/persistent/' : 'file:///persistent/'; + +function open(successCallback, errorCallback, data) { + var fullFilePath = data[0]; + //var contentType = data[1]; //Not needed in browser + //var openDialog = data[2]; //Not needed in browser + + var dirPath = fullFilePath.substring(0, fullFilePath.lastIndexOf('/') + 1); + var fileName = fullFilePath.substring(fullFilePath.lastIndexOf('/') + 1, fullFilePath.length); + var fileSystemLocalPath = getLocalPathAndFileSystem(dirPath); + + if (!fileSystemLocalPath.error) { + window.requestFileSystem(fileSystemLocalPath.fileSystem, 0, (fs) => { + readFile(fs.root, fileSystemLocalPath.localPath + fileName).then((blob) => { + FileSaver.saveAs(blob, fileName); + successCallback(); + }).catch((error) => { + errorCallback(error); + }); + }, (error) => { + errorCallback(error); + }); + } else { + errorCallback('INVALID_PATH'); + } +} + +/** + * + * Gets the localPath according to the fileSystem (TEMPORARY or PERSISTENT). + * + * @param {String} Path to the file or directory to check + * @returns {Object} value with informations to requestFileSystem later + * @returns {string} value.localPath The localPath in relation with fileSystem. + * @returns {number} value.fileSystem the fileSystem (TEMPORARY or PERSISTENT). + * @returns {error} value.error if the path is not valid. + * @returns {message} value.message error message. + */ +function getLocalPathAndFileSystem(pathToCheck) { + let ret = { + localPath: '', + fileSystem: window.TEMPORARY + }; + + if (pathToCheck.startsWith(cacheDirectory)) { + ret.localPath = pathToCheck.replace(cacheDirectory, ''); + ret.fileSystem = window.TEMPORARY; + + } else if (pathToCheck.startsWith(dataDirectory)) { + ret.localPath = pathToCheck.replace(dataDirectory, ''); + ret.fileSystem = window.PERSISTENT; + + } else { + return {error: true, message: 'INVALID_PATH'}; + } + + if (!ret.localPath.endsWith('/')) ret.localPath += '/'; + + return ret; +} + +/** + * + * Reads a file in the fileSystem as an DataURL. + * + * @param {String} Root is the root folder of the fileSystem. + * @param {String} Path is the file to be red. + * @returns {Promise} which resolves with an Object containing DataURL, rejects if something went wrong. + */ +function readFile(root, filePath) { + return new Promise((resolve, reject) => { + if (filePath.startsWith('/')) filePath = filePath.substring(1); + + root.getFile(filePath, {}, (fileEntry) => { + fileEntry.file((file) => { + let reader = new FileReader(); + + reader.onload = function() { + resolve(reader.result); + }; + + reader.onerror = function() { + reject(reader.error); + } + + reader.readAsDataURL(file); + + }, (error) => { + reject(error); + }); + }, (error) => { + reject(error); + }); + }); +} + +module.exports = { + open: open +}; + +require( "cordova/exec/proxy" ).add( "FileOpener2", module.exports ); diff --git a/FileSaver.min.js b/FileSaver.min.js new file mode 100644 index 00000000..183d42a1 --- /dev/null +++ b/FileSaver.min.js @@ -0,0 +1,3 @@ +(function(a,b){if("function"==typeof define&&define.amd)define([],b);else if("undefined"!=typeof exports)b();else{b(),a.FileSaver={exports:{}}.exports}})(this,function(){"use strict";function b(a,b){return"undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Deprecated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(b,c,d){var e=new XMLHttpRequest;e.open("GET",b),e.responseType="blob",e.onload=function(){a(e.response,c,d)},e.onerror=function(){console.error("could not download file")},e.send()}function d(a){var b=new XMLHttpRequest;b.open("HEAD",a,!1);try{b.send()}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"))}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b)}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof global&&global.global===global?global:void 0,a=f.saveAs||("object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href)},4E4),setTimeout(function(){e(j)},0))}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else{var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i)})}}:function(a,b,d,e){if(e=e||open("","_blank"),e&&(e.document.title=e.document.body.innerText="downloading..."),"string"==typeof a)return c(a,b,d);var g="application/octet-stream"===a.type,h=/constructor/i.test(f.HTMLElement)||f.safari,i=/CriOS\/[\d]+/.test(navigator.userAgent);if((i||g&&h)&&"undefined"!=typeof FileReader){var j=new FileReader;j.onloadend=function(){var a=j.result;a=i?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),e?e.location.href=a:location=a,e=null},j.readAsDataURL(a)}else{var k=f.URL||f.webkitURL,l=k.createObjectURL(a);e?e.location=l:location.href=l,e=null,setTimeout(function(){k.revokeObjectURL(l)},4E4)}});f.saveAs=a.saveAs=a,"undefined"!=typeof module&&(module.exports=a)}); + +//# sourceMappingURL=FileSaver.min.js.map \ No newline at end of file diff --git a/isChrome.js b/isChrome.js new file mode 100644 index 00000000..853277d7 --- /dev/null +++ b/isChrome.js @@ -0,0 +1,26 @@ +/* + * + * Licensed to the Apache Software Foundation (ASF) under one + * or more contributor license agreements. See the NOTICE file + * distributed with this work for additional information + * regarding copyright ownership. The ASF licenses this file + * to you under the Apache License, Version 2.0 (the + * "License"); you may not use this file except in compliance + * with the License. You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, + * software distributed under the License is distributed on an + * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY + * KIND, either express or implied. See the License for the + * specific language governing permissions and limitations + * under the License. + * + */ + +module.exports = function () { + // window.webkitRequestFileSystem and window.webkitResolveLocalFileSystemURL are available only in Chrome and + // possibly a good flag to indicate that we're running in Chrome + return window.webkitRequestFileSystem && window.webkitResolveLocalFileSystemURL; +}; diff --git a/plugin.xml b/plugin.xml index b8da6d3f..96e0d6e4 100644 --- a/plugin.xml +++ b/plugin.xml @@ -62,4 +62,27 @@ + + + + + + + + + + + + + + + + + + + + + + From 179aa877c932573e293e703a6d31379cbee10663 Mon Sep 17 00:00:00 2001 From: Felix Chantrel Date: Thu, 21 Nov 2019 11:25:44 +1030 Subject: [PATCH 2/4] Rename isChrome.js to www/browser/isChrome.js --- isChrome.js => www/browser/isChrome.js | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename isChrome.js => www/browser/isChrome.js (100%) diff --git a/isChrome.js b/www/browser/isChrome.js similarity index 100% rename from isChrome.js rename to www/browser/isChrome.js From 190c4697e9e4d9a9e2a13b4d03970bc3beda3ce1 Mon Sep 17 00:00:00 2001 From: Felix Chantrel Date: Thu, 21 Nov 2019 11:26:22 +1030 Subject: [PATCH 3/4] Rename FileOpener2.js to src/browser/FileOpener2.js --- FileOpener2.js => src/browser/FileOpener2.js | 0 1 file changed, 0 insertions(+), 0 deletions(-) rename FileOpener2.js => src/browser/FileOpener2.js (100%) diff --git a/FileOpener2.js b/src/browser/FileOpener2.js similarity index 100% rename from FileOpener2.js rename to src/browser/FileOpener2.js From f193cbd323146fcb940ecd5a013e9da7ca3738a6 Mon Sep 17 00:00:00 2001 From: Felix Chantrel Date: Thu, 21 Nov 2019 11:26:46 +1030 Subject: [PATCH 4/4] Rename FileSaver.min.js to src/browser/FileSaver.min.js --- FileSaver.min.js => src/browser/FileSaver.min.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) rename FileSaver.min.js => src/browser/FileSaver.min.js (98%) diff --git a/FileSaver.min.js b/src/browser/FileSaver.min.js similarity index 98% rename from FileSaver.min.js rename to src/browser/FileSaver.min.js index 183d42a1..6d8f0202 100644 --- a/FileSaver.min.js +++ b/src/browser/FileSaver.min.js @@ -1,3 +1,3 @@ (function(a,b){if("function"==typeof define&&define.amd)define([],b);else if("undefined"!=typeof exports)b();else{b(),a.FileSaver={exports:{}}.exports}})(this,function(){"use strict";function b(a,b){return"undefined"==typeof b?b={autoBom:!1}:"object"!=typeof b&&(console.warn("Deprecated: Expected third argument to be a object"),b={autoBom:!b}),b.autoBom&&/^\s*(?:text\/\S*|application\/xml|\S*\/\S*\+xml)\s*;.*charset\s*=\s*utf-8/i.test(a.type)?new Blob(["\uFEFF",a],{type:a.type}):a}function c(b,c,d){var e=new XMLHttpRequest;e.open("GET",b),e.responseType="blob",e.onload=function(){a(e.response,c,d)},e.onerror=function(){console.error("could not download file")},e.send()}function d(a){var b=new XMLHttpRequest;b.open("HEAD",a,!1);try{b.send()}catch(a){}return 200<=b.status&&299>=b.status}function e(a){try{a.dispatchEvent(new MouseEvent("click"))}catch(c){var b=document.createEvent("MouseEvents");b.initMouseEvent("click",!0,!0,window,0,0,0,80,20,!1,!1,!1,!1,0,null),a.dispatchEvent(b)}}var f="object"==typeof window&&window.window===window?window:"object"==typeof self&&self.self===self?self:"object"==typeof global&&global.global===global?global:void 0,a=f.saveAs||("object"!=typeof window||window!==f?function(){}:"download"in HTMLAnchorElement.prototype?function(b,g,h){var i=f.URL||f.webkitURL,j=document.createElement("a");g=g||b.name||"download",j.download=g,j.rel="noopener","string"==typeof b?(j.href=b,j.origin===location.origin?e(j):d(j.href)?c(b,g,h):e(j,j.target="_blank")):(j.href=i.createObjectURL(b),setTimeout(function(){i.revokeObjectURL(j.href)},4E4),setTimeout(function(){e(j)},0))}:"msSaveOrOpenBlob"in navigator?function(f,g,h){if(g=g||f.name||"download","string"!=typeof f)navigator.msSaveOrOpenBlob(b(f,h),g);else if(d(f))c(f,g,h);else{var i=document.createElement("a");i.href=f,i.target="_blank",setTimeout(function(){e(i)})}}:function(a,b,d,e){if(e=e||open("","_blank"),e&&(e.document.title=e.document.body.innerText="downloading..."),"string"==typeof a)return c(a,b,d);var g="application/octet-stream"===a.type,h=/constructor/i.test(f.HTMLElement)||f.safari,i=/CriOS\/[\d]+/.test(navigator.userAgent);if((i||g&&h)&&"undefined"!=typeof FileReader){var j=new FileReader;j.onloadend=function(){var a=j.result;a=i?a:a.replace(/^data:[^;]*;/,"data:attachment/file;"),e?e.location.href=a:location=a,e=null},j.readAsDataURL(a)}else{var k=f.URL||f.webkitURL,l=k.createObjectURL(a);e?e.location=l:location.href=l,e=null,setTimeout(function(){k.revokeObjectURL(l)},4E4)}});f.saveAs=a.saveAs=a,"undefined"!=typeof module&&(module.exports=a)}); -//# sourceMappingURL=FileSaver.min.js.map \ No newline at end of file +//# sourceMappingURL=FileSaver.min.js.map