Added filtering for on-device searching

pull/209/head
M66B 2 years ago
parent 2be041841a
commit 781738089f

@ -149,13 +149,13 @@ For the record the stack trace:
``` ```
android.database.sqlite.SQLiteDiskIOException: disk I/O error (code 778) android.database.sqlite.SQLiteDiskIOException: disk I/O error (code 778)
at io.requery.android.database.sqlite.SQLiteConnection.nativeExecute(SourceFile:-2) at io.requery.android.database.sqlite.SQLiteConnection.nativeExecute(SourceFile:-2)
at io.requery.android.database.sqlite.SQLiteConnection.execute(SQLiteConnection:595) at io.requery.android.database.sqlite.SQLiteConnection.execute(SQLiteConnection:595)
at io.requery.android.database.sqlite.SQLiteSession.endTransactionUnchecked(SQLiteSession:447) at io.requery.android.database.sqlite.SQLiteSession.endTransactionUnchecked(SQLiteSession:447)
at io.requery.android.database.sqlite.SQLiteSession.endTransaction(SQLiteSession:411) at io.requery.android.database.sqlite.SQLiteSession.endTransaction(SQLiteSession:411)
at io.requery.android.database.sqlite.SQLiteDatabase.endTransaction(SQLiteDatabase:551) at io.requery.android.database.sqlite.SQLiteDatabase.endTransaction(SQLiteDatabase:551)
at androidx.room.RoomDatabase.internalEndTransaction(RoomDatabase:594) at androidx.room.RoomDatabase.internalEndTransaction(RoomDatabase:594)
at androidx.room.RoomDatabase.endTransaction(RoomDatabase:584) at androidx.room.RoomDatabase.endTransaction(RoomDatabase:584)
``` ```
The cause might be [changes in Android 7 Nougat](https://ericsink.com/entries/sqlite_android_n.html), which is why sqlite isn't bundled anymore since version 1.1970. The cause might be [changes in Android 7 Nougat](https://ericsink.com/entries/sqlite_android_n.html), which is why sqlite isn't bundled anymore since version 1.1970.
@ -1158,16 +1158,16 @@ keyword:<keyword>
There should be no space between the prefix and the search term, which will be applied as an AND-condition. There should be no space between the prefix and the search term, which will be applied as an AND-condition.
Search expressions can be used for searching on the email server only, and not for searching on the device. Only AND conditions (+) and NOT conditions (-) can be used for on-device searching (since version 1.1981).
If you try to use other search expressions, you get the error *Select a folder for a complex search*,
which means that a folder in an account's folder list must be selected in order to perform the search on the server.
Since version 1.1733 it is possible to save searches, which means that a named entry in the navigation menu will be created to repeat the same search later. Since version 1.1733 it is possible to save searches, which means that a named entry in the navigation menu will be created to repeat the same search later.
You can save a search after searching by tapping on the save button in the top action bar. You can save a search after searching by tapping on the save button in the top action bar.
After repeating a search there will be a delete button at the same place to delete a saved search again. After repeating a search there will be a delete button at the same place to delete a saved search again.
A saved search might be useful to quickly search for starred messages, or for messages from a specific email address, etc. A saved search might be useful to quickly search for starred messages, or for messages from a specific email address, etc.
Searching on the device is a free feature, using the search index and searching on the server is a pro feature. Using the search index is a pro feature.
Note that you can download as many messages to your device as you like.
The easiest way is to use the menu item *Fetch more messages* in the three-dots menu of the start screen.
<br /> <br />

@ -60,6 +60,7 @@ import java.util.List;
import java.util.Map; import java.util.Map;
import java.util.Objects; import java.util.Objects;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.regex.Pattern;
import javax.mail.Address; import javax.mail.Address;
import javax.mail.FetchProfile; import javax.mail.FetchProfile;
@ -269,10 +270,25 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
int found = 0; int found = 0;
if (criteria.fts && criteria.query != null) { List<String> word = new ArrayList<>();
List<String> plus = new ArrayList<>();
List<String> minus = new ArrayList<>();
if (criteria.query != null) {
for (String w : criteria.query.trim().split("\\s+"))
if (w.length() > 1 && w.startsWith("+"))
plus.add(w.substring(1));
else if (w.length() > 1 && w.startsWith("-"))
minus.add(w.substring(1));
else
word.add(w);
if (word.size() == 0 && plus.size() > 0)
word.add(plus.get(0));
}
if (criteria.fts && word.size() > 0) {
if (state.ids == null) { if (state.ids == null) {
SQLiteDatabase sdb = Fts4DbHelper.getInstance(context); SQLiteDatabase sdb = Fts4DbHelper.getInstance(context);
state.ids = Fts4DbHelper.match(sdb, account, folder, exclude, criteria); state.ids = Fts4DbHelper.match(sdb, account, folder, exclude, criteria, TextUtils.join(" ", word));
EntityLog.log(context, "Boundary FTS " + EntityLog.log(context, "Boundary FTS " +
" account=" + account + " account=" + account +
" folder=" + folder + " folder=" + folder +
@ -335,21 +351,21 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
} }
if (!matched && criteria.in_subject) { if (!matched && criteria.in_subject) {
if (contains(message.subject, criteria.query)) if (contains(message.subject, criteria.query, false))
matched = true; matched = true;
} }
if (!matched && criteria.in_keywords) { if (!matched && criteria.in_keywords) {
if (message.keywords != null) if (message.keywords != null)
for (String keyword : message.keywords) for (String keyword : message.keywords)
if (contains(keyword, criteria.query)) { if (contains(keyword, criteria.query, false)) {
matched = true; matched = true;
break; break;
} }
} }
if (!matched && criteria.in_notes) { if (!matched && criteria.in_notes) {
if (contains(message.notes, criteria.query)) if (contains(message.notes, criteria.query, false))
matched = true; matched = true;
} }
@ -358,9 +374,9 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
File file = EntityMessage.getFile(context, id); File file = EntityMessage.getFile(context, id);
if (file.exists()) { if (file.exists()) {
String html = Helper.readText(file); String html = Helper.readText(file);
if (contains(html, criteria.query)) { if (contains(html, criteria.query, true)) {
String text = HtmlHelper.getFullText(html); String text = HtmlHelper.getFullText(html);
if (contains(text, criteria.query)) if (contains(text, criteria.query, false))
matched = true; matched = true;
} }
} }
@ -386,9 +402,10 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
while (found < pageSize && !state.destroyed) { while (found < pageSize && !state.destroyed) {
if (state.matches == null || if (state.matches == null ||
(state.matches.size() > 0 && state.index >= state.matches.size())) { (state.matches.size() > 0 && state.index >= state.matches.size())) {
String query = (word.size() == 0 ? null : '%' + TextUtils.join("%", word) + '%');
state.matches = db.message().matchMessages( state.matches = db.message().matchMessages(
account, folder, exclude, account, folder, exclude,
criteria.query == null ? null : "%" + criteria.query + "%", query,
criteria.in_senders, criteria.in_senders,
criteria.in_recipients, criteria.in_recipients,
criteria.in_subject, criteria.in_subject,
@ -412,6 +429,7 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
" account=" + account + " account=" + account +
" folder=" + folder + " folder=" + folder +
" criteria=" + criteria + " criteria=" + criteria +
" query=" + query +
" offset=" + state.offset + " offset=" + state.offset +
" size=" + state.matches.size()); " size=" + state.matches.size());
state.offset += Math.min(state.matches.size(), SEARCH_LIMIT_DEVICE); state.offset += Math.min(state.matches.size(), SEARCH_LIMIT_DEVICE);
@ -425,19 +443,28 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
state.index = i + 1; state.index = i + 1;
TupleMatch match = state.matches.get(i); TupleMatch match = state.matches.get(i);
boolean matched = (match.matched != null && match.matched); boolean matched = (match.matched != null && match.matched &&
plus.size() + minus.size() == 0);
if (!matched && criteria.query != null &&
criteria.in_subject && plus.size() + minus.size() > 0) {
EntityMessage message = db.message().getMessage(match.id);
if (message != null)
matched = contains(message.subject, criteria.query, false);
}
if (!matched && criteria.query != null && (criteria.in_message || criteria.in_html)) if (!matched && criteria.query != null &&
(criteria.in_message || criteria.in_html))
try { try {
File file = EntityMessage.getFile(context, match.id); File file = EntityMessage.getFile(context, match.id);
if (file.exists()) { if (file.exists()) {
String html = Helper.readText(file); String html = Helper.readText(file);
if (contains(html, criteria.query)) { if (contains(html, criteria.query, true)) {
if (criteria.in_html) if (criteria.in_html)
matched = true; matched = true;
else { else {
String text = HtmlHelper.getFullText(html); String text = HtmlHelper.getFullText(html);
if (contains(text, criteria.query)) if (contains(text, criteria.query, false))
matched = true; matched = true;
} }
} }
@ -795,16 +822,33 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
if (addresses == null) if (addresses == null)
return false; return false;
for (Address address : addresses) for (Address address : addresses)
if (contains(address.toString(), query)) if (contains(address.toString(), query, false))
return true; return true;
return false; return false;
} }
private static boolean contains(String text, String query) { private static boolean contains(String text, String query, boolean html) {
if (TextUtils.isEmpty(text)) if (TextUtils.isEmpty(text))
return false; return false;
return Fts4DbHelper.preprocessText(text)
.contains(Fts4DbHelper.preprocessText(query)); text = Fts4DbHelper.preprocessText(text);
query = Fts4DbHelper.preprocessText(query);
List<String> word = new ArrayList<>();
for (String w : query.trim().split("\\s+"))
if (w.length() > 1 && (w.startsWith("+") || w.startsWith("-"))) {
if (w.startsWith("+") && !text.contains(w.substring(1)))
return false;
if (w.startsWith("-") && text.contains(w.substring(1)))
return false;
} else
word.add(w);
if (word.size() == 0)
return true;
Pattern pat = Pattern.compile(".*" + TextUtils.join("\\s+", word) + ".*", Pattern.DOTALL);
return pat.matcher(text).matches();
} }
State getState() { State getState() {
@ -910,11 +954,7 @@ public class BoundaryCallbackMessages extends PagedList.BoundaryCallback<TupleMe
return false; return false;
for (String w : query.trim().split("\\s+")) for (String w : query.trim().split("\\s+"))
if (w.length() > 1 && w.startsWith("+")) if (w.length() > 1 && w.startsWith("?"))
return true;
else if (w.length() > 1 && w.startsWith("-"))
return true;
else if (w.length() > 1 && w.startsWith("?"))
return true; return true;
else if (w.length() > FROM.length() && w.startsWith(FROM)) else if (w.length() > FROM.length() && w.startsWith(FROM))
return true; return true;

@ -187,9 +187,8 @@ public class Fts4DbHelper extends SQLiteOpenHelper {
static List<Long> match( static List<Long> match(
SQLiteDatabase db, SQLiteDatabase db,
Long account, Long folder, long[] exclude, Long account, Long folder, long[] exclude,
BoundaryCallbackMessages.SearchCriteria criteria) { BoundaryCallbackMessages.SearchCriteria criteria, String query) {
String search = escape(breakText(query));
String search = escape(breakText(criteria.query));
String select = ""; String select = "";
if (account != null) if (account != null)

@ -122,6 +122,7 @@ import java.util.LinkedHashMap;
import java.util.List; import java.util.List;
import java.util.Locale; import java.util.Locale;
import java.util.Map; import java.util.Map;
import java.util.regex.MatchResult;
import java.util.regex.Matcher; import java.util.regex.Matcher;
import java.util.regex.Pattern; import java.util.regex.Pattern;
@ -2934,52 +2935,67 @@ public class HtmlHelper {
return ssb; return ssb;
} }
static Document highlightSearched(Context context, Document document, String searched) { static Document highlightSearched(Context context, Document document, String query) {
String find = searched.toLowerCase();
int color = Helper.resolveColor(context, R.attr.colorHighlight); int color = Helper.resolveColor(context, R.attr.colorHighlight);
NodeTraversor.traverse(new NodeVisitor() { List<String> word = new ArrayList<>();
@Override List<String> plus = new ArrayList<>();
public void head(Node node, int depth) { for (String w : query.trim().split("\\s+"))
if (node instanceof TextNode) { if (w.length() > 1 && (w.startsWith("+") || w.startsWith("-"))) {
TextNode tnode = (TextNode) node; if (w.startsWith("+"))
String text = tnode.getWholeText(); plus.add(w.substring(1));
} else
int start = text.toLowerCase().indexOf(find); word.add(w);
if (start < 0)
return;
int prev = 0;
Element holder = document.createElement("span");
while (start >= 0) { int flags = Pattern.DOTALL | Pattern.CASE_INSENSITIVE;
if (start > prev) List<Pattern> pat = new ArrayList<>();
holder.appendText(text.substring(prev, start)); pat.add(Pattern.compile(".*(" + TextUtils.join("\\s+", word) + ").*", flags));
for (String w : plus)
pat.add(Pattern.compile(".*(" + w + ").*", flags));
Element span = document.createElement("span"); for (Pattern p : pat)
span.attr("style", mergeStyles( NodeTraversor.traverse(new NodeVisitor() {
span.attr("style"), @Override
"font-size:larger; background-color:" + encodeWebColor(color) public void head(Node node, int depth) {
)); if (node instanceof TextNode)
span.text(text.substring(start, start + find.length())); try {
holder.appendChild(span); TextNode tnode = (TextNode) node;
String text = tnode.getWholeText();
prev = start + find.length();
start = text.toLowerCase().indexOf(find, prev); Matcher result = p.matcher(text);
} if (!result.matches())
return;
int prev = 0;
Element holder = document.createElement("span");
for (int i = 1; i <= result.groupCount(); i++) {
holder.appendText(text.substring(prev, result.start(i)));
Element span = document.createElement("span");
span.attr("style", mergeStyles(
span.attr("style"),
"font-size:larger; background-color:" + encodeWebColor(color)
));
span.text(text.substring(result.start(i), result.end(i)));
holder.appendChild(span);
prev = result.end(i);
}
if (prev < text.length()) if (prev < text.length())
holder.appendText(text.substring(prev)); holder.appendText(text.substring(prev));
tnode.before(holder); tnode.before(holder);
tnode.text(""); tnode.text("");
} catch (Throwable ex) {
Log.e(ex);
}
} }
}
@Override @Override
public void tail(Node node, int depth) { public void tail(Node node, int depth) {
} }
}, document); }, document);
return document; return document;
} }

Loading…
Cancel
Save