125 lines
4.0 KiB
Kotlin
125 lines
4.0 KiB
Kotlin
/*
|
||
* Copyright (c) 2020 Hemanth Savarla.
|
||
*
|
||
* Licensed under the GNU General Public License v3
|
||
*
|
||
* This 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 of the License, or (at your option) any later version.
|
||
*
|
||
* This software 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.
|
||
*
|
||
*/
|
||
package code.name.monkey.retromusic.extensions
|
||
|
||
import android.animation.ObjectAnimator
|
||
import android.content.Context
|
||
import android.view.LayoutInflater
|
||
import android.view.View
|
||
import android.view.ViewGroup
|
||
import android.view.ViewTreeObserver
|
||
import android.view.inputmethod.InputMethodManager
|
||
import android.widget.EditText
|
||
import androidx.annotation.LayoutRes
|
||
import androidx.core.animation.doOnEnd
|
||
import androidx.core.animation.doOnStart
|
||
import code.name.monkey.appthemehelper.ThemeStore
|
||
import code.name.monkey.appthemehelper.util.TintHelper
|
||
import com.google.android.material.bottomsheet.BottomSheetBehavior
|
||
import com.google.android.material.imageview.ShapeableImageView
|
||
import com.google.android.material.shape.ShapeAppearanceModel
|
||
|
||
@Suppress("UNCHECKED_CAST")
|
||
fun <T : View> ViewGroup.inflate(@LayoutRes layout: Int): T {
|
||
return LayoutInflater.from(context).inflate(layout, this, false) as T
|
||
}
|
||
|
||
fun View.show() {
|
||
visibility = View.VISIBLE
|
||
}
|
||
|
||
fun View.hide() {
|
||
visibility = View.GONE
|
||
}
|
||
|
||
fun View.hidden() {
|
||
visibility = View.INVISIBLE
|
||
}
|
||
|
||
fun View.showOrHide(show: Boolean) = if (show) show() else hide()
|
||
|
||
fun EditText.appHandleColor(): EditText {
|
||
TintHelper.colorHandles(this, ThemeStore.accentColor(context))
|
||
return this
|
||
}
|
||
|
||
fun View.translateYAnimate(value: Float) {
|
||
ObjectAnimator.ofFloat(this, "translationY", value)
|
||
.apply {
|
||
duration = 300
|
||
doOnStart {
|
||
if (value == 0f) {
|
||
show()
|
||
}
|
||
}
|
||
doOnEnd {
|
||
if (value != 0f) {
|
||
hide()
|
||
}
|
||
}
|
||
start()
|
||
}
|
||
}
|
||
|
||
fun BottomSheetBehavior<*>.peekHeightAnimate(value: Int) {
|
||
ObjectAnimator.ofInt(this, "peekHeight", value)
|
||
.apply {
|
||
duration = 300
|
||
start()
|
||
}
|
||
}
|
||
|
||
fun View.focusAndShowKeyboard() {
|
||
/**
|
||
* This is to be called when the window already has focus.
|
||
*/
|
||
fun View.showTheKeyboardNow() {
|
||
if (isFocused) {
|
||
post {
|
||
// We still post the call, just in case we are being notified of the windows focus
|
||
// but InputMethodManager didn't get properly setup yet.
|
||
val imm =
|
||
context.getSystemService(Context.INPUT_METHOD_SERVICE) as InputMethodManager
|
||
imm.showSoftInput(this, InputMethodManager.SHOW_IMPLICIT)
|
||
}
|
||
}
|
||
}
|
||
|
||
requestFocus()
|
||
if (hasWindowFocus()) {
|
||
// No need to wait for the window to get focus.
|
||
showTheKeyboardNow()
|
||
} else {
|
||
// We need to wait until the window gets focus.
|
||
viewTreeObserver.addOnWindowFocusChangeListener(
|
||
object : ViewTreeObserver.OnWindowFocusChangeListener {
|
||
override fun onWindowFocusChanged(hasFocus: Boolean) {
|
||
// This notification will arrive just before the InputMethodManager gets set up.
|
||
if (hasFocus) {
|
||
this@focusAndShowKeyboard.showTheKeyboardNow()
|
||
// It’s very important to remove this listener once we are done.
|
||
viewTreeObserver.removeOnWindowFocusChangeListener(this)
|
||
}
|
||
}
|
||
})
|
||
}
|
||
}
|
||
|
||
fun ShapeableImageView.setCircleShape(boolean: Boolean) {
|
||
addOnLayoutChangeListener { _, _, _, _, _, _, _, _, _ ->
|
||
val radius = width / 2f
|
||
shapeAppearanceModel = ShapeAppearanceModel().withCornerSize(radius)
|
||
}
|
||
} |