Fix positioning bug when toggling a checkbox

This commit is contained in:
Stefan Niedermann 2021-01-04 00:41:32 +01:00
parent 9d5284d792
commit 49847ed53a
5 changed files with 57 additions and 79 deletions

View file

@ -84,67 +84,13 @@ public class NotePreviewFragment extends SearchableBaseNoteFragment implements O
public View onCreateView(@NonNull LayoutInflater inflater, @Nullable ViewGroup
container, @Nullable Bundle savedInstanceState) {
binding = FragmentNotePreviewBinding.inflate(inflater, container, false);
binding.singleNoteContent.getMarkdownString().observe(requireActivity(), (newContent) -> {
changedText = newContent.toString();
binding.singleNoteContent.setMarkdownString(newContent);
saveNote(null);
});
return binding.getRoot();
}
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
// MarkDownUtil.getMarkDownConfiguration(binding.singleNoteContent.getContext())
// .setOnTodoClickCallback((view, line, lineNumber) -> {
// try {
// String[] lines = TextUtils.split(note.getContent(), "\\r?\\n");
// /*
// * Workaround for RxMarkdown-bug:
// * When (un)checking a checkbox in a note which contains code-blocks, the "`"-characters get stripped out in the TextView and therefore the given lineNumber is wrong
// * Find number of lines starting with ``` before lineNumber
// */
// boolean inCodefence = false;
// for (int i = 0; i < lines.length; i++) {
// if (lines[i].startsWith("```")) {
// inCodefence = !inCodefence;
// lineNumber++;
// }
// if (inCodefence && TextUtils.isEmpty(lines[i])) {
// lineNumber++;
// }
// if (i == lineNumber) {
// break;
// }
// }
//
// /*
// * Workaround for multiple RxMarkdown-bugs:
// * When (un)checking a checkbox which is in the last line, every time it gets toggled, the last character of the line gets lost.
// * When (un)checking a checkbox, every markdown gets stripped in the given line argument
// */
// if (lines[lineNumber].startsWith(CHECKBOX_UNCHECKED_MINUS) || lines[lineNumber].startsWith(CHECKBOX_UNCHECKED_STAR)) {
// lines[lineNumber] = lines[lineNumber].replace(CHECKBOX_UNCHECKED_MINUS, CHECKBOX_CHECKED_MINUS);
// lines[lineNumber] = lines[lineNumber].replace(CHECKBOX_UNCHECKED_STAR, CHECKBOX_CHECKED_STAR);
// } else {
// lines[lineNumber] = lines[lineNumber].replace(CHECKBOX_CHECKED_MINUS, CHECKBOX_UNCHECKED_MINUS);
// lines[lineNumber] = lines[lineNumber].replace(CHECKBOX_CHECKED_STAR, CHECKBOX_UNCHECKED_STAR);
// }
//
// changedText = TextUtils.join("\n", lines);
// binding.singleNoteContent.setText(parseCompat(markdownProcessor, changedText));
// saveNote(null);
// } catch (IndexOutOfBoundsException e) {
// Toast.makeText(getActivity(), R.string.checkbox_could_not_be_toggled, Toast.LENGTH_SHORT).show();
// e.printStackTrace();
// }
// return line;
// }
// )
// .build());
TextProcessorChain chain = defaultTextProcessorChain(note);
final TextProcessorChain chain = defaultTextProcessorChain(note);
binding.singleNoteContent.setMarkdownString(chain.apply(note.getContent()));
changedText = note.getContent();
binding.singleNoteContent.setMovementMethod(LinkMovementMethod.getInstance());
@ -152,11 +98,16 @@ public class NotePreviewFragment extends SearchableBaseNoteFragment implements O
db = NotesDatabase.getInstance(requireContext());
binding.swiperefreshlayout.setOnRefreshListener(this);
SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(requireActivity().getApplicationContext());
final SharedPreferences sp = PreferenceManager.getDefaultSharedPreferences(requireActivity().getApplicationContext());
binding.singleNoteContent.setTextSize(TypedValue.COMPLEX_UNIT_PX, getFontSizeFromPreferences(requireContext(), sp));
if (sp.getBoolean(getString(R.string.pref_key_font), false)) {
binding.singleNoteContent.setTypeface(Typeface.MONOSPACE);
}
binding.singleNoteContent.getMarkdownString().observe(requireActivity(), (newContent) -> {
changedText = newContent.toString();
saveNote(null);
});
}
@Override

View file

@ -87,6 +87,30 @@ public class MarkwonMarkdownUtilTest extends TestCase {
@Test
public void testLineStartsWithCheckbox() {
final Map<String, Boolean> lines = new HashMap<>();
lines.put(" - [ ] a", true);
lines.put(" - [x] a", true);
lines.put(" * [ ] a", true);
lines.put(" * [x] a", true);
lines.put(" + [ ] a", true);
lines.put(" + [x] a", true);
lines.put("- [ ] a", true);
lines.put("- [x] a", true);
lines.put("* [ ] a", true);
lines.put("* [x] a", true);
lines.put("+ [ ] a", true);
lines.put("+ [x] a", true);
lines.put(" - [ ] ", true);
lines.put(" - [x] ", true);
lines.put(" * [ ] ", true);
lines.put(" * [x] ", true);
lines.put(" + [ ] ", true);
lines.put(" + [x] ", true);
lines.put(" - [ ]", true);
lines.put(" - [x]", true);
lines.put(" * [ ]", true);
lines.put(" * [x]", true);
lines.put(" + [ ]", true);
lines.put(" + [x]", true);
lines.put("- [ ] ", true);
lines.put("- [x] ", true);
lines.put("* [ ] ", true);

View file

@ -109,6 +109,25 @@ public class MarkwonMarkdownUtil {
return null;
}
public static CharSequence setCheckboxStatus(@NonNull String markdownString, int targetCheckboxIndex, boolean newCheckedState) {
final String[] lines = markdownString.split("\n");
int checkboxIndex = 0;
for (int i = 0; i < lines.length; i++) {
if (lineStartsWithCheckbox(lines[i]) && lines[i].trim().length() > EListType.DASH.checkboxChecked.length()) {
if (checkboxIndex == targetCheckboxIndex) {
final int indexOfStartingBracket = lines[i].indexOf("[");
final String toggledLine = lines[i].substring(0, indexOfStartingBracket + 1) +
(newCheckedState ? 'x' : ' ') +
lines[i].substring(indexOfStartingBracket + 2);
lines[i] = toggledLine;
break;
}
checkboxIndex++;
}
}
return TextUtils.join("\n", lines);
}
public static boolean lineStartsWithCheckbox(@NonNull String line) {
for (EListType listType : EListType.values()) {
if (lineStartsWithCheckbox(line, listType)) {
@ -119,7 +138,8 @@ public class MarkwonMarkdownUtil {
}
public static boolean lineStartsWithCheckbox(@NonNull String line, @NonNull EListType listType) {
return line.startsWith(listType.checkboxUnchecked) || line.startsWith(listType.checkboxChecked);
final String trimmedLine = line.trim();
return (trimmedLine.startsWith(listType.checkboxUnchecked) || trimmedLine.startsWith(listType.checkboxChecked));
}
/**

View file

@ -46,28 +46,12 @@ public class MarkwonMarkdownViewer extends AppCompatTextView implements Markdown
super(context, attrs, defStyleAttr);
this.markwon = MarkwonMarkdownUtil.initMarkwonViewer(context)
.usePlugin(new ToggleableTaskListPlugin((toggledCheckboxPosition, newCheckedState) -> {
// TODO move logic to MarkwonMarkdownUtil
Log.v(TAG, "new text: " + toggledCheckboxPosition);
final CharSequence unrenderedText = unrenderedText$.getValue();
if (unrenderedText == null) {
final CharSequence oldUnrenderedText = unrenderedText$.getValue();
if (oldUnrenderedText == null) {
throw new IllegalStateException("Checkbox #" + toggledCheckboxPosition + ", but unrenderedText$ value is null.");
}
final String[] lines = unrenderedText.toString().split("\n");
int checkboxIndex = 0;
for (int i = 0; i < lines.length; i++) {
if (MarkwonMarkdownUtil.lineStartsWithCheckbox(lines[i].trim())) {
if (checkboxIndex == toggledCheckboxPosition) {
final int indexOfStartingBracket = lines[i].indexOf("[");
final String toggledLine = lines[i].substring(0, indexOfStartingBracket + 1) +
(newCheckedState ? 'x' : ' ') +
lines[i].substring(indexOfStartingBracket + 2);
lines[i] = toggledLine;
break;
}
checkboxIndex++;
}
}
this.unrenderedText$.setValue(TextUtils.join("\n", lines));
final CharSequence newUnrenderedText = MarkwonMarkdownUtil.setCheckboxStatus(oldUnrenderedText.toString(), toggledCheckboxPosition, newCheckedState);
this.setMarkdownString(newUnrenderedText);
}))
.build();
this.renderService = Executors.newSingleThreadExecutor();

View file

@ -11,7 +11,6 @@ import androidx.annotation.NonNull;
import java.util.Arrays;
import java.util.function.BiConsumer;
import java.util.function.Consumer;
import io.noties.markwon.ext.tasklist.TaskListSpan;
@ -33,7 +32,7 @@ public class ToggleTaskListSpan extends ClickableSpan {
public void onClick(@NonNull View widget) {
span.setDone(!span.isDone());
widget.invalidate();
Log.i(TAG, "task-list click, isDone: " + span.isDone() + ", content: '" + content + "'");
Log.v(TAG, "task-list click, isDone: " + span.isDone() + ", content: '" + content + "'");
// it must be a TextView
final TextView textView = (TextView) widget;