2018-06-13 19:14:08 +02:00
|
|
|
package net.osmand.telegram.utils
|
|
|
|
|
|
|
|
import android.content.Context
|
|
|
|
import net.osmand.telegram.R
|
|
|
|
import net.osmand.telegram.TelegramApplication
|
2018-08-09 15:45:39 +02:00
|
|
|
import java.text.DateFormatSymbols
|
2018-06-13 19:14:08 +02:00
|
|
|
import java.text.DecimalFormat
|
|
|
|
import java.text.MessageFormat
|
2018-08-09 15:45:39 +02:00
|
|
|
import java.text.SimpleDateFormat
|
|
|
|
import java.util.*
|
2018-06-13 19:14:08 +02:00
|
|
|
|
|
|
|
object OsmandFormatter {
|
|
|
|
|
|
|
|
val METERS_IN_KILOMETER = 1000f
|
|
|
|
val METERS_IN_ONE_MILE = 1609.344f // 1609.344
|
|
|
|
val METERS_IN_ONE_NAUTICALMILE = 1852f // 1852
|
|
|
|
|
|
|
|
val YARDS_IN_ONE_METER = 1.0936f
|
|
|
|
val FEET_IN_ONE_METER = YARDS_IN_ONE_METER * 3f
|
|
|
|
private val fixed2 = DecimalFormat("0.00")
|
|
|
|
private val fixed1 = DecimalFormat("0.0")
|
2018-08-09 17:16:16 +02:00
|
|
|
|
|
|
|
private const val SHORT_TIME_FORMAT = "%02d:%02d"
|
2018-08-09 17:28:17 +02:00
|
|
|
private const val SIMPLE_TIME_OF_DAY_FORMAT = "HH:mm"
|
2018-08-17 12:52:15 +02:00
|
|
|
private const val SIMPLE_DATE_FORMAT = "dd MMM HH:mm:ss"
|
2018-06-13 19:14:08 +02:00
|
|
|
|
2018-08-17 14:38:24 +02:00
|
|
|
private const val MIN_DURATION_FOR_DATE_FORMAT = 48 * 60 * 60
|
|
|
|
|
2018-08-09 15:45:39 +02:00
|
|
|
private val dateFormatSymbols = DateFormatSymbols.getInstance()
|
|
|
|
private val localDaysStr = getLettersStringArray(dateFormatSymbols.shortWeekdays, 2)
|
|
|
|
|
2018-06-13 19:14:08 +02:00
|
|
|
init {
|
|
|
|
fixed2.minimumFractionDigits = 2
|
|
|
|
fixed1.minimumFractionDigits = 1
|
|
|
|
fixed1.minimumIntegerDigits = 1
|
|
|
|
fixed2.minimumIntegerDigits = 1
|
|
|
|
}
|
|
|
|
|
2018-08-10 12:16:46 +02:00
|
|
|
fun getFormattedDuration(ctx: Context, seconds: Long, short: Boolean = false): String {
|
2018-06-13 19:14:08 +02:00
|
|
|
val hours = seconds / (60 * 60)
|
|
|
|
val minutes = seconds / 60 % 60
|
2018-08-10 10:12:29 +02:00
|
|
|
val secs = seconds - minutes * 60
|
2018-08-09 15:45:39 +02:00
|
|
|
if (short) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return String.format(SHORT_TIME_FORMAT, hours, minutes)
|
2018-08-09 15:45:39 +02:00
|
|
|
}
|
2018-08-08 15:50:24 +02:00
|
|
|
return when {
|
|
|
|
hours > 0 -> {
|
|
|
|
var res = "$hours ${ctx.getString(R.string.shared_string_hour_short)}"
|
|
|
|
if (minutes > 0) {
|
|
|
|
res += " $minutes ${ctx.getString(R.string.shared_string_minute_short)}"
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
2018-08-10 10:12:29 +02:00
|
|
|
minutes > 0 -> {
|
|
|
|
var res = "$minutes ${ctx.getString(R.string.shared_string_minute_short)}"
|
|
|
|
if (secs > 0) {
|
|
|
|
res += " $secs ${ctx.getString(R.string.shared_string_second_short)}"
|
|
|
|
}
|
|
|
|
res
|
|
|
|
}
|
2018-08-08 15:50:24 +02:00
|
|
|
else -> "$seconds ${ctx.getString(R.string.shared_string_second_short)}"
|
2018-06-13 19:14:08 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-14 12:20:16 +02:00
|
|
|
fun getFormattedTime(seconds: Long): String {
|
2018-08-09 15:45:39 +02:00
|
|
|
val calendar = Calendar.getInstance()
|
|
|
|
calendar.timeInMillis = System.currentTimeMillis() + (seconds * 1000)
|
|
|
|
return if (isSameDay(calendar, Calendar.getInstance())) {
|
2018-08-09 17:28:17 +02:00
|
|
|
SimpleDateFormat(SIMPLE_TIME_OF_DAY_FORMAT, Locale.getDefault()).format(calendar.time)
|
2018-08-09 15:45:39 +02:00
|
|
|
} else {
|
2018-08-09 17:28:17 +02:00
|
|
|
SimpleDateFormat(SIMPLE_TIME_OF_DAY_FORMAT, Locale.getDefault()).format(calendar.time) +
|
2018-08-09 15:45:39 +02:00
|
|
|
" " + localDaysStr[calendar.get(Calendar.DAY_OF_WEEK)]
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-08-17 13:03:38 +02:00
|
|
|
fun getFormattedDate(seconds: Long): String =
|
|
|
|
SimpleDateFormat(SIMPLE_DATE_FORMAT, Locale.getDefault()).format(seconds * 1000L)
|
2018-08-17 14:38:24 +02:00
|
|
|
|
|
|
|
fun getListItemLiveTimeDescr(ctx: TelegramApplication, lastUpdated: Int, prefix: String = ""): String {
|
|
|
|
return if (lastUpdated > 0) {
|
2018-08-17 14:22:04 +02:00
|
|
|
val duration = System.currentTimeMillis() / 1000 - lastUpdated
|
2018-08-27 11:28:38 +02:00
|
|
|
when {
|
|
|
|
duration > MIN_DURATION_FOR_DATE_FORMAT -> prefix + getFormattedDate(lastUpdated.toLong())
|
|
|
|
duration > 0 -> prefix + getFormattedDuration(ctx, duration) + " " + ctx.getString(R.string.time_ago)
|
|
|
|
else -> ""
|
2018-08-17 14:22:04 +02:00
|
|
|
}
|
2018-08-17 14:38:24 +02:00
|
|
|
} else {
|
|
|
|
""
|
2018-08-17 14:22:04 +02:00
|
|
|
}
|
|
|
|
}
|
2018-08-17 12:29:57 +02:00
|
|
|
|
2018-06-13 19:14:08 +02:00
|
|
|
fun calculateRoundedDist(distInMeters: Double, ctx: TelegramApplication): Double {
|
|
|
|
val mc = ctx.settings.metricsConstants
|
|
|
|
var mainUnitInMeter = 1.0
|
|
|
|
var metersInSecondUnit = METERS_IN_KILOMETER.toDouble()
|
|
|
|
if (mc == MetricsConstants.MILES_AND_FEET) {
|
|
|
|
mainUnitInMeter = FEET_IN_ONE_METER.toDouble()
|
|
|
|
metersInSecondUnit = METERS_IN_ONE_MILE.toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_METERS) {
|
|
|
|
mainUnitInMeter = 1.0
|
|
|
|
metersInSecondUnit = METERS_IN_ONE_MILE.toDouble()
|
|
|
|
} else if (mc == MetricsConstants.NAUTICAL_MILES) {
|
|
|
|
mainUnitInMeter = 1.0
|
|
|
|
metersInSecondUnit = METERS_IN_ONE_NAUTICALMILE.toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_YARDS) {
|
|
|
|
mainUnitInMeter = YARDS_IN_ONE_METER.toDouble()
|
|
|
|
metersInSecondUnit = METERS_IN_ONE_MILE.toDouble()
|
|
|
|
}
|
|
|
|
|
|
|
|
// 1, 2, 5, 10, 20, 50, 100, 200, 500, 1000 ...
|
|
|
|
var generator = 1
|
|
|
|
var pointer: Byte = 1
|
|
|
|
var point = mainUnitInMeter
|
|
|
|
var roundDist = 1.0
|
|
|
|
while (distInMeters * point > generator) {
|
|
|
|
roundDist = generator / point
|
|
|
|
if (pointer++ % 3 == 2) {
|
|
|
|
generator = generator * 5 / 2
|
|
|
|
} else {
|
|
|
|
generator *= 2
|
|
|
|
}
|
|
|
|
|
|
|
|
if (point == mainUnitInMeter && metersInSecondUnit * mainUnitInMeter * 0.9 <= generator) {
|
|
|
|
point = 1 / metersInSecondUnit
|
|
|
|
generator = 1
|
|
|
|
pointer = 1
|
|
|
|
}
|
|
|
|
}
|
|
|
|
//Miles exceptions: 2000ft->0.5mi, 1000ft->0.25mi, 1000yd->0.5mi, 500yd->0.25mi, 1000m ->0.5mi, 500m -> 0.25mi
|
|
|
|
if (mc == MetricsConstants.MILES_AND_METERS && roundDist == 1000.0) {
|
|
|
|
roundDist = (0.5f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_METERS && roundDist == 500.0) {
|
|
|
|
roundDist = (0.25f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_FEET && roundDist == 2000 / FEET_IN_ONE_METER.toDouble()) {
|
|
|
|
roundDist = (0.5f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_FEET && roundDist == 1000 / FEET_IN_ONE_METER.toDouble()) {
|
|
|
|
roundDist = (0.25f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_YARDS && roundDist == 1000 / YARDS_IN_ONE_METER.toDouble()) {
|
|
|
|
roundDist = (0.5f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_YARDS && roundDist == 500 / YARDS_IN_ONE_METER.toDouble()) {
|
|
|
|
roundDist = (0.25f * METERS_IN_ONE_MILE).toDouble()
|
|
|
|
}
|
|
|
|
return roundDist
|
|
|
|
}
|
|
|
|
|
|
|
|
fun getFormattedRoundDistanceKm(meters: Float, digits: Int, ctx: TelegramApplication): String {
|
|
|
|
val mainUnitStr = R.string.km
|
|
|
|
val mainUnitInMeters = METERS_IN_KILOMETER
|
|
|
|
return if (digits == 0) {
|
|
|
|
(meters / mainUnitInMeters + 0.5).toInt().toString() + " " + ctx.getString(mainUnitStr) //$NON-NLS-1$
|
|
|
|
} else if (digits == 1) {
|
|
|
|
fixed1.format((meters / mainUnitInMeters).toDouble()) + " " + ctx.getString(mainUnitStr)
|
|
|
|
} else {
|
|
|
|
fixed2.format((meters / mainUnitInMeters).toDouble()) + " " + ctx.getString(mainUnitStr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
@JvmOverloads
|
2018-08-09 17:16:16 +02:00
|
|
|
fun getFormattedDistance(meters: Float, ctx: TelegramApplication, forceTrailingZeros: Boolean = true): String {
|
2018-06-13 19:14:08 +02:00
|
|
|
val format1 = if (forceTrailingZeros) "{0,number,0.0} " else "{0,number,0.#} "
|
|
|
|
val format2 = if (forceTrailingZeros) "{0,number,0.00} " else "{0,number,0.##} "
|
|
|
|
|
|
|
|
val mc = ctx.settings.metricsConstants
|
|
|
|
val mainUnitStr: Int
|
|
|
|
val mainUnitInMeters: Float
|
|
|
|
if (mc == MetricsConstants.KILOMETERS_AND_METERS) {
|
|
|
|
mainUnitStr = R.string.km
|
|
|
|
mainUnitInMeters = METERS_IN_KILOMETER
|
|
|
|
} else if (mc == MetricsConstants.NAUTICAL_MILES) {
|
|
|
|
mainUnitStr = R.string.nm
|
|
|
|
mainUnitInMeters = METERS_IN_ONE_NAUTICALMILE
|
|
|
|
} else {
|
|
|
|
mainUnitStr = R.string.mile
|
|
|
|
mainUnitInMeters = METERS_IN_ONE_MILE
|
|
|
|
}
|
|
|
|
|
|
|
|
if (meters >= 100 * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return (meters / mainUnitInMeters + 0.5).toInt().toString() + " " + ctx.getString(mainUnitStr) //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (meters > 9.99f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format1 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (meters > 0.999f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format2 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (mc == MetricsConstants.MILES_AND_FEET && meters > 0.249f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format2 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (mc == MetricsConstants.MILES_AND_METERS && meters > 0.249f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format2 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (mc == MetricsConstants.MILES_AND_YARDS && meters > 0.249f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format2 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else if (mc == MetricsConstants.NAUTICAL_MILES && meters > 0.99f * mainUnitInMeters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
return MessageFormat.format(format2 + ctx.getString(mainUnitStr), meters / mainUnitInMeters).replace('\n', ' ') //$NON-NLS-1$
|
2018-06-13 19:14:08 +02:00
|
|
|
} else {
|
|
|
|
if (mc == MetricsConstants.KILOMETERS_AND_METERS || mc == MetricsConstants.MILES_AND_METERS) {
|
|
|
|
return (meters + 0.5).toInt().toString() + " " + ctx.getString(R.string.m) //$NON-NLS-1$
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_FEET) {
|
|
|
|
val feet = (meters * FEET_IN_ONE_METER + 0.5).toInt()
|
|
|
|
return feet.toString() + " " + ctx.getString(R.string.foot) //$NON-NLS-1$
|
|
|
|
} else if (mc == MetricsConstants.MILES_AND_YARDS) {
|
|
|
|
val yards = (meters * YARDS_IN_ONE_METER + 0.5).toInt()
|
|
|
|
return yards.toString() + " " + ctx.getString(R.string.yard) //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
return (meters + 0.5).toInt().toString() + " " + ctx.getString(R.string.m) //$NON-NLS-1$
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun getFormattedAlt(alt: Double, ctx: TelegramApplication): String {
|
|
|
|
val mc = ctx.settings.metricsConstants
|
|
|
|
return if (mc == MetricsConstants.KILOMETERS_AND_METERS) {
|
|
|
|
(alt + 0.5).toInt().toString() + " " + ctx.getString(R.string.m)
|
|
|
|
} else {
|
|
|
|
(alt * FEET_IN_ONE_METER + 0.5).toInt().toString() + " " + ctx.getString(R.string.foot)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fun getFormattedSpeed(metersperseconds: Float, ctx: TelegramApplication): String {
|
|
|
|
val mc = ctx.settings.speedConstants
|
|
|
|
val kmh = metersperseconds * 3.6f
|
|
|
|
if (mc == SpeedConstants.KILOMETERS_PER_HOUR) {
|
|
|
|
// e.g. car case and for high-speeds: Display rounded to 1 km/h (5% precision at 20 km/h)
|
|
|
|
if (kmh >= 20) {
|
|
|
|
return Math.round(kmh).toString() + " " + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
// for smaller values display 1 decimal digit x.y km/h, (0.5% precision at 20 km/h)
|
|
|
|
val kmh10 = Math.round(kmh * 10f)
|
|
|
|
return (kmh10 / 10f).toString() + " " + mc.toShortString(ctx)
|
|
|
|
} else if (mc == SpeedConstants.MILES_PER_HOUR) {
|
|
|
|
val mph = kmh * METERS_IN_KILOMETER / METERS_IN_ONE_MILE
|
|
|
|
if (mph >= 20) {
|
|
|
|
return Math.round(mph).toString() + " " + mc.toShortString(ctx)
|
|
|
|
} else {
|
|
|
|
val mph10 = Math.round(mph * 10f)
|
|
|
|
return (mph10 / 10f).toString() + " " + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
} else if (mc == SpeedConstants.NAUTICALMILES_PER_HOUR) {
|
|
|
|
val mph = kmh * METERS_IN_KILOMETER / METERS_IN_ONE_NAUTICALMILE
|
|
|
|
if (mph >= 20) {
|
|
|
|
return Math.round(mph).toString() + " " + mc.toShortString(ctx)
|
|
|
|
} else {
|
|
|
|
val mph10 = Math.round(mph * 10f)
|
|
|
|
return (mph10 / 10f).toString() + " " + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
} else if (mc == SpeedConstants.MINUTES_PER_KILOMETER) {
|
|
|
|
if (metersperseconds < 0.111111111) {
|
|
|
|
return "-" + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
val minperkm = METERS_IN_KILOMETER / (metersperseconds * 60)
|
|
|
|
if (minperkm >= 10) {
|
|
|
|
return Math.round(minperkm).toString() + " " + mc.toShortString(ctx)
|
|
|
|
} else {
|
|
|
|
val mph10 = Math.round(minperkm * 10f)
|
|
|
|
return (mph10 / 10f).toString() + " " + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
} else if (mc == SpeedConstants.MINUTES_PER_MILE) {
|
|
|
|
if (metersperseconds < 0.111111111) {
|
|
|
|
return "-" + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
val minperm = METERS_IN_ONE_MILE / (metersperseconds * 60)
|
|
|
|
if (minperm >= 10) {
|
|
|
|
return Math.round(minperm).toString() + " " + mc.toShortString(ctx)
|
|
|
|
} else {
|
|
|
|
val mph10 = Math.round(minperm * 10f)
|
|
|
|
return (mph10 / 10f).toString() + " " + mc.toShortString(ctx)
|
|
|
|
}
|
|
|
|
} else
|
|
|
|
/*if (mc == SpeedConstants.METERS_PER_SECOND) */ {
|
|
|
|
if (metersperseconds >= 10) {
|
|
|
|
return Math.round(metersperseconds).toString() + " " + SpeedConstants.METERS_PER_SECOND.toShortString(ctx)
|
|
|
|
}
|
|
|
|
// for smaller values display 1 decimal digit x.y km/h, (0.5% precision at 20 km/h)
|
|
|
|
val kmh10 = Math.round(metersperseconds * 10f)
|
|
|
|
return (kmh10 / 10f).toString() + " " + SpeedConstants.METERS_PER_SECOND.toShortString(ctx)
|
|
|
|
}
|
|
|
|
}
|
2018-08-09 15:45:39 +02:00
|
|
|
|
|
|
|
private fun isSameDay(cal1: Calendar, cal2: Calendar): Boolean {
|
|
|
|
return cal1.get(Calendar.ERA) == cal2.get(Calendar.ERA) &&
|
|
|
|
cal1.get(Calendar.YEAR) == cal2.get(Calendar.YEAR) &&
|
|
|
|
cal1.get(Calendar.DAY_OF_YEAR) == cal2.get(Calendar.DAY_OF_YEAR)
|
|
|
|
}
|
2018-06-13 19:14:08 +02:00
|
|
|
|
2018-08-09 15:45:39 +02:00
|
|
|
private fun getLettersStringArray(strings: Array<String>, letters: Int): Array<String?> {
|
|
|
|
val newStrings = arrayOfNulls<String>(strings.size)
|
|
|
|
for (i in strings.indices) {
|
|
|
|
if (strings[i].length > letters) {
|
2018-08-09 17:16:16 +02:00
|
|
|
newStrings[i] = (strings[i].substring(0, letters)).capitalize()
|
2018-08-09 15:45:39 +02:00
|
|
|
} else {
|
2018-08-09 17:16:16 +02:00
|
|
|
newStrings[i] = strings[i].capitalize()
|
2018-08-09 15:45:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return newStrings
|
|
|
|
}
|
|
|
|
|
2018-06-13 19:14:08 +02:00
|
|
|
enum class MetricsConstants private constructor(private val key: Int) {
|
|
|
|
KILOMETERS_AND_METERS(R.string.si_km_m),
|
|
|
|
MILES_AND_FEET(R.string.si_mi_feet),
|
|
|
|
MILES_AND_METERS(R.string.si_mi_meters),
|
|
|
|
MILES_AND_YARDS(R.string.si_mi_yard),
|
|
|
|
NAUTICAL_MILES(R.string.si_nm);
|
|
|
|
|
|
|
|
fun toHumanString(ctx: Context): String {
|
|
|
|
return ctx.getString(key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
enum class SpeedConstants private constructor(private val key: Int, private val descr: Int) {
|
|
|
|
KILOMETERS_PER_HOUR(R.string.km_h, R.string.si_kmh),
|
|
|
|
MILES_PER_HOUR(R.string.mile_per_hour, R.string.si_mph),
|
|
|
|
METERS_PER_SECOND(R.string.m_s, R.string.si_m_s),
|
|
|
|
MINUTES_PER_MILE(R.string.min_mile, R.string.si_min_m),
|
|
|
|
MINUTES_PER_KILOMETER(R.string.min_km, R.string.si_min_km),
|
|
|
|
NAUTICALMILES_PER_HOUR(R.string.nm_h, R.string.si_nm_h);
|
|
|
|
|
|
|
|
fun toHumanString(ctx: Context): String {
|
|
|
|
return ctx.getString(descr)
|
|
|
|
}
|
|
|
|
|
|
|
|
fun toShortString(ctx: Context): String {
|
|
|
|
return ctx.getString(key)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|