OsmAnd/OsmAnd-java/src/main/java/net/osmand/CollatorStringMatcher.java

176 lines
4.8 KiB
Java
Raw Normal View History

package net.osmand;
2015-06-15 21:32:35 +02:00
import java.util.Locale;
/**
* Abstract collator matcher that basically supports subclasses with some collator
* matching.
*
* @author pavol.zibrita
*/
public class CollatorStringMatcher implements StringMatcher {
private final Collator collator;
private final StringMatcherMode mode;
private final String part;
public static enum StringMatcherMode {
CHECK_ONLY_STARTS_WITH,
CHECK_STARTS_FROM_SPACE,
CHECK_STARTS_FROM_SPACE_NOT_BEGINNING,
2016-06-21 01:24:25 +02:00
CHECK_EQUALS_FROM_SPACE,
2018-10-03 13:56:18 +02:00
CHECK_CONTAINS,
2018-10-08 21:44:45 +02:00
CHECK_ONLY_STARTS_WITH_TRIM,
CHECK_EQUALS,
}
public CollatorStringMatcher(String part, StringMatcherMode mode) {
2014-04-12 23:37:18 +02:00
this.collator = OsmAndCollator.primaryCollator();
2015-06-15 21:32:35 +02:00
this.part = part.toLowerCase(Locale.getDefault());
this.mode = mode;
}
public Collator getCollator() {
return collator;
}
@Override
public boolean matches(String name) {
return cmatches(collator, name, part, mode);
}
public static boolean cmatches(Collator collator, String base, String part, StringMatcherMode mode){
switch (mode) {
case CHECK_CONTAINS:
return ccontains(collator, base, part);
2016-06-21 01:24:25 +02:00
case CHECK_EQUALS_FROM_SPACE:
2018-10-03 13:56:18 +02:00
return cstartsWith(collator, base, part, true, true, true, false);
case CHECK_STARTS_FROM_SPACE:
2018-10-03 13:56:18 +02:00
return cstartsWith(collator, base, part, true, true, false, false);
case CHECK_STARTS_FROM_SPACE_NOT_BEGINNING:
2018-10-03 13:56:18 +02:00
return cstartsWith(collator, base, part, false, true, false, false);
case CHECK_ONLY_STARTS_WITH:
2018-10-03 13:56:18 +02:00
return cstartsWith(collator, base, part, true, false, false, false);
case CHECK_ONLY_STARTS_WITH_TRIM:
return cstartsWith(collator, base, part, true, false, false, true);
2018-10-08 21:44:45 +02:00
case CHECK_EQUALS:
return cstartsWith(collator, base, part, false, false, true, false);
}
return false;
}
/**
* Check if part contains in base
*
* @param collator Collator to use
* @param part String to search
* @param base String where to search
* @return true if part is contained in base
*/
public static boolean ccontains(Collator collator, String base, String part) {
2016-08-04 11:15:31 +02:00
// int pos = 0;
// if (part.length() > 3) {
// // improve searching by searching first 3 characters
// pos = cindexOf(collator, pos, part.substring(0, 3), base);
// if (pos == -1) {
// return false;
// }
// }
// pos = cindexOf(collator, pos, part, base);
// if (pos == -1) {
// return false;
// }
// return true;
if (base.length() <= part.length())
return collator.equals(base, part);
for (int pos = 0; pos <= base.length() - part.length() + 1; pos++) {
String temp = base.substring(pos, base.length());
for (int length = temp.length(); length >= 0; length--) {
String temp2 = temp.substring(0, length);
if (collator.equals(temp2, part))
return true;
}
}
2016-08-04 11:15:31 +02:00
return false;
}
private static int cindexOf(Collator collator, int start, String part, String base) {
for (int pos = start; pos <= base.length() - part.length(); pos++) {
if (collator.equals(base.substring(pos, pos + part.length()), part)) {
return pos;
}
}
return -1;
}
/**
* Checks if string starts with another string.
* Special check try to find as well in the middle of name
*
* @param collator
2018-10-03 13:56:18 +02:00
* @param searchInParam
* @param theStart
2018-10-03 13:56:18 +02:00
* @param trim - trim theStart to searchInParam length if searchInParam non empty
* @return true if searchIn starts with token
*/
2015-06-15 21:32:35 +02:00
public static boolean cstartsWith(Collator collator, String searchInParam, String theStart,
2018-10-03 13:56:18 +02:00
boolean checkBeginning, boolean checkSpaces, boolean equals, boolean trim) {
2015-06-15 21:32:35 +02:00
String searchIn = searchInParam.toLowerCase(Locale.getDefault());
int searchInLength = searchIn.length();
2018-10-03 13:56:18 +02:00
if (trim && searchInLength > 0 && theStart.length() > searchInLength) {
theStart = theStart.substring(0, searchInLength);
}
int startLength = theStart.length();
if (startLength == 0) {
return true;
}
if (startLength > searchInLength) {
return false;
}
// simulate starts with for collator
if (checkBeginning) {
boolean starts = collator.equals(searchIn.substring(0, startLength), theStart);
if (starts) {
2016-06-21 01:24:25 +02:00
if (equals) {
if (startLength == searchInLength || isSpace(searchIn.charAt(startLength))) {
return true;
}
} else {
return true;
}
}
}
if (checkSpaces) {
for (int i = 1; i <= searchInLength - startLength; i++) {
if (isSpace(searchIn.charAt(i - 1)) && !isSpace(searchIn.charAt(i))) {
if (collator.equals(searchIn.substring(i, i + startLength), theStart)) {
2016-06-21 01:24:25 +02:00
if(equals) {
if(i + startLength == searchInLength || isSpace(searchIn.charAt(i + startLength))) {
return true;
}
} else {
return true;
}
}
}
}
}
2018-10-08 21:44:45 +02:00
if (!checkBeginning && !checkSpaces && equals) {
return collator.equals(searchIn, theStart);
}
return false;
}
private static boolean isSpace(char c){
return !Character.isLetter(c) && !Character.isDigit(c);
}
}