session-android/app/src/main/java/org/thoughtcrime/securesms/keyboard/emoji/KeyboardPageSearchView.kt

195 lines
6 KiB
Kotlin
Raw Normal View History

Add emoji reacts support (#889) * feat: Add emoji reacts support * Remove message multi-selection * Add emoji reaction model * Add emoji reaction panel * Blur reacts panel background * Show emoji keyboard * Add emoji sprites * Update reaction proto * Emoji database updates * Emoji database refactor * Emoji reaction persistence * Optimize reactions retrieval * Fix emoji group query * Display emojis * Fix emoji persistence * Cleanup * Persistence refactor * Add reactions bottom sheet * Cleanup * Ui tweaks * React with any emoji * Show emoji react notifications * Remove reaction * Show reactions modal on long press * Click to react (+1) with an emoji * Click to react with an emoji * Enable emoji expand/collapse * fix: some compile issues from merge conflicts * fix: compile issues merging quote and media message UI * fix: xml IDs and adding in legacy is selected for future inclusion * Fix view constraints * Fix merge issue * Add message selection option in conversation context menu * Add sogs emoji integration * Handle sogs emoji reactions * Enable sending/deleting sogs emojis * fix: improve the visible message layout * fix: add file IDs to request parameters for message send (#940) * Fix open group polling from seqno instead of last hash (#939) * fix: reset seqno to get recent messages from open groups * build: upgrade build numbers * fix: actually run the migration * Using StringBuilder to construct request url * Fix reaction filter * fix: is_mms added in second projection query * Update default emojis * fix: include legacy and new open groups in server ID tracking (#941) * feat: add hidden moderator and admin roles, separated as they may be used independently in future (#942) * Cleanup * Fix view constraints * Add reactions capability check * Fix reactions alignment * Ui fixes * Display reactions list * feat: add formatted count strings * fix: account for negatives and add tests * Migrate old official open group locations for polling and adding (#932) * feat: adding in first part of open group migrations and tests for migration logic / helpers * feat: test code and migration logic for open groups in the case of no conflicts * feat: add in extra test cases and refactor code for migrator * refactor: migrate open group join URLs and references to server in adding new open groups to catch legacy and re-write it * refactor: joining open groups using OpenGroupUrlParser.kt now * fix: add in compile issues for renamed OpenGroupApi.kt from OpenGroupV2 * fix: prevent duplicates of http/https for new open group DNS and prevent adding new groups based on public key * fix: room and server swapped parameters * fix: replace default server for config messages * fix: actually using public key to de-dupe didn't work for rooms * build: bump version code and name * Display reactions list on open groups for moderators * Ui tweaks * Ui tweaks for moderation * Refactor * fix: compile issue * fix: de-duping joined queries in the get X from cursor * Restore import * fix: colouring the reaction overlay scrubber * fix: highlight colour, show reaction count if 1 or above * Cleanup * fix: light mode accent * fix: light / dark mode themeing in reactions dialog fragment * Emoji notification blinded id check * fix: show reaction list correctly and pass isUserModerator to bind methods * fix: remove unnecessary places for the moderator * fix: X button for removing own react not showing up properly * feat: add clear all header view * fix: migrate the clear all to the correct location * fix: use display instead of base * Truncate emoji sender ids * feat: add notify thread function in thread db * Notify threads on reaction received * fix: design fixes for the reaction list * fix: emoji reactions bottom sheet dialog UI designs * feat: add unsupported emoji reaction * fix: crash and doing vector properly * Fix reaction database queries * Fix background open group adder job * Show new open group reactions * Fetch a maximum of 5 reactors * Handle open group reactions polling conflicts * Add count to user reaction * Show number of additional reactors * fix: unreads set same as the unread query * fix: design changes * fix: update dependency to improve flexboxlayout behaviour, design consistencies * Add select message icon and update long press menu items order and wording * Fix crash on reactors dialog * fix: colours and backgrounds to match designs * fix: add header in recipient item * fix: margins * fix: alignments and layout issues for emoji reactions view * feat: add overflow previews and logic for overflow * Dim action bar * Add emoji search * Search index fix * Set count for 1:1 and closed group reactions when inserting in local database * Use on screen toolbar to allow overlaying * Show/hide scroll to bottom button * feat: add extended properties so it doesn't collapse on re-bind * Cleanup * feat: prevent keeping extended on rebinding if we get a new message ID * fix: long press works on devices now, fix release lint issue and crash for emoji search DBs from emoji builds * Display message timestamp * Fix modal items alignment * fix: sort order and emoji count in compareTo * Scale down really large messages to fit * Prevent closed group crash * Fix reaction author Co-authored-by: charles <charles@oxen.io> Co-authored-by: jubb <hjubb@users.noreply.github.com>
2022-09-04 13:03:32 +02:00
package org.thoughtcrime.securesms.keyboard.emoji
import android.animation.Animator
import android.content.Context
import android.content.res.ColorStateList
import android.graphics.drawable.ColorDrawable
import android.util.AttributeSet
import android.view.View
import android.widget.EditText
import androidx.appcompat.widget.AppCompatImageView
import androidx.constraintlayout.widget.ConstraintLayout
import androidx.core.content.ContextCompat
import androidx.core.content.res.use
import androidx.core.view.ViewCompat
import androidx.core.view.isVisible
import androidx.core.widget.ImageViewCompat
import androidx.core.widget.doAfterTextChanged
import network.loki.messenger.R
import org.thoughtcrime.securesms.animation.AnimationCompleteListener
import org.thoughtcrime.securesms.animation.ResizeAnimation
import org.thoughtcrime.securesms.conversation.v2.ViewUtil
private const val REVEAL_DURATION = 250L
/**
* Search bar to be used in the various keyboard views (emoji, sticker, gif)
*/
class KeyboardPageSearchView @JvmOverloads constructor(
context: Context,
attrs: AttributeSet? = null,
defStyleAttr: Int = 0
) : ConstraintLayout(context, attrs, defStyleAttr) {
var callbacks: Callbacks? = null
private var state: State = State.HIDE_REQUESTED
private var targetInputWidth: Int = -1
private val navButton: AppCompatImageView
private val clearButton: AppCompatImageView
private val input: EditText
init {
inflate(context, R.layout.keyboard_pager_search_bar, this)
navButton = findViewById(R.id.emoji_search_nav_icon)
clearButton = findViewById(R.id.emoji_search_clear_icon)
input = findViewById(R.id.emoji_search_entry)
input.doAfterTextChanged {
if (it.isNullOrEmpty()) {
clearButton.setImageDrawable(null)
clearButton.isClickable = false
} else {
clearButton.setImageResource(R.drawable.ic_x)
clearButton.isClickable = true
}
if (it.isNullOrEmpty()) {
callbacks?.onQueryChanged("")
} else {
callbacks?.onQueryChanged(it.toString())
}
}
input.setOnFocusChangeListener { _, hasFocus ->
if (hasFocus) {
callbacks?.onFocusGained()
} else {
callbacks?.onFocusLost()
}
}
clearButton.setOnClickListener { clearQuery() }
context.obtainStyledAttributes(attrs, R.styleable.KeyboardPageSearchView, 0, 0).use { typedArray ->
val showAlways: Boolean = typedArray.getBoolean(R.styleable.KeyboardPageSearchView_show_always, false)
if (showAlways) {
alpha = 1f
state = State.SHOW_REQUESTED
} else {
alpha = 0f
input.layoutParams = input.layoutParams.apply { width = 1 }
state = State.HIDE_REQUESTED
}
input.hint = typedArray.getString(R.styleable.KeyboardPageSearchView_search_hint) ?: ""
val backgroundTint = typedArray.getColor(R.styleable.KeyboardPageSearchView_search_bar_tint, ContextCompat.getColor(context, R.color.signal_background_primary))
val backgroundTintList = ColorStateList.valueOf(backgroundTint)
input.background = ColorDrawable(backgroundTint)
ViewCompat.setBackgroundTintList(findViewById(R.id.emoji_search_nav), backgroundTintList)
ViewCompat.setBackgroundTintList(findViewById(R.id.emoji_search_clear), backgroundTintList)
val iconTint = typedArray.getColorStateList(R.styleable.KeyboardPageSearchView_search_icon_tint) ?: ContextCompat.getColorStateList(context, R.color.signal_icon_tint_tab_selected)
ImageViewCompat.setImageTintList(navButton, iconTint)
ImageViewCompat.setImageTintList(clearButton, iconTint)
val clickOnly: Boolean = typedArray.getBoolean(R.styleable.KeyboardPageSearchView_click_only, false)
if (clickOnly) {
val clickIntercept: View = findViewById(R.id.keyboard_search_click_only)
clickIntercept.isVisible = true
clickIntercept.setOnClickListener { callbacks?.onClicked() }
}
}
}
fun showRequested(): Boolean = state == State.SHOW_REQUESTED
fun enableBackNavigation(enable: Boolean = true) {
navButton.setImageResource(if (enable) R.drawable.ic_arrow_left_24 else R.drawable.ic_search_24)
if (enable) {
navButton.setImageResource(R.drawable.ic_arrow_left_24)
navButton.setOnClickListener { callbacks?.onNavigationClicked() }
} else {
navButton.setImageResource(R.drawable.ic_search_24)
navButton.setOnClickListener(null)
}
}
override fun onSizeChanged(w: Int, h: Int, oldw: Int, oldh: Int) {
targetInputWidth = w - ViewUtil.dpToPx(32) - ViewUtil.dpToPx(90)
}
fun show() {
if (state == State.SHOW_REQUESTED) {
return
}
visibility = VISIBLE
state = State.SHOW_REQUESTED
post {
animate()
.setDuration(REVEAL_DURATION)
.alpha(1f)
.setListener(null)
val resizeAnimation = ResizeAnimation(input, targetInputWidth, input.measuredHeight)
resizeAnimation.duration = REVEAL_DURATION
input.startAnimation(resizeAnimation)
}
}
fun hide() {
if (state == State.HIDE_REQUESTED) {
return
}
state = State.HIDE_REQUESTED
post {
animate()
.setDuration(REVEAL_DURATION)
.alpha(0f)
.setListener(object : AnimationCompleteListener() {
override fun onAnimationEnd(animation: Animator?) {
visibility = INVISIBLE
}
})
val resizeAnimation = ResizeAnimation(input, 1, input.measuredHeight)
resizeAnimation.duration = REVEAL_DURATION
input.startAnimation(resizeAnimation)
}
}
fun presentForEmojiSearch() {
ViewUtil.focusAndShowKeyboard(input)
enableBackNavigation()
}
override fun clearFocus() {
super.clearFocus()
clearChildFocus(input)
}
fun clearQuery() {
input.text.clear()
}
interface Callbacks {
fun onFocusLost() = Unit
fun onFocusGained() = Unit
fun onNavigationClicked() = Unit
fun onQueryChanged(query: String) = Unit
fun onClicked() = Unit
}
enum class State {
SHOW_REQUESTED,
HIDE_REQUESTED
}
}