mirror of
https://github.com/M66B/FairEmail.git
synced 2026-03-26 19:05:23 +01:00
Added draft message printing
This commit is contained in:
@@ -21,21 +21,54 @@ package eu.faircode.email;
|
||||
|
||||
import android.app.Activity;
|
||||
import android.app.Dialog;
|
||||
import android.content.ActivityNotFoundException;
|
||||
import android.content.Context;
|
||||
import android.content.DialogInterface;
|
||||
import android.content.SharedPreferences;
|
||||
import android.os.Bundle;
|
||||
import android.print.PrintAttributes;
|
||||
import android.print.PrintDocumentAdapter;
|
||||
import android.print.PrintJob;
|
||||
import android.print.PrintManager;
|
||||
import android.text.Spanned;
|
||||
import android.text.TextUtils;
|
||||
import android.view.LayoutInflater;
|
||||
import android.view.View;
|
||||
import android.webkit.WebSettings;
|
||||
import android.webkit.WebView;
|
||||
import android.webkit.WebViewClient;
|
||||
import android.widget.CheckBox;
|
||||
import android.widget.CompoundButton;
|
||||
import android.widget.Toast;
|
||||
|
||||
import androidx.annotation.NonNull;
|
||||
import androidx.annotation.Nullable;
|
||||
import androidx.appcompat.app.AlertDialog;
|
||||
import androidx.fragment.app.FragmentActivity;
|
||||
import androidx.fragment.app.FragmentManager;
|
||||
import androidx.preference.PreferenceManager;
|
||||
|
||||
import org.jsoup.nodes.Document;
|
||||
import org.jsoup.nodes.Element;
|
||||
import org.jsoup.select.Elements;
|
||||
|
||||
import java.io.File;
|
||||
import java.io.FileOutputStream;
|
||||
import java.io.IOException;
|
||||
import java.io.OutputStream;
|
||||
import java.net.HttpURLConnection;
|
||||
import java.net.URL;
|
||||
import java.nio.charset.StandardCharsets;
|
||||
import java.text.DateFormat;
|
||||
import java.text.SimpleDateFormat;
|
||||
import java.util.ArrayList;
|
||||
import java.util.List;
|
||||
import java.util.concurrent.Callable;
|
||||
import java.util.concurrent.Future;
|
||||
|
||||
public class FragmentDialogPrint extends FragmentDialogBase {
|
||||
private static WebView printWebView = null;
|
||||
|
||||
@NonNull
|
||||
@Override
|
||||
public Dialog onCreateDialog(@Nullable Bundle savedInstanceState) {
|
||||
@@ -86,4 +119,276 @@ public class FragmentDialogPrint extends FragmentDialogBase {
|
||||
})
|
||||
.create();
|
||||
}
|
||||
|
||||
static void print(ActivityBase activity, FragmentManager fm, Bundle args) {
|
||||
if (activity == null) {
|
||||
Log.w("Print no activity");
|
||||
return;
|
||||
}
|
||||
|
||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(activity);
|
||||
boolean print_html_header = prefs.getBoolean("print_html_header", true);
|
||||
boolean print_html_images = prefs.getBoolean("print_html_images", true);
|
||||
|
||||
args.putBoolean("print_html_header", print_html_header);
|
||||
args.putBoolean("print_html_images", print_html_images);
|
||||
|
||||
new SimpleTask<String[]>() {
|
||||
@Override
|
||||
protected String[] onExecute(Context context, Bundle args) throws IOException {
|
||||
long id = args.getLong("id");
|
||||
boolean headers = args.getBoolean("headers");
|
||||
boolean print_html_header = args.getBoolean("print_html_header");
|
||||
boolean print_html_images = args.getBoolean("print_html_images");
|
||||
CharSequence selected = args.getCharSequence("selected");
|
||||
boolean draft = args.getBoolean("draft");
|
||||
|
||||
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
|
||||
int timeout = prefs.getInt("timeout", ImageHelper.DOWNLOAD_TIMEOUT) * 1000;
|
||||
|
||||
DB db = DB.getInstance(context);
|
||||
EntityMessage message = db.message().getMessage(id);
|
||||
if (message == null || !message.content)
|
||||
return null;
|
||||
|
||||
File file = message.getFile(context);
|
||||
if (!file.exists())
|
||||
return null;
|
||||
|
||||
List<EntityAttachment> attachments = db.attachment().getAttachments(message.id);
|
||||
if (attachments == null)
|
||||
return null;
|
||||
|
||||
Document document;
|
||||
if (!TextUtils.isEmpty(selected) && selected instanceof Spanned)
|
||||
document = JsoupEx.parse(HtmlHelper.toHtml((Spanned) selected, context));
|
||||
else
|
||||
document = JsoupEx.parse(file);
|
||||
|
||||
boolean monospaced_pre = prefs.getBoolean("monospaced_pre", false);
|
||||
if (message.isPlainOnly() && monospaced_pre)
|
||||
HtmlHelper.restorePre(document);
|
||||
|
||||
HtmlHelper.markText(document);
|
||||
|
||||
HtmlHelper.embedInlineImages(context, id, document, true);
|
||||
|
||||
// onPageFinished will not be called if not all images can be loaded
|
||||
File dir = new File(context.getFilesDir(), "images");
|
||||
List<Future<Void>> futures = new ArrayList<>();
|
||||
Elements imgs = document.select("img");
|
||||
for (int i = 0; i < imgs.size(); i++) {
|
||||
Element img = imgs.get(i);
|
||||
String src = img.attr("src");
|
||||
if (src.startsWith("http:") || src.startsWith("https:")) {
|
||||
final File out = new File(dir, id + "." + i + ".print");
|
||||
img.attr("src", "file:" + out.getAbsolutePath());
|
||||
|
||||
if (print_html_images) {
|
||||
if (out.exists() && out.length() > 0)
|
||||
continue;
|
||||
} else {
|
||||
out.delete();
|
||||
continue;
|
||||
}
|
||||
|
||||
futures.add(Helper.getDownloadTaskExecutor().submit(new Callable<Void>() {
|
||||
@Override
|
||||
public Void call() throws Exception {
|
||||
try (OutputStream os = new FileOutputStream(out)) {
|
||||
URL url = new URL(src);
|
||||
Log.i("Caching url=" + url);
|
||||
|
||||
HttpURLConnection connection = null;
|
||||
try {
|
||||
connection = ConnectionHelper.openConnectionUnsafe(context, src, timeout, timeout);
|
||||
Helper.copy(connection.getInputStream(), os);
|
||||
} finally {
|
||||
if (connection != null)
|
||||
connection.disconnect();
|
||||
}
|
||||
} catch (Throwable ex) {
|
||||
Log.w(ex);
|
||||
}
|
||||
|
||||
return null;
|
||||
}
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
for (Future<Void> future : futures)
|
||||
try {
|
||||
future.get();
|
||||
} catch (Throwable ex) {
|
||||
Log.w(ex);
|
||||
}
|
||||
|
||||
// @page WordSection1 {size:612.0pt 792.0pt; margin:70.85pt 70.85pt 70.85pt 70.85pt;}
|
||||
// div.WordSection1 {page:WordSection1;}
|
||||
// <body><div class=WordSection1>
|
||||
|
||||
for (Element element : document.body().select("div[class]")) {
|
||||
String clazz = element.attr("class");
|
||||
if (clazz.startsWith("WordSection"))
|
||||
element.removeClass(clazz);
|
||||
}
|
||||
|
||||
if (print_html_header) {
|
||||
Element header = document.createElement("p");
|
||||
|
||||
if (draft) {
|
||||
Element div = document.createElement("div");
|
||||
div.attr("style", "text-align: center;");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_compose));
|
||||
strong.attr("style", "text-transform: uppercase;");
|
||||
div.appendChild(strong);
|
||||
header.appendChild(div);
|
||||
header.appendElement("hr");
|
||||
}
|
||||
|
||||
if (message.from != null && message.from.length > 0) {
|
||||
Element span = document.createElement("span");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_from));
|
||||
span.appendChild(strong);
|
||||
span.appendText(" " + MessageHelper.formatAddresses(message.from));
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
if (message.to != null && message.to.length > 0) {
|
||||
Element span = document.createElement("span");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_to));
|
||||
span.appendChild(strong);
|
||||
span.appendText(" " + MessageHelper.formatAddresses(message.to));
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
if (message.cc != null && message.cc.length > 0) {
|
||||
Element span = document.createElement("span");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_cc));
|
||||
span.appendChild(strong);
|
||||
span.appendText(" " + MessageHelper.formatAddresses(message.cc));
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
if (message.received != null && !draft) {
|
||||
DateFormat DTF = Helper.getDateTimeInstance(context, SimpleDateFormat.LONG, SimpleDateFormat.LONG);
|
||||
|
||||
Element span = document.createElement("span");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_received));
|
||||
span.appendChild(strong);
|
||||
span.appendText(" " + DTF.format(message.received));
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
for (EntityAttachment attachment : attachments)
|
||||
if (attachment.isAttachment()) {
|
||||
Element span = document.createElement("span");
|
||||
Element strong = document.createElement("strong");
|
||||
strong.text(context.getString(R.string.title_attachment));
|
||||
span.appendChild(strong);
|
||||
if (!TextUtils.isEmpty(attachment.name))
|
||||
span.appendText(" " + attachment.name);
|
||||
if (attachment.size != null)
|
||||
span.appendText(" " + Helper.humanReadableByteCount(attachment.size));
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
if (!TextUtils.isEmpty(message.subject)) {
|
||||
Element span = document.createElement("span");
|
||||
span.appendText(message.subject);
|
||||
span.appendElement("br");
|
||||
header.appendChild(span);
|
||||
}
|
||||
|
||||
if (headers && message.headers != null) {
|
||||
header.appendElement("hr");
|
||||
Element pre = document.createElement("pre");
|
||||
pre.text(message.headers);
|
||||
header.appendChild(pre);
|
||||
}
|
||||
|
||||
header.appendElement("hr").appendElement("br");
|
||||
|
||||
document.body().prependChild(header);
|
||||
}
|
||||
|
||||
return new String[]{message.subject, document.body().html()};
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onExecuted(Bundle args, final String[] data) {
|
||||
if (data == null) {
|
||||
Log.w("Print no data");
|
||||
return;
|
||||
}
|
||||
|
||||
final Context context = activity.getOriginalContext();
|
||||
boolean print_html_images = args.getBoolean("print_html_images");
|
||||
|
||||
// https://developer.android.com/training/printing/html-docs.html
|
||||
printWebView = new WebView(context);
|
||||
|
||||
WebSettings settings = printWebView.getSettings();
|
||||
settings.setUserAgentString(WebViewEx.getUserAgent(context, printWebView));
|
||||
settings.setLoadsImagesAutomatically(print_html_images);
|
||||
settings.setMixedContentMode(WebSettings.MIXED_CONTENT_ALWAYS_ALLOW);
|
||||
settings.setAllowFileAccess(true);
|
||||
|
||||
printWebView.setWebViewClient(new WebViewClient() {
|
||||
public boolean shouldOverrideUrlLoading(WebView view, String url) {
|
||||
return false;
|
||||
}
|
||||
|
||||
@Override
|
||||
public void onPageFinished(WebView view, String url) {
|
||||
Log.i("Print page finished");
|
||||
|
||||
try {
|
||||
if (printWebView == null) {
|
||||
Log.w("Print no view");
|
||||
return;
|
||||
}
|
||||
|
||||
PrintManager printManager = (PrintManager) context.getSystemService(Context.PRINT_SERVICE);
|
||||
String jobName = activity.getString(R.string.app_name);
|
||||
if (!TextUtils.isEmpty(data[0]))
|
||||
jobName += " - " + data[0];
|
||||
|
||||
Log.i("Print queue job=" + jobName);
|
||||
PrintDocumentAdapter adapter = printWebView.createPrintDocumentAdapter(jobName);
|
||||
PrintJob job = printManager.print(jobName, adapter, new PrintAttributes.Builder().build());
|
||||
EntityLog.log(context, "Print queued job=" + job.getInfo());
|
||||
} catch (Throwable ex) {
|
||||
try {
|
||||
Log.unexpectedError(fm, ex, !(ex instanceof ActivityNotFoundException));
|
||||
} catch (IllegalStateException exex) {
|
||||
ToastEx.makeText(context, Log.formatThrowable(ex), Toast.LENGTH_LONG).show();
|
||||
}
|
||||
} finally {
|
||||
printWebView = null;
|
||||
}
|
||||
}
|
||||
});
|
||||
|
||||
Log.i("Print load data");
|
||||
printWebView.loadDataWithBaseURL("about:blank", data[1], "text/html", StandardCharsets.UTF_8.name(), null);
|
||||
}
|
||||
|
||||
@Override
|
||||
protected void onException(Bundle args, Throwable ex) {
|
||||
Log.unexpectedError(fm, ex);
|
||||
}
|
||||
}.execute(activity, args, "print");
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user