/*
* This file is part of GNU Taler
* (C) 2020 Taler Systems S.A.
*
* GNU Taler is free software; you can redistribute it and/or modify it under the
* terms of the GNU General Public License as published by the Free Software
* Foundation; either version 3, or (at your option) any later version.
*
* GNU Taler is distributed in the hope that it will be useful, but WITHOUT ANY
* WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
* A PARTICULAR PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with
* GNU Taler; see the file COPYING. If not, see
*/
package net.taler.merchantpos
import android.content.Context
import android.text.format.DateUtils.DAY_IN_MILLIS
import android.text.format.DateUtils.FORMAT_ABBREV_MONTH
import android.text.format.DateUtils.FORMAT_ABBREV_RELATIVE
import android.text.format.DateUtils.FORMAT_NO_YEAR
import android.text.format.DateUtils.FORMAT_SHOW_DATE
import android.text.format.DateUtils.FORMAT_SHOW_TIME
import android.text.format.DateUtils.MINUTE_IN_MILLIS
import android.text.format.DateUtils.formatDateTime
import android.text.format.DateUtils.getRelativeTimeSpanString
import android.view.View
import android.view.View.INVISIBLE
import android.view.View.VISIBLE
import androidx.annotation.StringRes
import androidx.fragment.app.Fragment
import androidx.lifecycle.LiveData
import androidx.lifecycle.MediatorLiveData
import androidx.lifecycle.Observer
import androidx.navigation.NavController
import androidx.navigation.NavDirections
import androidx.navigation.fragment.findNavController
import com.google.android.material.snackbar.BaseTransientBottomBar.ANIMATION_MODE_FADE
import com.google.android.material.snackbar.BaseTransientBottomBar.Duration
import com.google.android.material.snackbar.Snackbar.make
object Utils {
private const val HEX_CHARS = "0123456789ABCDEF"
fun hexStringToByteArray(data: String): ByteArray {
val result = ByteArray(data.length / 2)
for (i in data.indices step 2) {
val firstIndex = HEX_CHARS.indexOf(data[i])
val secondIndex = HEX_CHARS.indexOf(data[i + 1])
val octet = firstIndex.shl(4).or(secondIndex)
result[i.shr(1)] = octet.toByte()
}
return result
}
private val HEX_CHARS_ARRAY = HEX_CHARS.toCharArray()
@Suppress("unused")
fun toHex(byteArray: ByteArray): String {
val result = StringBuffer()
byteArray.forEach {
val octet = it.toInt()
val firstIndex = (octet and 0xF0).ushr(4)
val secondIndex = octet and 0x0F
result.append(HEX_CHARS_ARRAY[firstIndex])
result.append(HEX_CHARS_ARRAY[secondIndex])
}
return result.toString()
}
}
fun View.fadeIn(endAction: () -> Unit = {}) {
if (visibility == VISIBLE) return
alpha = 0f
visibility = VISIBLE
animate().alpha(1f).withEndAction {
if (context != null) endAction.invoke()
}.start()
}
fun View.fadeOut(endAction: () -> Unit = {}) {
if (visibility == INVISIBLE) return
animate().alpha(0f).withEndAction {
if (context == null) return@withEndAction
visibility = INVISIBLE
alpha = 1f
endAction.invoke()
}.start()
}
fun topSnackbar(view: View, text: CharSequence, @Duration duration: Int) {
make(view, text, duration)
.setAnimationMode(ANIMATION_MODE_FADE)
.setAnchorView(R.id.navHostFragment)
.show()
}
fun topSnackbar(view: View, @StringRes resId: Int, @Duration duration: Int) {
topSnackbar(view, view.resources.getText(resId), duration)
}
fun NavDirections.navigate(nav: NavController) = nav.navigate(this)
fun Fragment.navigate(directions: NavDirections) = findNavController().navigate(directions)
fun Long.toRelativeTime(context: Context): CharSequence {
val now = System.currentTimeMillis()
return if (now - this > DAY_IN_MILLIS * 2) {
val flags = FORMAT_SHOW_TIME or FORMAT_SHOW_DATE or FORMAT_ABBREV_MONTH or FORMAT_NO_YEAR
formatDateTime(context, this, flags)
} else getRelativeTimeSpanString(this, now, MINUTE_IN_MILLIS, FORMAT_ABBREV_RELATIVE)
}
class CombinedLiveData(
source1: LiveData,
source2: LiveData,
private val combine: (data1: T?, data2: K?) -> S
) : MediatorLiveData() {
private var data1: T? = null
private var data2: K? = null
init {
super.addSource(source1) { t ->
data1 = t
value = combine(data1, data2)
}
super.addSource(source2) { k ->
data2 = k
value = combine(data1, data2)
}
}
override fun addSource(source: LiveData, onChanged: Observer) {
throw UnsupportedOperationException()
}
override fun removeSource(toRemote: LiveData) {
throw UnsupportedOperationException()
}
}
/**
* Use this with 'when' expressions when you need it to handle all possibilities/branches.
*/
val T.exhaustive: T
get() = this