Merge remote-tracking branch 'origin/master' into merge-turn-lanes
This commit is contained in:
commit
db6d99dc92
14 changed files with 202 additions and 218 deletions
|
@ -1,16 +1,16 @@
|
||||||
package net.osmand.router;
|
package net.osmand.router;
|
||||||
|
|
||||||
|
import gnu.trove.list.array.TIntArrayList;
|
||||||
|
import gnu.trove.set.hash.TIntHashSet;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
import java.text.MessageFormat;
|
import java.text.MessageFormat;
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
import java.util.Arrays;
|
import java.util.Arrays;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.HashSet;
|
|
||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
|
|
||||||
import org.apache.commons.logging.Log;
|
|
||||||
|
|
||||||
import net.osmand.PlatformUtil;
|
import net.osmand.PlatformUtil;
|
||||||
import net.osmand.binary.BinaryMapIndexReader;
|
import net.osmand.binary.BinaryMapIndexReader;
|
||||||
import net.osmand.binary.BinaryMapRouteReaderAdapter.RouteTypeRule;
|
import net.osmand.binary.BinaryMapRouteReaderAdapter.RouteTypeRule;
|
||||||
|
@ -21,6 +21,8 @@ import net.osmand.router.BinaryRoutePlanner.RouteSegment;
|
||||||
import net.osmand.router.RoutePlannerFrontEnd.RouteCalculationMode;
|
import net.osmand.router.RoutePlannerFrontEnd.RouteCalculationMode;
|
||||||
import net.osmand.util.MapUtils;
|
import net.osmand.util.MapUtils;
|
||||||
|
|
||||||
|
import org.apache.commons.logging.Log;
|
||||||
|
|
||||||
public class RouteResultPreparation {
|
public class RouteResultPreparation {
|
||||||
|
|
||||||
public static boolean PRINT_TO_CONSOLE_ROUTE_INFORMATION_TO_TEST = false;
|
public static boolean PRINT_TO_CONSOLE_ROUTE_INFORMATION_TO_TEST = false;
|
||||||
|
@ -514,7 +516,10 @@ public class RouteResultPreparation {
|
||||||
} else {
|
} else {
|
||||||
t = TurnType.valueOf(TurnType.TU, leftSide);
|
t = TurnType.valueOf(TurnType.TU, leftSide);
|
||||||
}
|
}
|
||||||
assignLanesInfo(prev, t, leftSide);
|
int[] lanes = getTurnLanesInfo(prev, t.getValue(), leftSide, true);
|
||||||
|
if(lanes != null) {
|
||||||
|
t.setLanes(lanes);
|
||||||
|
}
|
||||||
} else if (mpi < -TURN_DEGREE_MIN) {
|
} else if (mpi < -TURN_DEGREE_MIN) {
|
||||||
if (mpi > -60) {
|
if (mpi > -60) {
|
||||||
t = TurnType.valueOf(TurnType.TSLR, leftSide);
|
t = TurnType.valueOf(TurnType.TSLR, leftSide);
|
||||||
|
@ -525,7 +530,10 @@ public class RouteResultPreparation {
|
||||||
} else {
|
} else {
|
||||||
t = TurnType.valueOf(TurnType.TU, leftSide);
|
t = TurnType.valueOf(TurnType.TU, leftSide);
|
||||||
}
|
}
|
||||||
assignLanesInfo(prev, t, leftSide);
|
int[] lanes = getTurnLanesInfo(prev, t.getValue(), leftSide, false);
|
||||||
|
if(lanes != null) {
|
||||||
|
t.setLanes(lanes);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
t = attachKeepLeftInfoAndLanes(leftSide, prev, rr, t);
|
t = attachKeepLeftInfoAndLanes(leftSide, prev, rr, t);
|
||||||
}
|
}
|
||||||
|
@ -536,59 +544,52 @@ public class RouteResultPreparation {
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void assignLanesInfo(RouteSegmentResult prevSegm, TurnType t, boolean leftSide) {
|
private int[] getTurnLanesInfo(RouteSegmentResult prevSegm, int mainTurnType, boolean leftSide,
|
||||||
int lanes = countLanes(prevSegm);
|
boolean leftTurn) {
|
||||||
if (lanes <= 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
String turnLanes = getTurnLanesString(prevSegm);
|
String turnLanes = getTurnLanesString(prevSegm);
|
||||||
if (turnLanes == null) {
|
if (turnLanes == null) {
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
|
||||||
String[] splitLaneOptions = turnLanes.split("\\|", -1);
|
String[] splitLaneOptions = turnLanes.split("\\|", -1);
|
||||||
if (splitLaneOptions.length != lanes) {
|
if (splitLaneOptions.length != countLanesMinOne(prevSegm)) {
|
||||||
// Error in data or missing data
|
// Error in data or missing data
|
||||||
return;
|
return null;
|
||||||
}
|
}
|
||||||
|
int[] lanesArray = calculateRawTurnLanes(splitLaneOptions, mainTurnType);
|
||||||
int[] lanesArray = new int[lanes];
|
// Manually set the allowed lanes.
|
||||||
t.setLanes(lanesArray);
|
boolean isSet = setAllowedLanes(mainTurnType, lanesArray);
|
||||||
assignTurns(splitLaneOptions, t);
|
if(!isSet && lanesArray.length > 0) {
|
||||||
|
// In some cases (at least in the US), the rightmost lane might not have a right turn indicated as per turn:lanes,
|
||||||
// In some cases (at least in the US), the rightmost lane might not have a right turn indicated as per turn:lanes, but is allowed and being used here. This section adds in that indicator. The same applies for where leftSide is true.
|
// but is allowed and being used here. This section adds in that indicator. The same applies for where leftSide is true.
|
||||||
if (leftSide) {
|
int ind = leftTurn? 0 : lanesArray.length - 1;
|
||||||
if (t.getValue() == TurnType.TL
|
final int tt = TurnType.getPrimaryTurn(lanesArray[ind]);
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[0]) != TurnType.TL
|
if (leftTurn) {
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[0]) != TurnType.TSLL
|
if (!TurnType.isLeftTurn(tt)) {
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[0]) != TurnType.TSHL) {
|
|
||||||
if (TurnType.getPrimaryTurn(lanesArray[0]) != 0) {
|
|
||||||
// This was just to make sure that there's no bad data.
|
// This was just to make sure that there's no bad data.
|
||||||
t.setSecondaryTurn(0, TurnType.getPrimaryTurn(lanesArray[0]));
|
TurnType.setSecondaryTurn(lanesArray, ind, tt);
|
||||||
t.setPrimaryTurn(0, TurnType.TL);
|
TurnType.setPrimaryTurn(lanesArray, ind, TurnType.TL);
|
||||||
}
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
int lastIndex = lanesArray.length - 1;
|
if (!TurnType.isRightTurn(tt)) {
|
||||||
if (t.getValue() == TurnType.TR
|
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[lastIndex]) != TurnType.TR
|
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[lastIndex]) != TurnType.TSLR
|
|
||||||
&& TurnType.getPrimaryTurn(lanesArray[lastIndex]) != TurnType.TSHR) {
|
|
||||||
if (TurnType.getPrimaryTurn(lanesArray[lastIndex]) != 0) {
|
|
||||||
// This was just to make sure that there's no bad data.
|
// This was just to make sure that there's no bad data.
|
||||||
t.setSecondaryTurn(lastIndex, TurnType.getPrimaryTurn(lanesArray[lastIndex]));
|
TurnType.setSecondaryTurn(lanesArray, ind, tt);
|
||||||
t.setPrimaryTurn(lastIndex, TurnType.TR);
|
TurnType.setPrimaryTurn(lanesArray, ind, TurnType.TR);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
setAllowedLanes(lanesArray[ind], lanesArray);
|
||||||
|
}
|
||||||
|
return lanesArray;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Manually set the allowed lanes.
|
protected boolean setAllowedLanes(int mainTurnType, int[] lanesArray) {
|
||||||
|
boolean turnSet = false;
|
||||||
for (int i = 0; i < lanesArray.length; i++) {
|
for (int i = 0; i < lanesArray.length; i++) {
|
||||||
if (TurnType.getPrimaryTurn(lanesArray[i]) == t.getValue()) {
|
if (TurnType.getPrimaryTurn(lanesArray[i]) == mainTurnType) {
|
||||||
lanesArray[i] |= 1;
|
lanesArray[i] |= 1;
|
||||||
|
turnSet = true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return turnSet;
|
||||||
}
|
}
|
||||||
|
|
||||||
private TurnType processRoundaboutTurn(List<RouteSegmentResult> result, int i, boolean leftSide, RouteSegmentResult prev,
|
private TurnType processRoundaboutTurn(List<RouteSegmentResult> result, int i, boolean leftSide, RouteSegmentResult prev,
|
||||||
|
@ -625,7 +626,6 @@ public class RouteResultPreparation {
|
||||||
|
|
||||||
private TurnType attachKeepLeftInfoAndLanes(boolean leftSide, RouteSegmentResult prevSegm, RouteSegmentResult currentSegm, TurnType t) {
|
private TurnType attachKeepLeftInfoAndLanes(boolean leftSide, RouteSegmentResult prevSegm, RouteSegmentResult currentSegm, TurnType t) {
|
||||||
// keep left/right
|
// keep left/right
|
||||||
int[] lanes = null;
|
|
||||||
boolean kl = false;
|
boolean kl = false;
|
||||||
boolean kr = false;
|
boolean kr = false;
|
||||||
List<RouteSegmentResult> attachedRoutes = currentSegm.getAttachedRoutes(currentSegm.getStartPointIndex());
|
List<RouteSegmentResult> attachedRoutes = currentSegm.getAttachedRoutes(currentSegm.getStartPointIndex());
|
||||||
|
@ -646,21 +646,11 @@ public class RouteResultPreparation {
|
||||||
if (rsSpeakPriority != MAX_SPEAK_PRIORITY || speakPriority == MAX_SPEAK_PRIORITY) {
|
if (rsSpeakPriority != MAX_SPEAK_PRIORITY || speakPriority == MAX_SPEAK_PRIORITY) {
|
||||||
if ((ex < TURN_DEGREE_MIN || mpi < TURN_DEGREE_MIN) && ex >= 0) {
|
if ((ex < TURN_DEGREE_MIN || mpi < TURN_DEGREE_MIN) && ex >= 0) {
|
||||||
kl = true;
|
kl = true;
|
||||||
int lns = countLanes(attached);
|
right += countLanesMinOne(attached);
|
||||||
if (lns <= 0) {
|
|
||||||
right += 1;
|
|
||||||
} else {
|
|
||||||
right += lns;
|
|
||||||
}
|
|
||||||
speak = speak || rsSpeakPriority <= speakPriority;
|
speak = speak || rsSpeakPriority <= speakPriority;
|
||||||
} else if ((ex > -TURN_DEGREE_MIN || mpi < TURN_DEGREE_MIN) && ex <= 0) {
|
} else if ((ex > -TURN_DEGREE_MIN || mpi < TURN_DEGREE_MIN) && ex <= 0) {
|
||||||
kr = true;
|
kr = true;
|
||||||
int lns = countLanes(attached);
|
left += countLanesMinOne(attached);
|
||||||
if (lns <= 0) {
|
|
||||||
left += 1;
|
|
||||||
} else {
|
|
||||||
left += lns;
|
|
||||||
}
|
|
||||||
speak = speak || rsSpeakPriority <= speakPriority;
|
speak = speak || rsSpeakPriority <= speakPriority;
|
||||||
} else if (mpi >= TURN_DEGREE_MIN) {
|
} else if (mpi >= TURN_DEGREE_MIN) {
|
||||||
// Indicate that there are other turns at this intersection, and displaying the lanes may be helpful here.
|
// Indicate that there are other turns at this intersection, and displaying the lanes may be helpful here.
|
||||||
|
@ -674,13 +664,8 @@ public class RouteResultPreparation {
|
||||||
} else if(kl && right == 0) {
|
} else if(kl && right == 0) {
|
||||||
right = 1;
|
right = 1;
|
||||||
}
|
}
|
||||||
int current = countLanes(currentSegm);
|
int current = countLanesMinOne(currentSegm);
|
||||||
// attachedRoutes covers all allowed outbound routes at that point except currentSegm.
|
int[] lanes = new int[current + left + right];
|
||||||
if (current <= 0) {
|
|
||||||
current = 1;
|
|
||||||
}
|
|
||||||
// if(ls >= 0 /*&& current + left + right >= ls*/){
|
|
||||||
lanes = new int[current + left + right];
|
|
||||||
ls = current + left + right;
|
ls = current + left + right;
|
||||||
for (int it = 0; it < ls; it++) {
|
for (int it = 0; it < ls; it++) {
|
||||||
if (it < left || it >= left + current) {
|
if (it < left || it >= left + current) {
|
||||||
|
@ -693,7 +678,6 @@ public class RouteResultPreparation {
|
||||||
if ((current <= left + right) && (left > 1 || right > 1)) {
|
if ((current <= left + right) && (left > 1 || right > 1)) {
|
||||||
speak = true;
|
speak = true;
|
||||||
}
|
}
|
||||||
// }
|
|
||||||
|
|
||||||
double devation = Math.abs(MapUtils.degreesDiff(prevSegm.getBearingEnd(), currentSegm.getBearingBegin()));
|
double devation = Math.abs(MapUtils.degreesDiff(prevSegm.getBearingEnd(), currentSegm.getBearingBegin()));
|
||||||
boolean makeSlightTurn = devation > 5 && (!isMotorway(prevSegm) || !isMotorway(currentSegm));
|
boolean makeSlightTurn = devation > 5 && (!isMotorway(prevSegm) || !isMotorway(currentSegm));
|
||||||
|
@ -730,30 +714,52 @@ public class RouteResultPreparation {
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
if (t != null && lanes != null) {
|
if (t != null && lanes != null) {
|
||||||
t.setLanes(lanes);
|
int[] calcLanes = attachTurnLanesData(prevSegm, lanes);
|
||||||
|
if(calcLanes != lanes) {
|
||||||
|
int tp = inferTurnFromLanes(calcLanes);
|
||||||
|
if (tp != 0 && tp != t.getValue()) {
|
||||||
|
TurnType derivedTurnType = TurnType.valueOf(tp, leftSide);
|
||||||
|
derivedTurnType.setLanes(calcLanes);
|
||||||
|
derivedTurnType.setSkipToSpeak(t.isSkipToSpeak());
|
||||||
|
// Because only the primary turn is displayed, if the turn to be taken is currently set as the
|
||||||
|
// secondary turn, then that needs to be switched around.
|
||||||
|
for (int i = 0; i < calcLanes.length; i++) {
|
||||||
|
if (TurnType.getSecondaryTurn(calcLanes[i]) == tp) {
|
||||||
|
derivedTurnType.setSecondaryTurn(i, TurnType.getPrimaryTurn(calcLanes[i]));
|
||||||
|
derivedTurnType.setPrimaryTurn(i, tp);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
t.setLanes(calcLanes);
|
||||||
|
|
||||||
t = attachTurnLanesData(leftSide, prevSegm, t);
|
|
||||||
}
|
}
|
||||||
return t;
|
return t;
|
||||||
}
|
}
|
||||||
|
|
||||||
protected int countLanes(RouteSegmentResult attached) {
|
|
||||||
if (attached.getObject().getOneway() == 0) {
|
protected int countLanesMinOne(RouteSegmentResult attached) {
|
||||||
|
final boolean oneway = attached.getObject().getOneway() != 0;
|
||||||
|
int lns = attached.getObject().getLanes();
|
||||||
|
if(lns == 0) {
|
||||||
|
String tls = getTurnLanesString(attached);
|
||||||
|
if(tls != null) {
|
||||||
|
return Math.max(1, countOccurrences(tls, '|'));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (oneway) {
|
||||||
|
return Math.max(1, lns);
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
if (attached.isForwardDirection() && attached.getObject().getValue("lanes:forward") != null) {
|
if (attached.isForwardDirection() && attached.getObject().getValue("lanes:forward") != null) {
|
||||||
return Integer.parseInt(attached.getObject().getValue("lanes:forward"));
|
return Integer.parseInt(attached.getObject().getValue("lanes:forward"));
|
||||||
} else if (!attached.isForwardDirection() && attached.getObject().getValue("lanes:backward") != null) {
|
} else if (!attached.isForwardDirection() && attached.getObject().getValue("lanes:backward") != null) {
|
||||||
return Integer.parseInt(attached.getObject().getValue("lanes:backward"));
|
return Integer.parseInt(attached.getObject().getValue("lanes:backward"));
|
||||||
} else {
|
|
||||||
return -1;
|
|
||||||
}
|
}
|
||||||
} catch(NumberFormatException e) {
|
} catch(NumberFormatException e) {
|
||||||
e.printStackTrace();
|
e.printStackTrace();
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return attached.getObject().getLanes();
|
|
||||||
}
|
}
|
||||||
|
return Math.max(1, (lns + 1) / 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
protected String getTurnLanesString(RouteSegmentResult segment) {
|
protected String getTurnLanesString(RouteSegmentResult segment) {
|
||||||
|
@ -768,87 +774,44 @@ public class RouteResultPreparation {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private TurnType attachTurnLanesData(boolean leftSide, RouteSegmentResult prevSegm, TurnType t) {
|
private int[] attachTurnLanesData(RouteSegmentResult prevSegm, int[] outgoingCalcLanes) {
|
||||||
int lanes = countLanes(prevSegm);
|
|
||||||
String turnLanes = getTurnLanesString(prevSegm);
|
String turnLanes = getTurnLanesString(prevSegm);
|
||||||
|
|
||||||
if (turnLanes == null) {
|
if (turnLanes == null) {
|
||||||
return t;
|
return outgoingCalcLanes;
|
||||||
}
|
}
|
||||||
|
|
||||||
String[] splitLaneOptions = turnLanes.split("\\|", -1);
|
String[] splitLaneOptions = turnLanes.split("\\|", -1);
|
||||||
|
if (splitLaneOptions.length != countLanesMinOne(prevSegm)) {
|
||||||
if (splitLaneOptions.length != lanes) {
|
// Error in data or missing data
|
||||||
log.warn("Number of lanes in lanes key (" + lanes + ") does not match number of lanes from turn:lanes key (" + splitLaneOptions.length + "). Errors may occur.");
|
return outgoingCalcLanes;
|
||||||
|
|
||||||
int leftLanes = 0;
|
|
||||||
int rightLanes = 0;
|
|
||||||
boolean processingLeft = true;
|
|
||||||
for (int i = 0; i < t.getLanes().length; i++) {
|
|
||||||
if (t.getLanes()[i] == 0) {
|
|
||||||
if (processingLeft) {
|
|
||||||
leftLanes++;
|
|
||||||
} else {
|
|
||||||
rightLanes++;
|
|
||||||
}
|
}
|
||||||
} else {
|
int[] usableLanes = mergeOutgoingLanesUsingTurnLanes(outgoingCalcLanes, splitLaneOptions);
|
||||||
processingLeft = false;
|
int[] rawLanes = calculateRawTurnLanes(splitLaneOptions, 0);
|
||||||
|
for(int k = 0 ; k < splitLaneOptions.length; k++) {
|
||||||
|
if((usableLanes[k] & 1) != 0) {
|
||||||
|
rawLanes[k] |= 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return rawLanes;
|
||||||
int[] adjustedLanes = new int[lanes + leftLanes + rightLanes];
|
|
||||||
|
|
||||||
for (int i = leftLanes; i < leftLanes + lanes; i++) {
|
|
||||||
adjustedLanes[i] = 1;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
t.setLanes(adjustedLanes);
|
protected int[] mergeOutgoingLanesUsingTurnLanes(int[] outgoingCalcLanes, String[] splitLaneOptions) {
|
||||||
}
|
int[] usableLanes = outgoingCalcLanes;
|
||||||
|
if (outgoingCalcLanes.length != splitLaneOptions.length) {
|
||||||
if (t.getLanes().length != lanes) {
|
usableLanes = new int[splitLaneOptions.length];
|
||||||
// The lanes from prevSegm don't easily match up to the target roads (it's not one-to-one).
|
// The turn:lanes don't easily match up to the target road.
|
||||||
List<Integer> sourceLanes = new ArrayList<Integer>();
|
|
||||||
|
|
||||||
int outgoingLanesIndex = 0;
|
int outgoingLanesIndex = 0;
|
||||||
int sourceLanesIndex = 0;
|
int sourceLanesIndex = 0;
|
||||||
|
while (outgoingLanesIndex < outgoingCalcLanes.length &&
|
||||||
while (outgoingLanesIndex < t.getLanes().length && sourceLanesIndex < splitLaneOptions.length) {
|
sourceLanesIndex < splitLaneOptions.length) {
|
||||||
if (splitLaneOptions[sourceLanesIndex].contains(";")) {
|
|
||||||
// Two or more allowed turns for this lane
|
|
||||||
int options = countOccurrences(splitLaneOptions[sourceLanesIndex], ';');
|
int options = countOccurrences(splitLaneOptions[sourceLanesIndex], ';');
|
||||||
if (options == 1) {
|
for (int k = 0; k <= options && outgoingLanesIndex < outgoingCalcLanes.length; k++) {
|
||||||
if (outgoingLanesIndex + 1 >= t.getLanes().length) {
|
usableLanes[sourceLanesIndex] |= outgoingCalcLanes[outgoingLanesIndex];
|
||||||
// Likely an error in data
|
|
||||||
return t;
|
|
||||||
}
|
|
||||||
int usability = t.getLanes()[outgoingLanesIndex] | t.getLanes()[outgoingLanesIndex + 1];
|
|
||||||
sourceLanes.add(usability);
|
|
||||||
outgoingLanesIndex += 2;
|
|
||||||
sourceLanesIndex++;
|
|
||||||
} else {
|
|
||||||
// Not supported
|
|
||||||
return t;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Only one allowed turn; behave normally
|
|
||||||
sourceLanes.add(t.getLanes()[outgoingLanesIndex]);
|
|
||||||
outgoingLanesIndex++;
|
outgoingLanesIndex++;
|
||||||
|
}
|
||||||
sourceLanesIndex++;
|
sourceLanesIndex++;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return usableLanes;
|
||||||
int[] newLanes = new int[sourceLanes.size()];
|
|
||||||
|
|
||||||
for (int i = 0; i < sourceLanes.size(); i++) {
|
|
||||||
newLanes[i] = sourceLanes.get(i);
|
|
||||||
}
|
|
||||||
|
|
||||||
t.setLanes(newLanes);
|
|
||||||
}
|
|
||||||
|
|
||||||
assignTurns(splitLaneOptions, t);
|
|
||||||
t = inferTurnFromLanes(t, leftSide);
|
|
||||||
return t;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private int countOccurrences(String haystack, char needle) {
|
private int countOccurrences(String haystack, char needle) {
|
||||||
|
@ -861,7 +824,8 @@ public class RouteResultPreparation {
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
private void assignTurns(String[] splitLaneOptions, TurnType t) {
|
private int[] calculateRawTurnLanes(String[] splitLaneOptions, int calcTurnType) {
|
||||||
|
int[] lanes = new int[splitLaneOptions.length];
|
||||||
for (int i = 0; i < splitLaneOptions.length; i++) {
|
for (int i = 0; i < splitLaneOptions.length; i++) {
|
||||||
String[] laneOptions = splitLaneOptions[i].split(";");
|
String[] laneOptions = splitLaneOptions[i].split(";");
|
||||||
|
|
||||||
|
@ -888,80 +852,64 @@ public class RouteResultPreparation {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (TurnType.getPrimaryTurn(t.getLanes()[i]) == 0) {
|
if (TurnType.getPrimaryTurn(lanes[i]) == 0) {
|
||||||
t.setPrimaryTurn(i, turn);
|
TurnType.setPrimaryTurn(lanes, i, turn);
|
||||||
} else {
|
} else {
|
||||||
if (turn == t.getValue()) {
|
if (turn == calcTurnType) {
|
||||||
t.setSecondaryTurn(i, TurnType.getPrimaryTurn(t.getLanes()[i]));
|
TurnType.setSecondaryTurn(lanes, i, TurnType.getPrimaryTurn(lanes[i]));
|
||||||
t.setPrimaryTurn(i, turn);
|
TurnType.setPrimaryTurn(lanes, i, turn);
|
||||||
} else {
|
} else {
|
||||||
t.setSecondaryTurn(i, turn);
|
TurnType.setSecondaryTurn(lanes, i, turn);
|
||||||
}
|
}
|
||||||
|
|
||||||
break; // Move on to the next lane
|
break; // Move on to the next lane
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return lanes;
|
||||||
}
|
}
|
||||||
|
|
||||||
private TurnType inferTurnFromLanes(TurnType t, boolean leftSide) {
|
private int inferTurnFromLanes(int[] oLanes) {
|
||||||
List<Integer> possibleTurns = new ArrayList<Integer>();
|
TIntHashSet possibleTurns = new TIntHashSet();
|
||||||
for (int i = 0; i < t.getLanes().length; i++) {
|
for (int i = 0; i < oLanes.length; i++) {
|
||||||
if ((t.getLanes()[i] & 1) == 0) {
|
if ((oLanes[i] & 1) == 0) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (possibleTurns.isEmpty()) {
|
if (possibleTurns.isEmpty()) {
|
||||||
// Nothing is in the list to compare to, so add the first elements
|
// Nothing is in the list to compare to, so add the first elements
|
||||||
possibleTurns.add(TurnType.getPrimaryTurn(t.getLanes()[i]));
|
possibleTurns.add(TurnType.getPrimaryTurn(oLanes[i]));
|
||||||
if (TurnType.getSecondaryTurn(t.getLanes()[i]) != 0) {
|
if (TurnType.getSecondaryTurn(oLanes[i]) != 0) {
|
||||||
possibleTurns.add(TurnType.getSecondaryTurn(t.getLanes()[i]));
|
possibleTurns.add(TurnType.getSecondaryTurn(oLanes[i]));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
List<Integer> laneTurns = new ArrayList<Integer>();
|
TIntArrayList laneTurns = new TIntArrayList();
|
||||||
laneTurns.add(TurnType.getPrimaryTurn(t.getLanes()[i]));
|
laneTurns.add(TurnType.getPrimaryTurn(oLanes[i]));
|
||||||
if (TurnType.getSecondaryTurn(t.getLanes()[i]) != 0) {
|
if (TurnType.getSecondaryTurn(oLanes[i]) != 0) {
|
||||||
laneTurns.add(TurnType.getSecondaryTurn(t.getLanes()[i]));
|
laneTurns.add(TurnType.getSecondaryTurn(oLanes[i]));
|
||||||
}
|
}
|
||||||
possibleTurns.retainAll(laneTurns);
|
possibleTurns.retainAll(laneTurns);
|
||||||
|
|
||||||
if (possibleTurns.isEmpty()) {
|
if (possibleTurns.isEmpty()) {
|
||||||
// No common turns, so can't determine anything.
|
// No common turns, so can't determine anything.
|
||||||
return t;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove all turns from lanes not selected...because those aren't it
|
// Remove all turns from lanes not selected...because those aren't it
|
||||||
for (int i = 0; i < t.getLanes().length; i++) {
|
for (int i = 0; i < oLanes.length; i++) {
|
||||||
if ((t.getLanes()[i] & 1) == 0 && !possibleTurns.isEmpty()) {
|
if ((oLanes[i] & 1) == 0 && !possibleTurns.isEmpty()) {
|
||||||
List<Integer> notLaneTurns = new ArrayList<Integer>();
|
possibleTurns.remove((Integer) TurnType.getPrimaryTurn(oLanes[i]));
|
||||||
notLaneTurns.add(TurnType.getPrimaryTurn(t.getLanes()[i]));
|
if (TurnType.getSecondaryTurn(oLanes[i]) != 0) {
|
||||||
if (TurnType.getSecondaryTurn(t.getLanes()[i]) != 0) {
|
possibleTurns.remove((Integer) TurnType.getSecondaryTurn(oLanes[i]));
|
||||||
notLaneTurns.add(TurnType.getSecondaryTurn(t.getLanes()[i]));
|
|
||||||
}
|
}
|
||||||
possibleTurns.removeAll(notLaneTurns);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Checking to see that there is only one unique turn
|
// Checking to see that there is only one unique turn
|
||||||
if (new HashSet<Integer>(possibleTurns).size() == 1) {
|
if (possibleTurns.size() == 1) {
|
||||||
TurnType derivedTurnType = TurnType.valueOf(possibleTurns.get(0), leftSide);
|
return possibleTurns.iterator().next();
|
||||||
derivedTurnType.setLanes(t.getLanes());
|
|
||||||
derivedTurnType.setSkipToSpeak(t.isSkipToSpeak());
|
|
||||||
t = derivedTurnType;
|
|
||||||
|
|
||||||
// Because only the primary turn is displayed, if the turn to be taken is currently set as the secondary turn, then that needs to be switched around.
|
|
||||||
for (int i = 0; i < t.getLanes().length; i++) {
|
|
||||||
if (TurnType.getSecondaryTurn(t.getLanes()[i]) == t.getValue()) {
|
|
||||||
int temp = TurnType.getSecondaryTurn(t.getLanes()[i]);
|
|
||||||
t.setSecondaryTurn(i, TurnType.getPrimaryTurn(t.getLanes()[i]));
|
|
||||||
t.setPrimaryTurn(i, temp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return t;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isMotorway(RouteSegmentResult s){
|
private boolean isMotorway(RouteSegmentResult s){
|
||||||
|
|
|
@ -163,6 +163,10 @@ public class TurnType {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Note that there is no "weight" or ordering between the primary and secondary turns.
|
// Note that there is no "weight" or ordering between the primary and secondary turns.
|
||||||
|
public static void setPrimaryTurn(int[] lanes, int lane, int turnType) {
|
||||||
|
lanes[lane] |= (turnType << 1);
|
||||||
|
}
|
||||||
|
|
||||||
public void setPrimaryTurn(int lane, int turnType) {
|
public void setPrimaryTurn(int lane, int turnType) {
|
||||||
lanes[lane] |= (turnType << 1);
|
lanes[lane] |= (turnType << 1);
|
||||||
}
|
}
|
||||||
|
@ -172,6 +176,10 @@ public class TurnType {
|
||||||
return (laneValue >> 1) & ((1 << 4) - 1);
|
return (laneValue >> 1) & ((1 << 4) - 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static void setSecondaryTurn(int[] lanes, int lane, int turnType) {
|
||||||
|
lanes[lane] |= (turnType << 5);
|
||||||
|
}
|
||||||
|
|
||||||
public void setSecondaryTurn(int lane, int turnType) {
|
public void setSecondaryTurn(int lane, int turnType) {
|
||||||
lanes[lane] |= (turnType << 5);
|
lanes[lane] |= (turnType << 5);
|
||||||
}
|
}
|
||||||
|
@ -237,4 +245,12 @@ public class TurnType {
|
||||||
}
|
}
|
||||||
return super.toString();
|
return super.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public static boolean isLeftTurn(int type) {
|
||||||
|
return type == TL || type == TSHL || type == TSLL;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static boolean isRightTurn(int type) {
|
||||||
|
return type == TR || type == TSHR || type == TSLR;
|
||||||
|
}
|
||||||
}
|
}
|
|
@ -1999,4 +1999,6 @@ OsmAnd мае адкрыты зыходны код і актыўна разві
|
||||||
<string name="wake_on_voice_never">Ніколі</string>
|
<string name="wake_on_voice_never">Ніколі</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s патрабуе дазвол на выключэньне экрану для энергазберажэньня.</string>
|
<string name="lock_screen_request_explanation">%1$s патрабуе дазвол на выключэньне экрану для энергазберажэньня.</string>
|
||||||
<string name="advanced_settings">Дадатковыя</string>
|
<string name="advanced_settings">Дадатковыя</string>
|
||||||
|
<string name="osmo_use_https_descr">Выкарыстоўваць бясьпечнае злучэньне з серверам</string>
|
||||||
|
<string name="osmo_use_https">Выкарыстоўваць HTTPS</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -493,7 +493,7 @@
|
||||||
<string name="poi_shoemaker">Skomager</string>
|
<string name="poi_shoemaker">Skomager</string>
|
||||||
<string name="poi_stand_builder">Udstillingsbygger</string>
|
<string name="poi_stand_builder">Udstillingsbygger</string>
|
||||||
<string name="poi_stonemason">Stenhugger</string>
|
<string name="poi_stonemason">Stenhugger</string>
|
||||||
<string name="poi_sun_protection_craft">Solafskærmning håndværk</string>
|
<string name="poi_sun_protection_craft">Solafskærmning</string>
|
||||||
<string name="poi_sweep">Skorstensfejer</string>
|
<string name="poi_sweep">Skorstensfejer</string>
|
||||||
<string name="poi_tailor">Skrædder</string>
|
<string name="poi_tailor">Skrædder</string>
|
||||||
<string name="poi_tiler">Fliselægger</string>
|
<string name="poi_tiler">Fliselægger</string>
|
||||||
|
@ -552,14 +552,14 @@
|
||||||
<string name="poi_seamark_anchorage">Ankringsområde</string>
|
<string name="poi_seamark_anchorage">Ankringsområde</string>
|
||||||
<string name="poi_seamark_anchor_berth">Ankerplads</string>
|
<string name="poi_seamark_anchor_berth">Ankerplads</string>
|
||||||
<string name="poi_seamark_beacon">Båke</string>
|
<string name="poi_seamark_beacon">Båke</string>
|
||||||
<string name="poi_seamark_beacon_cardinal">Cardinal båke</string>
|
<string name="poi_seamark_beacon_cardinal">Kardinal båke</string>
|
||||||
<string name="poi_seamark_beacon_lateral">Lateral båke</string>
|
<string name="poi_seamark_beacon_lateral">Lateral båke</string>
|
||||||
<string name="poi_seamark_beacon_safe_water">Sikkervand båke</string>
|
<string name="poi_seamark_beacon_safe_water">Sikkervand båke</string>
|
||||||
<string name="poi_seamark_beacon_special_purpose">Speciel formål båke</string>
|
<string name="poi_seamark_beacon_special_purpose">Speciel formål båke</string>
|
||||||
<string name="poi_seamark_berth">Kajplads</string>
|
<string name="poi_seamark_berth">Kajplads</string>
|
||||||
<string name="poi_seamark_building">Sømærke bygning</string>
|
<string name="poi_seamark_building">Sømærke bygning</string>
|
||||||
<string name="poi_seamark_bridge">Sømærke bro</string>
|
<string name="poi_seamark_bridge">Sømærke bro</string>
|
||||||
<string name="poi_seamark_buoy_cardinal">Cardinal tønde</string>
|
<string name="poi_seamark_buoy_cardinal">Kardinal tønde</string>
|
||||||
<string name="poi_seamark_buoy_installation">Installation tønde</string>
|
<string name="poi_seamark_buoy_installation">Installation tønde</string>
|
||||||
<string name="poi_seamark_buoy_isolated_danger">Tønde, isoleret fare</string>
|
<string name="poi_seamark_buoy_isolated_danger">Tønde, isoleret fare</string>
|
||||||
<string name="poi_seamark_buoy_lateral">Lateral tønde</string>
|
<string name="poi_seamark_buoy_lateral">Lateral tønde</string>
|
||||||
|
@ -597,7 +597,7 @@
|
||||||
<string name="poi_military_danger_area">Farezone</string>
|
<string name="poi_military_danger_area">Farezone</string>
|
||||||
<string name="poi_military_range">Militærsøvelsesområde</string>
|
<string name="poi_military_range">Militærsøvelsesområde</string>
|
||||||
<string name="poi_military_naval_base">Militærflådebase</string>
|
<string name="poi_military_naval_base">Militærflådebase</string>
|
||||||
<string name="poi_military_nuclear_explosion_site">Nuklear eksplosion område</string>
|
<string name="poi_military_nuclear_explosion_site">Nuklear eksplosionsområde</string>
|
||||||
<string name="poi_city">Storby</string>
|
<string name="poi_city">Storby</string>
|
||||||
<string name="poi_town">By</string>
|
<string name="poi_town">By</string>
|
||||||
<string name="poi_village">Landsby</string>
|
<string name="poi_village">Landsby</string>
|
||||||
|
|
|
@ -1860,4 +1860,6 @@ Nogle af de centrale funktioner:
|
||||||
<string name="lock_screen_request_explanation">%1$s behøver denne tilladelse til at slukke for skærmen for strømbesparende funktion.</string>
|
<string name="lock_screen_request_explanation">%1$s behøver denne tilladelse til at slukke for skærmen for strømbesparende funktion.</string>
|
||||||
<string name="wake_on_voice_never">Aldrig</string>
|
<string name="wake_on_voice_never">Aldrig</string>
|
||||||
<string name="advanced_settings">Avanceret</string>
|
<string name="advanced_settings">Avanceret</string>
|
||||||
|
<string name="osmo_use_https_descr">Brug sikker forbindelse med server</string>
|
||||||
|
<string name="osmo_use_https">Brug https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1861,7 +1861,7 @@ Afganistán, Albania, Alemania, Andorra, Angola, Anguila, Antigua y Barbuda, Ant
|
||||||
<string name="distance">Distancia</string>
|
<string name="distance">Distancia</string>
|
||||||
<string name="record_plugin_name">Grabar tus viajes</string>
|
<string name="record_plugin_name">Grabar tus viajes</string>
|
||||||
<string name="record_plugin_description">Guardar tus trazas tocando un botón en la pantalla del mapa. Mostrar las opciones para grabar tus viajes a archivos GPX locales o usando un servicio web en línea.</string>
|
<string name="record_plugin_description">Guardar tus trazas tocando un botón en la pantalla del mapa. Mostrar las opciones para grabar tus viajes a archivos GPX locales o usando un servicio web en línea.</string>
|
||||||
<string name="rendering_attr_publicTransportMode_name">Rutas de autobús, trolebús, transporte</string>
|
<string name="rendering_attr_publicTransportMode_name">Rutas de autobús, trolebús y lanzadera</string>
|
||||||
<string name="save_track_to_gpx_globally">Guardar traza en fichero GPX</string>
|
<string name="save_track_to_gpx_globally">Guardar traza en fichero GPX</string>
|
||||||
<string name="save_track_interval_globally">Intervalo de guardado</string>
|
<string name="save_track_interval_globally">Intervalo de guardado</string>
|
||||||
<string name="save_track_to_gpx_globally_descr">El registro de la posición en un archivo GPX puede activarse o desactivarse usando el control de grabación GPX en la pantalla del mapa</string>
|
<string name="save_track_to_gpx_globally_descr">El registro de la posición en un archivo GPX puede activarse o desactivarse usando el control de grabación GPX en la pantalla del mapa</string>
|
||||||
|
@ -1915,8 +1915,10 @@ Afganistán, Albania, Alemania, Andorra, Angola, Anguila, Antigua y Barbuda, Ant
|
||||||
<string name="rendering_attr_busRoutes_name">Rutas de autobús</string>
|
<string name="rendering_attr_busRoutes_name">Rutas de autobús</string>
|
||||||
<string name="rendering_category_hide">Ocultar</string>
|
<string name="rendering_category_hide">Ocultar</string>
|
||||||
<string name="wake_on_voice">Encender pantalla</string>
|
<string name="wake_on_voice">Encender pantalla</string>
|
||||||
<string name="wake_on_voice_descr">Encender pantalla al aproximarse a un giro</string>
|
<string name="wake_on_voice_descr">Encender la pantalla al aproximarse a un giro</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s necesita este permiso para apagar la pantalla para la característica de ahorro energético.</string>
|
<string name="lock_screen_request_explanation">%1$s necesita este permiso para apagar la pantalla para la característica de ahorro energético.</string>
|
||||||
<string name="wake_on_voice_never">Nunca</string>
|
<string name="wake_on_voice_never">Nunca</string>
|
||||||
<string name="advanced_settings">Avanzado</string>
|
<string name="advanced_settings">Avanzado</string>
|
||||||
|
<string name="osmo_use_https_descr">Usar conexión segura con el servidor</string>
|
||||||
|
<string name="osmo_use_https">Usar https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -2027,4 +2027,5 @@ Si consiglia di aggiungere uno o più punti intermedi per migliorarne le prestaz
|
||||||
<string name="wake_on_voice_never">Mai</string>
|
<string name="wake_on_voice_never">Mai</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s necessita di questa autorizzazione per spegnere lo schermo per la funzionalità di risparmio energetico.</string>
|
<string name="lock_screen_request_explanation">%1$s necessita di questa autorizzazione per spegnere lo schermo per la funzionalità di risparmio energetico.</string>
|
||||||
<string name="advanced_settings">Avanzato</string>
|
<string name="advanced_settings">Avanzato</string>
|
||||||
|
<string name="osmo_use_https">Usa https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -2085,4 +2085,5 @@
|
||||||
<string name="lock_screen_request_explanation">%1$의 절전 기능에 대한 화면을 해제 하려면 이 권한이 필요 합니다.</string>
|
<string name="lock_screen_request_explanation">%1$의 절전 기능에 대한 화면을 해제 하려면 이 권한이 필요 합니다.</string>
|
||||||
<string name="wake_on_voice_never">결코</string>
|
<string name="wake_on_voice_never">결코</string>
|
||||||
<string name="advanced_settings">고급</string>
|
<string name="advanced_settings">고급</string>
|
||||||
|
<string name="osmo_use_https">Https를 사용하기</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1878,4 +1878,7 @@ OsmAnd yra aktyviai tobulinama ir mūsų projektas bei jo tolesnis progresas pri
|
||||||
<string name="wake_on_voice_descr">Įjungti ekraną artėjant prie posūkio</string>
|
<string name="wake_on_voice_descr">Įjungti ekraną artėjant prie posūkio</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s prašo leidimo išjungti ekraną tausojant energiją.</string>
|
<string name="lock_screen_request_explanation">%1$s prašo leidimo išjungti ekraną tausojant energiją.</string>
|
||||||
<string name="wake_on_voice_never">Niekada</string>
|
<string name="wake_on_voice_never">Niekada</string>
|
||||||
|
<string name="osmo_use_https">Naudoti https</string>
|
||||||
|
<string name="osmo_use_https_descr">Prie serverio jungtis saugiu ryšiu</string>
|
||||||
|
<string name="advanced_settings">Plačiau</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -2008,4 +2008,6 @@ Afghanistan, Albanie, Algerije, Andorra, Angola, Anguilla, Antigua en Barbuda, A
|
||||||
<string name="wake_on_voice_never">Nooit</string>
|
<string name="wake_on_voice_never">Nooit</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s heeft toestemming nodig om het scherm uit te kunnen schakelen (als onderdeel van de stroombesparende functie).</string>
|
<string name="lock_screen_request_explanation">%1$s heeft toestemming nodig om het scherm uit te kunnen schakelen (als onderdeel van de stroombesparende functie).</string>
|
||||||
<string name="advanced_settings">Geavanceerd</string>
|
<string name="advanced_settings">Geavanceerd</string>
|
||||||
|
<string name="osmo_use_https_descr">Gebruik beveiligde verbinding met server</string>
|
||||||
|
<string name="osmo_use_https">Gebruik https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1781,4 +1781,6 @@ Si cussigiat de annànghere unu o prus puntos intermedios pro megiorare sas pres
|
||||||
<string name="wake_on_voice_never">Mai</string>
|
<string name="wake_on_voice_never">Mai</string>
|
||||||
<string name="lock_screen_request_explanation">%1$s tenet bisòngiu de custa autorizatzione pro istudare s\'ischermu pro sa funtzionalidade de rispàrmiu energèticu.</string>
|
<string name="lock_screen_request_explanation">%1$s tenet bisòngiu de custa autorizatzione pro istudare s\'ischermu pro sa funtzionalidade de rispàrmiu energèticu.</string>
|
||||||
<string name="advanced_settings">Avantzadas</string>
|
<string name="advanced_settings">Avantzadas</string>
|
||||||
|
<string name="osmo_use_https_descr">Imprea una cunnessione segura cun su server</string>
|
||||||
|
<string name="osmo_use_https">Imprea https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -2034,4 +2034,6 @@ Afganistan, Albánsko, Alžírsko, Andora, Angola, Anguilla, Antigua a Barbuda,
|
||||||
<string name="lock_screen_request_explanation">%1$s potrebuje toto oprávnenie na vypnutie obrazovky pre šetrenie energiou.</string>
|
<string name="lock_screen_request_explanation">%1$s potrebuje toto oprávnenie na vypnutie obrazovky pre šetrenie energiou.</string>
|
||||||
<string name="wake_on_voice_never">Nikdy</string>
|
<string name="wake_on_voice_never">Nikdy</string>
|
||||||
<string name="advanced_settings">Pokročilé</string>
|
<string name="advanced_settings">Pokročilé</string>
|
||||||
|
<string name="osmo_use_https_descr">Použiť zabezpečené spojenie so serverom</string>
|
||||||
|
<string name="osmo_use_https">Použiť HTTPS</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1731,4 +1731,5 @@
|
||||||
<string name="lock_screen_request_explanation">%1$s behöver denna tillåtelse för att stänga av skärmen för strömsparningsfunktionen.</string>
|
<string name="lock_screen_request_explanation">%1$s behöver denna tillåtelse för att stänga av skärmen för strömsparningsfunktionen.</string>
|
||||||
<string name="wake_on_voice_never">Aldrig</string>
|
<string name="wake_on_voice_never">Aldrig</string>
|
||||||
<string name="advanced_settings">Avancerat</string>
|
<string name="advanced_settings">Avancerat</string>
|
||||||
|
<string name="osmo_use_https">Använd https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
|
@ -1855,4 +1855,6 @@
|
||||||
<string name="lock_screen_request_explanation">為了節電作用 %1$s 需要此許可權才能關閉螢幕。</string>
|
<string name="lock_screen_request_explanation">為了節電作用 %1$s 需要此許可權才能關閉螢幕。</string>
|
||||||
<string name="wake_on_voice_never">永不</string>
|
<string name="wake_on_voice_never">永不</string>
|
||||||
<string name="advanced_settings">進階</string>
|
<string name="advanced_settings">進階</string>
|
||||||
|
<string name="osmo_use_https_descr">使用安全協定連線於伺服器</string>
|
||||||
|
<string name="osmo_use_https">使用 https</string>
|
||||||
</resources>
|
</resources>
|
||||||
|
|
Loading…
Reference in a new issue