FairEmail/app/src/main/java/eu/faircode/email/StyleHelper.java

615 lines
27 KiB
Java
Raw Normal View History

2019-09-27 17:05:34 +00:00
package eu.faircode.email;
2021-01-01 07:56:36 +00:00
/*
This file is part of FairEmail.
FairEmail is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
FairEmail is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with FairEmail. If not, see <http://www.gnu.org/licenses/>.
Copyright 2018-2021 by Marcel Bokhorst (M66B)
*/
2020-08-12 08:25:08 +00:00
import android.content.Context;
2020-06-28 21:36:31 +00:00
import android.content.DialogInterface;
2020-08-14 18:15:52 +00:00
import android.content.SharedPreferences;
2019-09-27 17:05:34 +00:00
import android.graphics.Typeface;
2020-08-12 06:18:39 +00:00
import android.os.Build;
2020-11-17 18:37:56 +00:00
import android.text.Editable;
2020-10-03 09:37:43 +00:00
import android.text.Layout;
2021-05-10 20:10:25 +00:00
import android.text.SpannableStringBuilder;
2019-09-27 17:05:34 +00:00
import android.text.Spanned;
2020-10-03 09:37:43 +00:00
import android.text.TextUtils;
import android.text.style.AlignmentSpan;
2020-08-11 16:46:26 +00:00
import android.text.style.BulletSpan;
2020-11-17 19:22:34 +00:00
import android.text.style.CharacterStyle;
2019-09-27 17:05:34 +00:00
import android.text.style.ForegroundColorSpan;
2019-09-29 10:10:19 +00:00
import android.text.style.ImageSpan;
2020-11-17 18:37:56 +00:00
import android.text.style.ParagraphStyle;
2020-10-01 08:22:20 +00:00
import android.text.style.QuoteSpan;
2019-09-27 17:05:34 +00:00
import android.text.style.RelativeSizeSpan;
2020-10-01 08:31:58 +00:00
import android.text.style.StrikethroughSpan;
2019-09-27 17:05:34 +00:00
import android.text.style.StyleSpan;
2020-06-28 16:39:27 +00:00
import android.text.style.TypefaceSpan;
2019-09-27 17:05:34 +00:00
import android.text.style.URLSpan;
import android.text.style.UnderlineSpan;
2020-10-03 12:17:13 +00:00
import android.util.Pair;
2020-06-28 15:18:42 +00:00
import android.view.MenuItem;
2020-08-12 08:25:08 +00:00
import android.view.SubMenu;
2020-06-28 15:18:42 +00:00
import android.view.View;
2019-09-27 17:05:34 +00:00
import android.widget.EditText;
2020-06-28 15:18:42 +00:00
import androidx.appcompat.widget.PopupMenu;
2020-11-12 14:08:34 +00:00
import androidx.lifecycle.LifecycleOwner;
2020-08-14 18:15:52 +00:00
import androidx.preference.PreferenceManager;
2020-06-28 15:18:42 +00:00
2020-06-28 21:36:31 +00:00
import com.flask.colorpicker.ColorPickerView;
import com.flask.colorpicker.builder.ColorPickerClickListener;
import com.flask.colorpicker.builder.ColorPickerDialogBuilder;
2019-09-27 17:05:34 +00:00
import java.util.ArrayList;
import java.util.List;
2020-10-03 09:37:43 +00:00
import java.util.Locale;
2019-09-27 17:05:34 +00:00
public class StyleHelper {
2020-11-12 14:08:34 +00:00
static boolean apply(int action, LifecycleOwner owner, View anchor, EditText etBody, Object... args) {
2019-09-27 17:05:34 +00:00
Log.i("Style action=" + action);
try {
2021-05-06 05:58:04 +00:00
int _start = etBody.getSelectionStart();
int _end = etBody.getSelectionEnd();
if (_start < 0)
_start = 0;
if (_end < 0)
_end = 0;
if (_start > _end) {
int tmp = _start;
_start = _end;
_end = tmp;
2019-09-27 17:05:34 +00:00
}
2021-05-06 05:58:04 +00:00
final Editable edit = etBody.getText();
final int start = _start;
final int end = _end;
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
if (action == R.id.menu_bold || action == R.id.menu_italic) {
int style = (action == R.id.menu_bold ? Typeface.BOLD : Typeface.ITALIC);
boolean has = false;
2021-05-06 05:58:04 +00:00
for (StyleSpan span : edit.getSpans(start, end, StyleSpan.class))
2021-02-05 10:15:02 +00:00
if (span.getStyle() == style) {
2019-09-27 17:05:34 +00:00
has = true;
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2019-09-27 17:05:34 +00:00
}
2021-02-05 10:15:02 +00:00
if (!has)
2021-05-06 05:58:04 +00:00
edit.setSpan(new StyleSpan(style), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2019-09-27 17:05:34 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
2021-02-05 10:15:02 +00:00
etBody.setSelection(start, end);
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
return true;
} else if (action == R.id.menu_underline) {
boolean has = false;
2021-05-06 05:58:04 +00:00
for (UnderlineSpan span : edit.getSpans(start, end, UnderlineSpan.class)) {
2021-02-05 10:15:02 +00:00
has = true;
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2019-09-27 17:05:34 +00:00
}
2021-02-05 10:15:02 +00:00
if (!has)
2021-05-06 05:58:04 +00:00
edit.setSpan(new UnderlineSpan(), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2021-02-05 10:15:02 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
2021-02-05 10:15:02 +00:00
etBody.setSelection(start, end);
return true;
} else if (action == R.id.menu_style) {
PopupMenuLifecycle popupMenu = new PopupMenuLifecycle(anchor.getContext(), owner, anchor);
popupMenu.inflate(R.menu.popup_style);
2021-05-07 07:41:26 +00:00
popupMenu.insertIcons(anchor.getContext());
2021-02-05 10:15:02 +00:00
2021-05-10 20:10:25 +00:00
String[] fontNameNames = anchor.getResources().getStringArray(R.array.fontNameNames);
String[] fontNameValues = anchor.getResources().getStringArray(R.array.fontNameValues);
2021-02-05 10:15:02 +00:00
SubMenu smenu = popupMenu.getMenu().findItem(R.id.menu_style_font).getSubMenu();
2021-05-10 20:10:25 +00:00
for (int i = 0; i < fontNameNames.length; i++) {
SpannableStringBuilder ssb = new SpannableStringBuilder(fontNameNames[i]);
ssb.setSpan(new TypefaceSpan(fontNameValues[i]), 0, ssb.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
smenu.add(R.id.group_style_font, i, 0, ssb);
}
smenu.add(R.id.group_style_font, fontNameNames.length, 0, R.string.title_style_font_default);
2021-02-05 10:15:02 +00:00
2021-05-06 06:22:04 +00:00
int level = -1;
BulletSpan[] spans = edit.getSpans(start, end, BulletSpan.class);
for (BulletSpan span : spans)
if (span instanceof NumberSpan)
level = ((NumberSpan) span).getLevel();
else if (span instanceof BulletSpanEx)
level = ((BulletSpanEx) span).getLevel();
popupMenu.getMenu().findItem(R.id.menu_style_list_increase).setVisible(level >= 0);
popupMenu.getMenu().findItem(R.id.menu_style_list_decrease).setVisible(level > 0);
2021-02-05 10:15:02 +00:00
popupMenu.setOnMenuItemClickListener(new PopupMenu.OnMenuItemClickListener() {
@Override
public boolean onMenuItemClick(MenuItem item) {
try {
int groupId = item.getGroupId();
if (groupId == R.id.group_style_size) {
return setSize(item);
} else if (groupId == R.id.group_style_color) {
return setColor(item);
2021-05-07 19:23:46 +00:00
} else if (groupId == R.id.group_style_font) {
return setFont(item);
2021-02-05 10:15:02 +00:00
} else if (groupId == R.id.group_style_align) {
return setAlignment(item);
} else if (groupId == R.id.group_style_list) {
2021-05-06 06:22:04 +00:00
if (item.getItemId() == R.id.menu_style_list_increase ||
item.getItemId() == R.id.menu_style_list_decrease)
return setListLevel(item);
else
return setList(item);
2021-02-05 10:15:02 +00:00
} else if (groupId == R.id.group_style_blockquote) {
return setBlockQuote(item);
} else if (groupId == R.id.group_style_strikethrough) {
return setStrikeThrough(item);
} else if (groupId == R.id.group_style_clear) {
return clear(item);
2020-06-28 21:36:31 +00:00
}
2021-02-05 10:15:02 +00:00
return false;
} catch (Throwable ex) {
Log.e(ex);
return false;
2020-06-28 21:36:31 +00:00
}
2021-02-05 10:15:02 +00:00
}
2020-06-28 16:39:27 +00:00
2021-02-05 10:15:02 +00:00
private boolean setSize(MenuItem item) {
2021-05-06 05:58:04 +00:00
RelativeSizeSpan[] spans = edit.getSpans(start, end, RelativeSizeSpan.class);
2021-02-05 10:15:02 +00:00
for (RelativeSizeSpan span : spans)
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2020-06-28 15:18:42 +00:00
2021-02-05 10:15:02 +00:00
Float size;
if (item.getItemId() == R.id.menu_style_size_small)
size = 0.8f;
else if (item.getItemId() == R.id.menu_style_size_large)
size = 1.25f;
else
size = null;
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
if (size != null)
2021-05-06 05:58:04 +00:00
edit.setSpan(new RelativeSizeSpan(size), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2019-09-27 17:05:34 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(start, end);
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2020-10-03 12:17:13 +00:00
2021-02-05 10:15:02 +00:00
private boolean setColor(MenuItem item) {
2021-04-27 14:40:27 +00:00
Helper.hideKeyboard(etBody);
2021-02-05 10:15:02 +00:00
Context context = etBody.getContext();
int editTextColor = Helper.resolveColor(context, android.R.attr.editTextColor);
ColorPickerDialogBuilder builder = ColorPickerDialogBuilder
.with(context)
.setTitle(R.string.title_color)
.showColorEdit(true)
.setColorEditTextColor(editTextColor)
.wheelType(ColorPickerView.WHEEL_TYPE.FLOWER)
.density(6)
.lightnessSliderOnly()
.setPositiveButton(android.R.string.ok, new ColorPickerClickListener() {
@Override
public void onClick(DialogInterface dialog, int selectedColor, Integer[] allColors) {
_setColor(selectedColor);
}
})
.setNegativeButton(R.string.title_reset, new DialogInterface.OnClickListener() {
@Override
public void onClick(DialogInterface dialog, int which) {
_setColor(null);
}
});
builder.build().show();
return true;
}
2020-10-03 09:37:43 +00:00
2021-02-05 10:15:02 +00:00
private void _setColor(Integer color) {
2021-05-06 05:58:04 +00:00
for (ForegroundColorSpan span : edit.getSpans(start, end, ForegroundColorSpan.class))
edit.removeSpan(span);
2020-10-03 09:37:43 +00:00
2021-02-05 10:15:02 +00:00
if (color != null)
2021-05-06 05:58:04 +00:00
edit.setSpan(new ForegroundColorSpan(color), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2020-10-03 09:37:43 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(start, end);
2021-02-05 10:15:02 +00:00
}
2020-10-03 09:37:43 +00:00
2021-02-05 10:15:02 +00:00
private boolean setAlignment(MenuItem item) {
2021-05-06 05:58:04 +00:00
Pair<Integer, Integer> paragraph = ensureParagraph(edit, start, end);
int s = paragraph.first;
int e = paragraph.second;
2021-02-05 10:15:02 +00:00
2021-05-06 05:58:04 +00:00
AlignmentSpan[] spans = edit.getSpans(s, e, AlignmentSpan.class);
2021-02-05 10:15:02 +00:00
for (AlignmentSpan span : spans)
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2021-02-05 10:15:02 +00:00
Layout.Alignment alignment = null;
boolean ltr = (TextUtils.getLayoutDirectionFromLocale(Locale.getDefault()) == View.LAYOUT_DIRECTION_LTR);
int itemId = item.getItemId();
if (itemId == R.id.menu_style_align_start) {
alignment = (ltr ? Layout.Alignment.ALIGN_NORMAL : Layout.Alignment.ALIGN_OPPOSITE);
} else if (itemId == R.id.menu_style_align_center) {
alignment = Layout.Alignment.ALIGN_CENTER;
} else if (itemId == R.id.menu_style_align_end) {
alignment = (ltr ? Layout.Alignment.ALIGN_OPPOSITE : Layout.Alignment.ALIGN_NORMAL);
2020-10-03 09:37:43 +00:00
}
2021-02-05 10:15:02 +00:00
if (alignment != null)
2021-05-06 05:58:04 +00:00
edit.setSpan(new AlignmentSpan.Standard(alignment),
s, e, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE | Spanned.SPAN_PARAGRAPH);
2020-08-12 08:25:08 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(s, e);
2020-08-14 18:32:05 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2020-08-12 06:18:39 +00:00
2021-05-06 06:22:04 +00:00
private boolean setListLevel(MenuItem item) {
Context context = etBody.getContext();
int add = (item.getItemId() == R.id.menu_style_list_increase ? 1 : -1);
boolean renum = false;
BulletSpan[] spans = edit.getSpans(start, end, BulletSpan.class);
for (BulletSpan span : spans)
if (span instanceof BulletSpanEx) {
BulletSpanEx bs = (BulletSpanEx) span;
bs.setLevel(bs.getLevel() + add);
} else if (span instanceof NumberSpan) {
renum = true;
NumberSpan ns = (NumberSpan) span;
ns.setLevel(ns.getLevel() + add);
}
if (renum)
renumber(edit, false, context);
etBody.setText(edit);
etBody.setSelection(start, end);
return true;
}
2021-02-05 10:15:02 +00:00
private boolean setList(MenuItem item) {
Context context = etBody.getContext();
int colorAccent = Helper.resolveColor(context, R.attr.colorAccent);
2021-05-07 12:09:16 +00:00
int bulletGap = context.getResources().getDimensionPixelSize(R.dimen.bullet_gap_size);
int bulletRadius = context.getResources().getDimensionPixelSize(R.dimen.bullet_radius_size);
int bulletIndent = context.getResources().getDimensionPixelSize(R.dimen.bullet_indent_size);
2021-02-05 10:15:02 +00:00
SharedPreferences prefs = PreferenceManager.getDefaultSharedPreferences(context);
int message_zoom = prefs.getInt("message_zoom", 100);
float textSize = Helper.getTextSize(context, 0) * message_zoom / 100f;
2021-05-06 05:58:04 +00:00
Pair<Integer, Integer> paragraph = ensureParagraph(edit, start, end);
int s = paragraph.first;
int e = paragraph.second;
2021-02-05 10:15:02 +00:00
// Remove existing bullets
2021-05-06 05:58:04 +00:00
BulletSpan[] spans = edit.getSpans(s, e, BulletSpan.class);
2021-02-05 10:15:02 +00:00
for (BulletSpan span : spans)
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2021-02-05 10:15:02 +00:00
2021-05-06 05:58:04 +00:00
int i = s;
int j = s + 1;
2021-02-05 10:15:02 +00:00
int index = 1;
2021-05-06 05:58:04 +00:00
while (j < e) {
if (i > 0 && edit.charAt(i - 1) == '\n' && edit.charAt(j) == '\n') {
Log.i("Insert " + i + "..." + (j + 1) + " size=" + e);
2021-02-05 10:15:02 +00:00
if (item.getItemId() == R.id.menu_style_list_bullets)
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P)
2021-05-07 12:09:16 +00:00
edit.setSpan(new BulletSpanEx(bulletIndent, bulletGap, colorAccent, 0), i, j + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE | Spanned.SPAN_PARAGRAPH);
2020-08-12 06:18:39 +00:00
else
2021-05-07 12:09:16 +00:00
edit.setSpan(new BulletSpanEx(bulletIndent, bulletGap, colorAccent, bulletRadius, 0), i, j + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE | Spanned.SPAN_PARAGRAPH);
2021-05-06 18:46:50 +00:00
else
2021-05-07 12:09:16 +00:00
edit.setSpan(new NumberSpan(bulletIndent, bulletGap, colorAccent, textSize, 0, index++), i, j + 1, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE | Spanned.SPAN_PARAGRAPH);
2020-08-12 06:18:39 +00:00
2021-02-05 10:15:02 +00:00
i = j + 1;
2020-08-11 16:46:26 +00:00
}
2021-02-05 10:15:02 +00:00
j++;
2020-08-11 16:46:26 +00:00
}
2021-05-06 18:46:50 +00:00
renumber(edit, false, context);
2021-05-06 06:22:04 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(s, e);
2020-06-28 21:36:31 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
private boolean setFont(MenuItem item) {
2021-05-06 05:58:04 +00:00
TypefaceSpan[] spans = edit.getSpans(start, end, TypefaceSpan.class);
2021-02-05 10:15:02 +00:00
for (TypefaceSpan span : spans)
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2020-06-28 21:36:31 +00:00
2021-02-05 10:15:02 +00:00
int id = item.getItemId();
String[] names = anchor.getResources().getStringArray(R.array.fontNameValues);
String face = (id < names.length ? names[id] : null);
2020-06-28 21:36:31 +00:00
2021-02-05 10:15:02 +00:00
if (face != null)
2021-05-06 05:58:04 +00:00
edit.setSpan(new TypefaceSpan(face), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2020-06-28 21:36:31 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(start, end);
2020-10-01 08:22:20 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2020-11-17 20:22:02 +00:00
2021-02-05 10:15:02 +00:00
private boolean setBlockQuote(MenuItem item) {
Context context = etBody.getContext();
2020-10-01 08:22:20 +00:00
2021-02-05 10:15:02 +00:00
int colorPrimary = Helper.resolveColor(context, R.attr.colorPrimary);
2021-05-07 12:19:59 +00:00
int quoteGap = context.getResources().getDimensionPixelSize(R.dimen.quote_gap_size);
int quoteStripe = context.getResources().getDimensionPixelSize(R.dimen.quote_stripe_width);
2020-10-01 08:22:20 +00:00
2021-05-06 05:58:04 +00:00
Pair<Integer, Integer> paragraph = ensureParagraph(edit, start, end);
int s = paragraph.first;
int e = paragraph.second;
2020-10-01 08:22:20 +00:00
2021-05-06 05:58:04 +00:00
QuoteSpan[] spans = edit.getSpans(s, e, QuoteSpan.class);
2021-02-05 10:15:02 +00:00
for (QuoteSpan span : spans)
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2020-10-01 08:22:20 +00:00
2021-02-05 10:15:02 +00:00
QuoteSpan q;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P)
q = new QuoteSpan(colorPrimary);
else
2021-05-07 12:19:59 +00:00
q = new QuoteSpan(colorPrimary, quoteStripe, quoteGap);
2021-05-06 05:58:04 +00:00
edit.setSpan(q, s, e, Spanned.SPAN_INCLUSIVE_EXCLUSIVE);
2020-10-01 08:31:58 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(s, e);
2020-10-01 08:31:58 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2020-10-01 08:31:58 +00:00
2021-02-05 10:15:02 +00:00
private boolean setStrikeThrough(MenuItem item) {
boolean has = false;
2021-05-06 05:58:04 +00:00
for (StrikethroughSpan span : edit.getSpans(start, end, StrikethroughSpan.class)) {
2021-02-05 10:15:02 +00:00
has = true;
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2020-10-01 08:31:58 +00:00
}
2021-02-05 10:15:02 +00:00
if (!has)
2021-05-06 05:58:04 +00:00
edit.setSpan(new StrikethroughSpan(), start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2020-11-17 20:55:02 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(start, end);
2020-11-17 20:55:02 +00:00
2021-02-05 10:15:02 +00:00
return true;
}
2020-11-17 19:22:34 +00:00
2021-02-05 10:15:02 +00:00
private boolean clear(MenuItem item) {
2021-05-06 05:58:04 +00:00
int e = end;
2020-06-28 21:36:31 +00:00
2021-02-05 10:15:02 +00:00
// Expand to paragraph (block quotes)
2021-05-06 05:58:04 +00:00
if (e + 1 < edit.length() && edit.charAt(e) == '\n')
e++;
2020-06-28 21:36:31 +00:00
2021-05-06 05:58:04 +00:00
for (Object span : edit.getSpans(start, e, Object.class))
2021-02-05 10:15:02 +00:00
if (!(span instanceof ImageSpan)) {
2021-05-06 05:58:04 +00:00
int sstart = edit.getSpanStart(span);
int send = edit.getSpanEnd(span);
int flags = edit.getSpanFlags(span);
2021-02-05 10:15:02 +00:00
if (sstart < start && send > start)
2021-05-06 05:58:04 +00:00
setSpan(edit, span, sstart, start, flags, etBody.getContext());
2021-02-05 10:15:02 +00:00
if (sstart < end && send > end)
2021-05-06 05:58:04 +00:00
setSpan(edit, span, e, send, flags, etBody.getContext());
2020-06-28 21:36:31 +00:00
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2021-02-05 10:15:02 +00:00
}
2019-09-27 17:05:34 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
etBody.setSelection(start, e);
2019-12-29 12:13:52 +00:00
2021-02-05 10:15:02 +00:00
return true;
2019-09-27 17:05:34 +00:00
}
2021-02-05 10:15:02 +00:00
});
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
popupMenu.show();
2019-12-29 12:13:52 +00:00
2021-02-05 10:15:02 +00:00
return true;
} else if (action == R.id.menu_link) {
String url = (String) args[0];
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
List<Object> spans = new ArrayList<>();
2021-05-06 05:58:04 +00:00
for (Object span : edit.getSpans(start, end, Object.class)) {
2021-02-05 10:15:02 +00:00
if (!(span instanceof URLSpan))
spans.add(span);
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2021-02-05 10:15:02 +00:00
}
2019-09-27 17:05:34 +00:00
2021-02-05 10:15:02 +00:00
if (url != null) {
2021-05-06 05:58:04 +00:00
int e = end;
2021-02-05 10:15:02 +00:00
if (start == end) {
etBody.getText().insert(start, url);
2021-05-06 05:58:04 +00:00
e += url.length();
2021-02-05 10:15:02 +00:00
}
2019-09-27 17:05:34 +00:00
2021-05-06 05:58:04 +00:00
edit.setSpan(new URLSpan(url), start, e, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2019-09-27 17:05:34 +00:00
}
2021-02-05 10:15:02 +00:00
// Restore other spans
for (Object span : spans)
2021-05-06 05:58:04 +00:00
edit.setSpan(span, start, end, Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
2019-09-29 10:10:19 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
2021-02-05 10:15:02 +00:00
etBody.setSelection(end, end);
2019-09-29 10:10:19 +00:00
2021-02-05 10:15:02 +00:00
return true;
} else if (action == R.id.menu_clear) {
2021-05-06 05:58:04 +00:00
for (Object span : edit.getSpans(0, etBody.length(), Object.class))
2021-02-05 10:15:02 +00:00
if (!(span instanceof ImageSpan))
2021-05-06 05:58:04 +00:00
edit.removeSpan(span);
2021-02-05 10:15:02 +00:00
2021-05-06 05:58:04 +00:00
etBody.setText(edit);
2021-02-05 10:15:02 +00:00
etBody.setSelection(start, end);
2019-09-29 10:10:19 +00:00
2021-02-05 10:15:02 +00:00
return true;
2019-09-27 17:05:34 +00:00
}
2021-02-05 10:15:02 +00:00
return false;
2019-09-27 17:05:34 +00:00
} catch (Throwable ex) {
Log.e(ex);
return false;
}
}
2020-11-17 18:37:56 +00:00
2021-05-06 05:58:04 +00:00
static void setSpan(Editable edit, Object span, int start, int end, int flags, Context context) {
2020-11-17 19:22:34 +00:00
if (span instanceof CharacterStyle)
2021-05-06 05:58:04 +00:00
edit.setSpan(CharacterStyle.wrap((CharacterStyle) span), start, end, flags);
2020-11-17 19:22:34 +00:00
else if (span instanceof QuoteSpan) {
2020-11-17 20:22:02 +00:00
ParagraphStyle ps = (ParagraphStyle) span;
2021-05-06 05:58:04 +00:00
Pair<Integer, Integer> p = ensureParagraph(edit, start, end);
edit.setSpan(clone(span, ps.getClass(), context), p.first, p.second, flags);
2020-11-17 20:22:02 +00:00
}
}
2021-05-06 05:58:04 +00:00
static private Pair<Integer, Integer> ensureParagraph(Editable edit, int s, int e) {
2020-11-17 20:22:02 +00:00
int start = s;
int end = e;
// Expand selection at start
2021-05-06 05:58:04 +00:00
while (start > 0 && edit.charAt(start - 1) != '\n')
2020-11-17 20:22:02 +00:00
start--;
// Expand selection at end
2021-05-06 05:58:04 +00:00
while (end > 0 && end < edit.length() && edit.charAt(end - 1) != '\n')
2020-11-17 20:22:02 +00:00
end++;
// Nothing to do
if (start == end)
return null;
// Create paragraph at start
2021-05-06 05:58:04 +00:00
if (start == 0 && edit.charAt(start) != '\n') {
edit.insert(0, "\n");
2020-11-17 20:22:02 +00:00
start++;
end++;
2020-11-17 19:22:34 +00:00
}
2020-11-17 20:22:02 +00:00
// Create paragraph at end
2021-05-06 05:58:04 +00:00
if (end == edit.length() && edit.charAt(end - 1) != '\n') {
edit.append("\n");
2020-11-17 20:22:02 +00:00
end++;
}
2021-05-06 05:58:04 +00:00
if (end == edit.length())
edit.append("\n"); // workaround Android bug
2020-11-17 20:22:02 +00:00
2021-05-06 06:22:04 +00:00
return new Pair<>(start, end);
2020-11-17 19:22:34 +00:00
}
2020-11-17 18:37:56 +00:00
static <T extends ParagraphStyle> T clone(Object span, Class<T> type, Context context) {
if (QuoteSpan.class.isAssignableFrom(type)) {
QuoteSpan q = (QuoteSpan) span;
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P)
return (T) new QuoteSpan(q.getColor());
else
return (T) new QuoteSpan(q.getColor(), q.getStripeWidth(), q.getGapWidth());
} else if (NumberSpan.class.isAssignableFrom(type)) {
NumberSpan n = (NumberSpan) span;
2021-05-07 12:09:16 +00:00
int bulletGap = context.getResources().getDimensionPixelSize(R.dimen.bullet_gap_size);
int bulletIndent = context.getResources().getDimensionPixelSize(R.dimen.bullet_indent_size);
2020-11-17 18:37:56 +00:00
int colorAccent = Helper.resolveColor(context, R.attr.colorAccent);
2021-05-07 12:09:16 +00:00
return (T) new NumberSpan(bulletIndent, bulletGap, colorAccent, n.getTextSize(), n.getLevel(), n.getIndex() + 1);
2021-05-06 06:22:04 +00:00
} else if (BulletSpanEx.class.isAssignableFrom(type)) {
BulletSpanEx b = (BulletSpanEx) span;
2021-05-07 12:09:16 +00:00
int bulletIndent = context.getResources().getDimensionPixelSize(R.dimen.bullet_indent_size);
2020-11-17 18:37:56 +00:00
if (Build.VERSION.SDK_INT < Build.VERSION_CODES.P) {
int colorAccent = Helper.resolveColor(context, R.attr.colorAccent);
2021-05-07 12:09:16 +00:00
int bulletGap = context.getResources().getDimensionPixelSize(R.dimen.bullet_gap_size);
return (T) new BulletSpanEx(bulletIndent, bulletGap, colorAccent, b.getLevel());
2020-11-17 18:37:56 +00:00
} else
2021-05-07 12:09:16 +00:00
return (T) new BulletSpanEx(bulletIndent, b.getGapWidth(), b.getColor(), b.getBulletRadius(), b.getLevel());
2020-11-17 18:37:56 +00:00
} else
throw new IllegalArgumentException(type.getName());
}
static void renumber(Editable text, boolean clean, Context context) {
2021-05-07 12:09:16 +00:00
int bulletGap = context.getResources().getDimensionPixelSize(R.dimen.bullet_gap_size);
int bulletIndent = context.getResources().getDimensionPixelSize(R.dimen.bullet_indent_size);
2020-11-17 18:37:56 +00:00
int colorAccent = Helper.resolveColor(context, R.attr.colorAccent);
Log.i("Renumber clean=" + clean + " text=" + text);
int next;
int pos = -1;
2021-05-06 06:22:04 +00:00
List<Integer> levels = new ArrayList<>();
2020-11-17 18:37:56 +00:00
for (int i = 0; i < text.length(); i = next) {
next = text.nextSpanTransition(i, text.length(), NumberSpan.class);
Log.i("Bullet span next=" + next);
BulletSpan[] spans = text.getSpans(i, next, BulletSpan.class);
for (BulletSpan span : spans) {
int start = text.getSpanStart(span);
int end = text.getSpanEnd(span);
int flags = text.getSpanFlags(span);
Log.i("Bullet span " + start + "..." + end);
if (clean && start == end) {
text.removeSpan(span);
continue;
}
2021-05-06 06:22:04 +00:00
int level;
if (span instanceof NumberSpan)
level = ((NumberSpan) span).getLevel();
else if (span instanceof BulletSpanEx)
level = ((BulletSpanEx) span).getLevel();
else
level = 0;
if (start != pos)
levels.clear();
while (levels.size() > level + 1)
levels.remove(levels.size() - 1);
if (levels.size() == level + 1 && !(span instanceof NumberSpan))
2021-05-06 18:46:50 +00:00
levels.remove(level);
2021-05-06 06:22:04 +00:00
while (levels.size() < level + 1)
levels.add(0);
int index = levels.get(level) + 1;
levels.remove(level);
levels.add(level, index);
2020-11-17 18:37:56 +00:00
2021-05-06 06:22:04 +00:00
if (span instanceof NumberSpan) {
2020-11-17 18:37:56 +00:00
NumberSpan ns = (NumberSpan) span;
if (index != ns.getIndex()) {
text.removeSpan(span);
2021-05-06 06:22:04 +00:00
// Text size needs measuring
2021-05-07 12:09:16 +00:00
NumberSpan clone = new NumberSpan(bulletIndent, bulletGap, colorAccent, ns.getTextSize(), level, index);
2020-11-17 18:37:56 +00:00
text.setSpan(clone, start, end, flags);
}
pos = end;
}
}
}
}
//TextUtils.dumpSpans(text, new LogPrinter(android.util.Log.INFO, "FairEmail"), "afterTextChanged ");
2019-09-27 17:05:34 +00:00
}