RichDocumentsWebView.java 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424
  1. /*
  2. * Nextcloud Android client application
  3. *
  4. * @author Tobias Kaminsky
  5. * @author Chris Narkiewicz
  6. *
  7. * Copyright (C) 2018 Tobias Kaminsky
  8. * Copyright (C) 2018 Nextcloud GmbH.
  9. * Copyright (C) 2019 Chris Narkiewicz <hello@ezaquarii.com>
  10. *
  11. * This program is free software: you can redistribute it and/or modify
  12. * it under the terms of the GNU General Public License as published by
  13. * the Free Software Foundation, either version 3 of the License, or
  14. * (at your option) any later version.
  15. *
  16. * This program is distributed in the hope that it will be useful,
  17. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. * GNU General Public License for more details.
  20. *
  21. * You should have received a copy of the GNU General Public License
  22. * along with this program. If not, see <https://www.gnu.org/licenses/>.
  23. */
  24. package com.owncloud.android.ui.activity;
  25. import android.accounts.Account;
  26. import android.annotation.SuppressLint;
  27. import android.app.DownloadManager;
  28. import android.content.ActivityNotFoundException;
  29. import android.content.Context;
  30. import android.content.Intent;
  31. import android.graphics.Bitmap;
  32. import android.net.Uri;
  33. import android.os.Build;
  34. import android.os.Bundle;
  35. import android.text.TextUtils;
  36. import android.view.View;
  37. import android.webkit.JavascriptInterface;
  38. import android.webkit.ValueCallback;
  39. import android.webkit.WebChromeClient;
  40. import android.webkit.WebView;
  41. import android.widget.ImageView;
  42. import android.widget.ProgressBar;
  43. import android.widget.TextView;
  44. import android.widget.Toast;
  45. import com.bumptech.glide.Glide;
  46. import com.google.android.material.snackbar.Snackbar;
  47. import com.nextcloud.client.account.CurrentAccountProvider;
  48. import com.owncloud.android.R;
  49. import com.owncloud.android.datamodel.OCFile;
  50. import com.owncloud.android.datamodel.Template;
  51. import com.owncloud.android.datamodel.ThumbnailsCacheManager;
  52. import com.owncloud.android.lib.common.operations.RemoteOperationResult;
  53. import com.owncloud.android.lib.common.utils.Log_OC;
  54. import com.owncloud.android.operations.RichDocumentsCreateAssetOperation;
  55. import com.owncloud.android.ui.asynctasks.LoadUrlTask;
  56. import com.owncloud.android.ui.fragment.OCFileListFragment;
  57. import com.owncloud.android.utils.DisplayUtils;
  58. import com.owncloud.android.utils.MimeTypeUtil;
  59. import com.owncloud.android.utils.glide.CustomGlideStreamLoader;
  60. import org.json.JSONException;
  61. import org.json.JSONObject;
  62. import org.parceler.Parcels;
  63. import javax.inject.Inject;
  64. import androidx.annotation.RequiresApi;
  65. import butterknife.BindView;
  66. import butterknife.ButterKnife;
  67. import butterknife.Unbinder;
  68. import lombok.Getter;
  69. import lombok.Setter;
  70. /**
  71. * Opens document for editing via Richdocuments app in a web view
  72. */
  73. @RequiresApi(api = Build.VERSION_CODES.LOLLIPOP)
  74. public class RichDocumentsWebView extends ExternalSiteWebView {
  75. private static final String TAG = RichDocumentsWebView.class.getSimpleName();
  76. private static final int REQUEST_REMOTE_FILE = 100;
  77. public static final int REQUEST_LOCAL_FILE = 101;
  78. public static final int MINIMUM_API = Build.VERSION_CODES.LOLLIPOP;
  79. private Unbinder unbinder;
  80. private OCFile file;
  81. @Getter @Setter private Snackbar loadingSnackbar;
  82. public ValueCallback<Uri[]> uploadMessage;
  83. @BindView(R.id.progressBar2)
  84. ProgressBar progressBar;
  85. @BindView(R.id.thumbnail)
  86. ImageView thumbnail;
  87. @BindView(R.id.filename)
  88. TextView fileName;
  89. @Inject
  90. protected CurrentAccountProvider currentAccountProvider;
  91. @SuppressLint("AddJavascriptInterface") // suppress warning as webview is only used >= Lollipop
  92. @Override
  93. protected void onCreate(Bundle savedInstanceState) {
  94. showToolbar = false;
  95. webViewLayout = R.layout.richdocuments_webview;
  96. super.onCreate(savedInstanceState);
  97. unbinder = ButterKnife.bind(this);
  98. file = getIntent().getParcelableExtra(EXTRA_FILE);
  99. // TODO make file nullable
  100. if (file == null) {
  101. fileName.setText(R.string.create_file_from_template);
  102. Template template = Parcels.unwrap(getIntent().getParcelableExtra(EXTRA_TEMPLATE));
  103. int placeholder;
  104. switch (template.getType()) {
  105. case "document":
  106. placeholder = R.drawable.file_doc;
  107. break;
  108. case "spreadsheet":
  109. placeholder = R.drawable.file_xls;
  110. break;
  111. case "presentation":
  112. placeholder = R.drawable.file_ppt;
  113. break;
  114. default:
  115. placeholder = R.drawable.file;
  116. break;
  117. }
  118. Glide.with(this).using(new CustomGlideStreamLoader(currentAccountProvider)).load(template.getThumbnailLink())
  119. .placeholder(placeholder)
  120. .error(placeholder)
  121. .into(thumbnail);
  122. } else {
  123. setThumbnail(file, thumbnail);
  124. fileName.setText(file.getFileName());
  125. }
  126. webview.addJavascriptInterface(new RichDocumentsMobileInterface(), "RichDocumentsMobileInterface");
  127. webview.setWebChromeClient(new WebChromeClient() {
  128. RichDocumentsWebView activity = RichDocumentsWebView.this;
  129. @Override
  130. public boolean onShowFileChooser(WebView webView, ValueCallback<Uri[]> filePathCallback,
  131. FileChooserParams fileChooserParams) {
  132. if (uploadMessage != null) {
  133. uploadMessage.onReceiveValue(null);
  134. uploadMessage = null;
  135. }
  136. activity.uploadMessage = filePathCallback;
  137. Intent intent = fileChooserParams.createIntent();
  138. intent.setType("image/*");
  139. intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP | Intent.FLAG_ACTIVITY_SINGLE_TOP);
  140. try {
  141. activity.startActivityForResult(intent, REQUEST_LOCAL_FILE);
  142. } catch (ActivityNotFoundException e) {
  143. uploadMessage = null;
  144. Toast.makeText(getBaseContext(), "Cannot open file chooser", Toast.LENGTH_LONG).show();
  145. return false;
  146. }
  147. return true;
  148. }
  149. });
  150. // load url in background
  151. url = getIntent().getStringExtra(EXTRA_URL);
  152. if (TextUtils.isEmpty(url)) {
  153. new LoadUrlTask(this, getAccount()).execute(file.getLocalId());
  154. } else {
  155. webview.loadUrl(url);
  156. }
  157. }
  158. private void setThumbnail(OCFile file, ImageView thumbnailView) {
  159. // Todo minimize: only icon by mimetype
  160. if (file.isFolder()) {
  161. thumbnailView.setImageDrawable(MimeTypeUtil.getFolderTypeIcon(file.isSharedWithMe() ||
  162. file.isSharedWithSharee(), file.isSharedViaLink(), file.isEncrypted(), file.getMountType(),
  163. this));
  164. } else {
  165. if ((MimeTypeUtil.isImage(file) || MimeTypeUtil.isVideo(file)) && file.getRemoteId() != null) {
  166. // Thumbnail in cache?
  167. Bitmap thumbnail = ThumbnailsCacheManager.getBitmapFromDiskCache(
  168. ThumbnailsCacheManager.PREFIX_THUMBNAIL + file.getRemoteId());
  169. if (thumbnail != null && !file.isUpdateThumbnailNeeded()) {
  170. if (MimeTypeUtil.isVideo(file)) {
  171. Bitmap withOverlay = ThumbnailsCacheManager.addVideoOverlay(thumbnail);
  172. thumbnailView.setImageBitmap(withOverlay);
  173. } else {
  174. thumbnailView.setImageBitmap(thumbnail);
  175. }
  176. } else {
  177. // generate new thumbnail
  178. if (ThumbnailsCacheManager.cancelPotentialThumbnailWork(file, thumbnailView)) {
  179. try {
  180. final ThumbnailsCacheManager.ThumbnailGenerationTask task =
  181. new ThumbnailsCacheManager.ThumbnailGenerationTask(thumbnailView,
  182. getStorageManager(), getAccount());
  183. if (thumbnail == null) {
  184. if (MimeTypeUtil.isVideo(file)) {
  185. thumbnail = ThumbnailsCacheManager.mDefaultVideo;
  186. } else {
  187. thumbnail = ThumbnailsCacheManager.mDefaultImg;
  188. }
  189. }
  190. final ThumbnailsCacheManager.AsyncThumbnailDrawable asyncDrawable =
  191. new ThumbnailsCacheManager.AsyncThumbnailDrawable(getResources(), thumbnail, task);
  192. thumbnailView.setImageDrawable(asyncDrawable);
  193. task.execute(new ThumbnailsCacheManager.ThumbnailGenerationTaskObject(file,
  194. file.getRemoteId()));
  195. } catch (IllegalArgumentException e) {
  196. Log_OC.d(TAG, "ThumbnailGenerationTask : " + e.getMessage());
  197. }
  198. }
  199. }
  200. if ("image/png".equalsIgnoreCase(file.getMimeType())) {
  201. thumbnailView.setBackgroundColor(getResources().getColor(R.color.background_color));
  202. }
  203. } else {
  204. thumbnailView.setImageDrawable(MimeTypeUtil.getFileTypeIcon(file.getMimeType(), file.getFileName(),
  205. getAccount(), this));
  206. }
  207. }
  208. }
  209. @Override
  210. protected void onNewIntent(Intent intent) {
  211. super.onNewIntent(intent);
  212. }
  213. private void openFileChooser() {
  214. Intent action = new Intent(this, FilePickerActivity.class);
  215. action.putExtra(OCFileListFragment.ARG_MIMETYPE, "image/");
  216. startActivityForResult(action, REQUEST_REMOTE_FILE);
  217. }
  218. private void openShareDialog() {
  219. Intent intent = new Intent(this, ShareActivity.class);
  220. intent.putExtra(FileActivity.EXTRA_FILE, file);
  221. intent.putExtra(FileActivity.EXTRA_ACCOUNT, getAccount());
  222. startActivity(intent);
  223. }
  224. @Override
  225. protected void onActivityResult(int requestCode, int resultCode, Intent data) {
  226. if (RESULT_OK != resultCode) {
  227. // TODO
  228. return;
  229. }
  230. switch (requestCode) {
  231. case REQUEST_LOCAL_FILE:
  232. handleLocalFile(data, resultCode);
  233. break;
  234. case REQUEST_REMOTE_FILE:
  235. handleRemoteFile(data);
  236. break;
  237. default:
  238. // unexpected, do nothing
  239. break;
  240. }
  241. super.onActivityResult(requestCode, resultCode, data);
  242. }
  243. private void handleLocalFile(Intent data, int resultCode) {
  244. if (uploadMessage == null) {
  245. return;
  246. }
  247. uploadMessage.onReceiveValue(WebChromeClient.FileChooserParams.parseResult(resultCode, data));
  248. uploadMessage = null;
  249. }
  250. private void handleRemoteFile(Intent data) {
  251. OCFile file = data.getParcelableExtra(FolderPickerActivity.EXTRA_FILES);
  252. new Thread(() -> {
  253. Account account = currentAccountProvider.getCurrentAccount();
  254. RichDocumentsCreateAssetOperation operation = new RichDocumentsCreateAssetOperation(file.getRemotePath());
  255. RemoteOperationResult result = operation.execute(account, this);
  256. if (result.isSuccess()) {
  257. String asset = (String) result.getSingleData();
  258. runOnUiThread(() -> webview.evaluateJavascript("OCA.RichDocuments.documentsMain.postAsset('" +
  259. file.getFileName() + "', '" + asset + "');", null));
  260. } else {
  261. runOnUiThread(() -> DisplayUtils.showSnackMessage(this, "Inserting image failed!"));
  262. }
  263. }).start();
  264. }
  265. @Override
  266. protected void onSaveInstanceState(Bundle outState) {
  267. outState.putString(EXTRA_URL, url);
  268. super.onSaveInstanceState(outState);
  269. }
  270. @Override
  271. public void onRestoreInstanceState(Bundle savedInstanceState) {
  272. url = savedInstanceState.getString(EXTRA_URL);
  273. super.onRestoreInstanceState(savedInstanceState);
  274. }
  275. @Override
  276. protected void onDestroy() {
  277. unbinder.unbind();
  278. webview.destroy();
  279. super.onDestroy();
  280. }
  281. public void closeView() {
  282. webview.destroy();
  283. finish();
  284. }
  285. private void hideLoading() {
  286. thumbnail.setVisibility(View.GONE);
  287. fileName.setVisibility(View.GONE);
  288. progressBar.setVisibility(View.GONE);
  289. webview.setVisibility(View.VISIBLE);
  290. if (loadingSnackbar != null) {
  291. loadingSnackbar.dismiss();
  292. }
  293. }
  294. @Override
  295. protected void onResume() {
  296. super.onResume();
  297. webview.evaluateJavascript("if (typeof OCA.RichDocuments.documentsMain.postGrabFocus !== 'undefined') " +
  298. "{ OCA.RichDocuments.documentsMain.postGrabFocus(); }", null);
  299. }
  300. private class RichDocumentsMobileInterface {
  301. @JavascriptInterface
  302. public void close() {
  303. runOnUiThread(RichDocumentsWebView.this::closeView);
  304. }
  305. @JavascriptInterface
  306. public void insertGraphic() {
  307. openFileChooser();
  308. }
  309. @JavascriptInterface
  310. public void share() {
  311. openShareDialog();
  312. }
  313. @JavascriptInterface
  314. public void documentLoaded() {
  315. runOnUiThread(RichDocumentsWebView.this::hideLoading);
  316. }
  317. @JavascriptInterface
  318. public void downloadAs(String json) {
  319. Uri downloadUrl;
  320. try {
  321. JSONObject downloadJson = new JSONObject(json);
  322. downloadUrl = Uri.parse(downloadJson.getString("URL"));
  323. } catch (JSONException e) {
  324. Log_OC.e(this, "Failed to parse rename json message: " + e);
  325. return;
  326. }
  327. DownloadManager downloadmanager = (DownloadManager) getSystemService(Context.DOWNLOAD_SERVICE);
  328. if (downloadmanager == null) {
  329. DisplayUtils.showSnackMessage(webview, getString(R.string.failed_to_download));
  330. return;
  331. }
  332. DownloadManager.Request request = new DownloadManager.Request(downloadUrl);
  333. request.allowScanningByMediaScanner();
  334. request.setNotificationVisibility(DownloadManager.Request.VISIBILITY_VISIBLE_NOTIFY_COMPLETED);
  335. downloadmanager.enqueue(request);
  336. }
  337. @JavascriptInterface
  338. public void fileRename(String renameString) {
  339. // when shared file is renamed in another instance, we will get notified about it
  340. // need to change filename for sharing
  341. try {
  342. JSONObject renameJson = new JSONObject(renameString);
  343. String newName = renameJson.getString("NewName");
  344. file.setFileName(newName);
  345. } catch (JSONException e) {
  346. Log_OC.e(this, "Failed to parse rename json message: " + e);
  347. }
  348. }
  349. }
  350. }