PreviewTextFragment.java 8.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245
  1. /*
  2. * ownCloud Android client application
  3. *
  4. * Copyright (C) 2016 ownCloud Inc.
  5. *
  6. * This program is free software: you can redistribute it and/or modify
  7. * it under the terms of the GNU General Public License version 2,
  8. * as published by the Free Software Foundation.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. *
  18. */
  19. package com.owncloud.android.ui.preview;
  20. import android.app.Activity;
  21. import android.content.Context;
  22. import android.content.Intent;
  23. import android.content.res.Resources;
  24. import android.graphics.Color;
  25. import android.graphics.PorterDuff;
  26. import android.net.Uri;
  27. import android.os.Bundle;
  28. import android.os.Handler;
  29. import android.text.Html;
  30. import android.text.Spanned;
  31. import android.text.TextUtils;
  32. import android.text.method.LinkMovementMethod;
  33. import android.view.LayoutInflater;
  34. import android.view.View;
  35. import android.view.ViewGroup;
  36. import android.widget.TextView;
  37. import com.nextcloud.client.account.UserAccountManager;
  38. import com.nextcloud.client.device.DeviceInfo;
  39. import com.nextcloud.client.di.Injectable;
  40. import com.owncloud.android.R;
  41. import com.owncloud.android.databinding.TextFilePreviewBinding;
  42. import com.owncloud.android.datamodel.OCFile;
  43. import com.owncloud.android.lib.common.utils.Log_OC;
  44. import com.owncloud.android.ui.activity.FileDisplayActivity;
  45. import com.owncloud.android.ui.fragment.FileFragment;
  46. import com.owncloud.android.utils.DisplayUtils;
  47. import com.owncloud.android.utils.MimeTypeUtil;
  48. import com.owncloud.android.utils.StringUtils;
  49. import com.owncloud.android.utils.theme.ThemeColorUtils;
  50. import javax.inject.Inject;
  51. import androidx.annotation.NonNull;
  52. import androidx.annotation.Nullable;
  53. import androidx.appcompat.widget.SearchView;
  54. import io.noties.markwon.AbstractMarkwonPlugin;
  55. import io.noties.markwon.Markwon;
  56. import io.noties.markwon.MarkwonConfiguration;
  57. import io.noties.markwon.core.MarkwonTheme;
  58. import io.noties.markwon.ext.strikethrough.StrikethroughPlugin;
  59. import io.noties.markwon.ext.tables.TablePlugin;
  60. import io.noties.markwon.ext.tasklist.TaskListDrawable;
  61. import io.noties.markwon.ext.tasklist.TaskListPlugin;
  62. import io.noties.markwon.html.HtmlPlugin;
  63. import io.noties.markwon.syntax.Prism4jTheme;
  64. import io.noties.markwon.syntax.Prism4jThemeDefault;
  65. import io.noties.markwon.syntax.SyntaxHighlightPlugin;
  66. import io.noties.prism4j.Prism4j;
  67. import io.noties.prism4j.annotations.PrismBundle;
  68. @PrismBundle(
  69. include = {
  70. "c", "clike", "clojure", "cpp", "csharp", "css", "dart", "git", "go", "groovy", "java", "javascript", "json",
  71. "kotlin", "latex", "makefile", "markdown", "markup", "python", "scala", "sql", "swift", "yaml"
  72. },
  73. grammarLocatorClassName = ".MarkwonGrammarLocator"
  74. )
  75. public abstract class PreviewTextFragment extends FileFragment implements SearchView.OnQueryTextListener, Injectable {
  76. private static final String TAG = PreviewTextFragment.class.getSimpleName();
  77. protected SearchView searchView;
  78. protected String searchQuery = "";
  79. protected boolean searchOpen;
  80. protected Handler handler;
  81. protected String originalText;
  82. @Inject UserAccountManager accountManager;
  83. @Inject DeviceInfo deviceInfo;
  84. protected TextFilePreviewBinding binding;
  85. /**
  86. * {@inheritDoc}
  87. */
  88. @Override
  89. public View onCreateView(@NonNull LayoutInflater inflater, ViewGroup container,
  90. Bundle savedInstanceState) {
  91. super.onCreateView(inflater, container, savedInstanceState);
  92. Log_OC.e(TAG, "onCreateView");
  93. binding = TextFilePreviewBinding.inflate(inflater, container, false);
  94. View view = binding.getRoot();
  95. binding.emptyListProgress.setVisibility(View.VISIBLE);
  96. return view;
  97. }
  98. @Override
  99. public void onStart() {
  100. super.onStart();
  101. Log_OC.e(TAG, "onStart");
  102. loadAndShowTextPreview();
  103. }
  104. @Override
  105. public void onDestroyView() {
  106. super.onDestroyView();
  107. binding = null;
  108. }
  109. abstract void loadAndShowTextPreview();
  110. @Override
  111. public boolean onQueryTextSubmit(String query) {
  112. performSearch(query, 0);
  113. return true;
  114. }
  115. @Override
  116. public boolean onQueryTextChange(final String newText) {
  117. performSearch(newText, 500);
  118. return true;
  119. }
  120. private void performSearch(final String query, int delay) {
  121. handler.removeCallbacksAndMessages(null);
  122. if (originalText != null) {
  123. if (getActivity() instanceof FileDisplayActivity) {
  124. FileDisplayActivity fileDisplayActivity = (FileDisplayActivity) getActivity();
  125. fileDisplayActivity.setSearchQuery(query);
  126. }
  127. handler.postDelayed(() -> markText(query), delay);
  128. }
  129. if (delay == 0 && searchView != null) {
  130. searchView.clearFocus();
  131. }
  132. }
  133. private void markText(String query) {
  134. // called asynchronously - must check preconditions in case of UI detachment
  135. if (binding == null) {
  136. return;
  137. }
  138. final Activity activity = getActivity();
  139. if (activity == null) {
  140. return;
  141. }
  142. final Resources resources = activity.getResources();
  143. if (resources == null) {
  144. return;
  145. }
  146. if (!TextUtils.isEmpty(query)) {
  147. String coloredText = StringUtils.searchAndColor(originalText,
  148. query,
  149. resources.getColor(R.color.primary));
  150. binding.textPreview.setText(Html.fromHtml(coloredText.replace("\n", "<br \\>")));
  151. } else {
  152. setText(binding.textPreview, originalText, getFile(), activity);
  153. }
  154. }
  155. protected static Spanned getRenderedMarkdownText(Activity activity, String markdown) {
  156. Prism4j prism4j = new Prism4j(new MarkwonGrammarLocator());
  157. Prism4jTheme prism4jTheme = Prism4jThemeDefault.create();
  158. TaskListDrawable drawable = new TaskListDrawable(Color.GRAY, Color.GRAY, Color.WHITE);
  159. drawable.setColorFilter(ThemeColorUtils.primaryColor(activity, true), PorterDuff.Mode.SRC_ATOP);
  160. final Markwon markwon = Markwon.builder(activity)
  161. .usePlugin(new AbstractMarkwonPlugin() {
  162. @Override
  163. public void configureTheme(@NonNull MarkwonTheme.Builder builder) {
  164. builder.linkColor(ThemeColorUtils.primaryColor(activity, true));
  165. builder.headingBreakHeight(0);
  166. }
  167. @Override
  168. public void configureConfiguration(@NonNull MarkwonConfiguration.Builder builder) {
  169. builder.linkResolver((view, link) -> {
  170. Intent intent = new Intent(Intent.ACTION_VIEW, Uri.parse(link));
  171. DisplayUtils.startIntentIfAppAvailable(intent, activity, R.string.no_browser_available);
  172. });
  173. }
  174. })
  175. .usePlugin(TablePlugin.create(activity))
  176. .usePlugin(TaskListPlugin.create(drawable))
  177. .usePlugin(StrikethroughPlugin.create())
  178. .usePlugin(HtmlPlugin.create())
  179. .usePlugin(SyntaxHighlightPlugin.create(prism4j, prism4jTheme))
  180. .build();
  181. return markwon.toMarkdown(markdown);
  182. }
  183. /**
  184. * Finishes the preview
  185. */
  186. protected void finish() {
  187. requireActivity().runOnUiThread(() -> requireActivity().onBackPressed());
  188. }
  189. public static void setText(TextView textView, String text, OCFile file, Activity activity) {
  190. setText(textView, text, file, activity, false, false);
  191. }
  192. public static void setText(TextView textView,
  193. @Nullable String text,
  194. @Nullable OCFile file,
  195. Activity activity,
  196. boolean ignoreMimetype,
  197. boolean preview) {
  198. if (text == null) {
  199. return;
  200. }
  201. if ((ignoreMimetype || file != null && MimeTypeUtil.MIMETYPE_TEXT_MARKDOWN.equals(file.getMimeType()))
  202. && activity != null) {
  203. if (!preview) {
  204. // clickable links prevent to open full view of rich workspace
  205. textView.setMovementMethod(LinkMovementMethod.getInstance());
  206. }
  207. textView.setText(getRenderedMarkdownText(activity, text));
  208. } else {
  209. textView.setText(text);
  210. }
  211. }
  212. }