package net.ktnx.mobileledger.model;
import android.content.res.Resources;
+import android.graphics.Color;
+import android.graphics.Typeface;
+import android.text.SpannableString;
+import android.text.Spanned;
+import android.text.style.ForegroundColorSpan;
+import android.text.style.StyleSpan;
+import android.text.style.UnderlineSpan;
import androidx.annotation.NonNull;
import org.jetbrains.annotations.Contract;
import org.jetbrains.annotations.NotNull;
+import java.util.Objects;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
import java.util.regex.PatternSyntaxException;
TemplateAccount pa = (TemplateAccount) p;
AccountRow acc = createAccountRow();
acc.setId(pa.getId());
+ acc.setPosition(pa.getPosition());
if (pa.getAccountNameMatchGroup() == null)
acc.setAccountName(Misc.nullIsEmpty(pa.getAccountName()));
result.setValue(initialValue);
return result;
}
+ public void copyFrom(@NonNull PossiblyMatchedValue<T> origin) {
+ literalValue = origin.literalValue;
+ value = origin.value;
+ matchGroup = origin.matchGroup;
+ }
public T getValue() {
if (!literalValue)
throw new IllegalStateException("Value is not literal");
private final PossiblyMatchedValue<String> accountComment =
PossiblyMatchedValue.withLiteralString("");
private final PossiblyMatchedValue<Float> amount =
- PossiblyMatchedValue.withLiteralFloat(0f);
+ PossiblyMatchedValue.withLiteralFloat(null);
private final PossiblyMatchedValue<Currency> currency = new PossiblyMatchedValue<>();
private boolean negateAmount;
- private AccountRow() {
+ public AccountRow() {
+ super(Type.ACCOUNT_ITEM);
+ }
+ public AccountRow(AccountRow origin) {
super(Type.ACCOUNT_ITEM);
+ id = origin.id;
+ position = origin.position;
+ accountName.copyFrom(origin.accountName);
+ accountComment.copyFrom(origin.accountComment);
+ amount.copyFrom(origin.amount);
+ currency.copyFrom(origin.currency);
+ negateAmount = origin.negateAmount;
}
public boolean isNegateAmount() {
return negateAmount;
public static class Header extends TemplateDetailsItem {
private String pattern = "";
private String testText = "";
+ private String name = "";
private Pattern compiledPattern;
private String patternError;
- private String name = "";
private PossiblyMatchedValue<String> transactionDescription =
PossiblyMatchedValue.withLiteralString("");
private PossiblyMatchedValue<String> transactionComment =
private PossiblyMatchedValue<Integer> dateYear = PossiblyMatchedValue.withLiteralInt(null);
private PossiblyMatchedValue<Integer> dateMonth = PossiblyMatchedValue.withLiteralInt(null);
private PossiblyMatchedValue<Integer> dateDay = PossiblyMatchedValue.withLiteralInt(null);
+ private SpannableString testMatch;
private Header() {
super(Type.HEADER);
}
id = origin.id;
name = origin.name;
testText = origin.testText;
+ testMatch = origin.testMatch;
setPattern(origin.pattern);
transactionDescription = new PossiblyMatchedValue<>(origin.transactionDescription);
dateMonth = new PossiblyMatchedValue<>(origin.dateMonth);
dateDay = new PossiblyMatchedValue<>(origin.dateDay);
}
+ private static StyleSpan capturedSpan() { return new StyleSpan(Typeface.BOLD); }
+ private static UnderlineSpan matchedSpan() { return new UnderlineSpan(); }
+ private static ForegroundColorSpan notMatchedSpan() {
+ return new ForegroundColorSpan(Color.GRAY);
+ }
public String getName() {
return name;
}
}
public void setPattern(String pattern) {
this.pattern = pattern;
- if (pattern != null) {
- try {
- this.compiledPattern = Pattern.compile(pattern);
- this.patternError = null;
- }
- catch (PatternSyntaxException e) {
- this.compiledPattern = null;
- this.patternError = e.getMessage();
- }
+ try {
+ this.compiledPattern = Pattern.compile(pattern);
+ checkPatternMatch();
}
- else {
- patternError = "Missing pattern";
+ catch (PatternSyntaxException ex) {
+ patternError = ex.getDescription();
+ compiledPattern = null;
+
+ testMatch = new SpannableString(testText);
+ testMatch.setSpan(notMatchedSpan(), 0, testText.length() - 1,
+ Spanned.SPAN_INCLUSIVE_INCLUSIVE);
}
}
@NonNull
}
public void setTestText(String testText) {
this.testText = testText;
+
+ checkPatternMatch();
}
public String getTransactionDescription() {
return transactionDescription.getValue();
return true;
return Misc.equalStrings(name, o.name) && Misc.equalStrings(pattern, o.pattern) &&
- Misc.equalStrings(testText, o.testText);
+ Misc.equalStrings(testText, o.testText) &&
+ Misc.equalStrings(patternError, o.patternError) &&
+ Objects.equals(testMatch, o.testMatch);
}
public String getMatchGroupText(int group) {
if (compiledPattern != null && testText != null) {
return result;
}
+ public SpannableString getTestMatch() {
+ return testMatch;
+ }
+ public void checkPatternMatch() {
+ patternError = null;
+ testMatch = null;
+
+ if (pattern != null) {
+ try {
+ if (Misc.emptyIsNull(testText) != null) {
+ SpannableString ss = new SpannableString(testText);
+ Matcher m = compiledPattern.matcher(testText);
+ if (m.find()) {
+ if (m.start() > 0)
+ ss.setSpan(notMatchedSpan(), 0, m.start(),
+ Spanned.SPAN_INCLUSIVE_INCLUSIVE);
+ if (m.end() < testText.length() - 1)
+ ss.setSpan(notMatchedSpan(), m.end(), testText.length(),
+ Spanned.SPAN_INCLUSIVE_INCLUSIVE);
+
+ ss.setSpan(matchedSpan(), m.start(0), m.end(0),
+ Spanned.SPAN_INCLUSIVE_INCLUSIVE);
+
+ if (m.groupCount() > 0) {
+ for (int g = 1; g <= m.groupCount(); g++) {
+ ss.setSpan(capturedSpan(), m.start(g), m.end(g),
+ Spanned.SPAN_INCLUSIVE_INCLUSIVE);
+ }
+ }
+ }
+ else {
+ patternError = "Pattern does not match";
+ ss.setSpan(new ForegroundColorSpan(Color.GRAY), 0,
+ testText.length() - 1, Spanned.SPAN_INCLUSIVE_INCLUSIVE);
+ }
+
+ testMatch = ss;
+ }
+ }
+ catch (PatternSyntaxException e) {
+ this.compiledPattern = null;
+ this.patternError = e.getMessage();
+ }
+ }
+ else {
+ patternError = "Missing pattern";
+ }
+ }
+ public String getPatternError() {
+ return patternError;
+ }
+ public SpannableString testMatch() {
+ return testMatch;
+ }
}
}