mirror of
https://github.com/Helium314/HeliBoard.git
synced 2025-04-20 22:29:10 +00:00
156 lines
5.9 KiB
Kotlin
156 lines
5.9 KiB
Kotlin
// SPDX-License-Identifier: GPL-3.0-only
|
|
|
|
package helium314.keyboard.latin
|
|
|
|
import android.content.ClipboardManager
|
|
import android.content.Context
|
|
import android.text.TextUtils
|
|
import androidx.preference.PreferenceManager
|
|
import kotlinx.serialization.encodeToString
|
|
import kotlinx.serialization.json.Json
|
|
import helium314.keyboard.compat.ClipboardManagerCompat
|
|
import helium314.keyboard.latin.settings.Settings
|
|
import kotlin.collections.ArrayList
|
|
|
|
class ClipboardHistoryManager(
|
|
private val latinIME: LatinIME
|
|
) : ClipboardManager.OnPrimaryClipChangedListener {
|
|
|
|
private lateinit var clipboardManager: ClipboardManager
|
|
private var onHistoryChangeListener: OnHistoryChangeListener? = null
|
|
|
|
fun onCreate() {
|
|
clipboardManager = latinIME.getSystemService(Context.CLIPBOARD_SERVICE) as ClipboardManager
|
|
fetchPrimaryClip()
|
|
clipboardManager.addPrimaryClipChangedListener(this)
|
|
loadPinnedClips()
|
|
}
|
|
|
|
fun onPinnedClipsAvailable(pinnedClips: List<ClipboardHistoryEntry>) {
|
|
historyEntries.addAll(pinnedClips)
|
|
sortHistoryEntries()
|
|
if (onHistoryChangeListener != null) {
|
|
pinnedClips.forEach {
|
|
onHistoryChangeListener?.onClipboardHistoryEntryAdded(historyEntries.indexOf(it))
|
|
}
|
|
}
|
|
}
|
|
|
|
fun onDestroy() {
|
|
clipboardManager.removePrimaryClipChangedListener(this)
|
|
}
|
|
|
|
override fun onPrimaryClipChanged() {
|
|
// Make sure we read clipboard content only if history settings is set
|
|
if (latinIME.mSettings.current?.mClipboardHistoryEnabled == true) {
|
|
fetchPrimaryClip()
|
|
}
|
|
}
|
|
|
|
private fun fetchPrimaryClip() {
|
|
val clipData = clipboardManager.primaryClip ?: return
|
|
if (clipData.itemCount == 0) return
|
|
clipData.getItemAt(0)?.let { clipItem ->
|
|
// Starting from API 30, onPrimaryClipChanged() can be called multiple times
|
|
// for the same clip. We can identify clips with their timestamps since API 26.
|
|
// We use that to prevent unwanted duplicates.
|
|
val timeStamp = ClipboardManagerCompat.getClipTimestamp(clipData)?.also { stamp ->
|
|
if (historyEntries.any { it.timeStamp == stamp }) return
|
|
} ?: System.currentTimeMillis()
|
|
|
|
val content = clipItem.coerceToText(latinIME)
|
|
if (TextUtils.isEmpty(content)) return
|
|
|
|
val entry = ClipboardHistoryEntry(timeStamp, content)
|
|
historyEntries.add(entry)
|
|
sortHistoryEntries()
|
|
val at = historyEntries.indexOf(entry)
|
|
onHistoryChangeListener?.onClipboardHistoryEntryAdded(at)
|
|
}
|
|
}
|
|
|
|
fun toggleClipPinned(ts: Long) {
|
|
val from = historyEntries.indexOfFirst { it.timeStamp == ts }
|
|
val historyEntry = historyEntries[from].apply {
|
|
timeStamp = System.currentTimeMillis()
|
|
isPinned = !isPinned
|
|
}
|
|
sortHistoryEntries()
|
|
val to = historyEntries.indexOf(historyEntry)
|
|
onHistoryChangeListener?.onClipboardHistoryEntryMoved(from, to)
|
|
savePinnedClips()
|
|
}
|
|
|
|
fun clearHistory() {
|
|
ClipboardManagerCompat.clearPrimaryClip(clipboardManager)
|
|
val pos = historyEntries.indexOfFirst { !it.isPinned }
|
|
val count = historyEntries.count { !it.isPinned }
|
|
historyEntries.removeAll { !it.isPinned }
|
|
if (onHistoryChangeListener != null) {
|
|
onHistoryChangeListener?.onClipboardHistoryEntriesRemoved(pos, count)
|
|
}
|
|
}
|
|
|
|
fun canRemove(index: Int) = historyEntries.getOrNull(index)?.isPinned != true
|
|
|
|
fun removeEntry(index: Int) {
|
|
if (canRemove(index))
|
|
historyEntries.removeAt(index)
|
|
}
|
|
|
|
private fun sortHistoryEntries() {
|
|
historyEntries.sort()
|
|
}
|
|
|
|
private fun checkClipRetentionElapsed() {
|
|
val mins = latinIME.mSettings.current.mClipboardHistoryRetentionTime
|
|
if (mins <= 0) return // No retention limit
|
|
val maxClipRetentionTime = mins * 60 * 1000L
|
|
val now = System.currentTimeMillis()
|
|
historyEntries.removeAll { !it.isPinned && (now - it.timeStamp) > maxClipRetentionTime }
|
|
}
|
|
|
|
// We do not want to update history while user is visualizing it, so we check retention only
|
|
// when history is about to be shown
|
|
fun prepareClipboardHistory() = checkClipRetentionElapsed()
|
|
|
|
fun getHistorySize() = historyEntries.size
|
|
|
|
fun getHistoryEntry(position: Int) = historyEntries[position]
|
|
|
|
fun getHistoryEntryContent(timeStamp: Long) = historyEntries.first { it.timeStamp == timeStamp }
|
|
|
|
fun setHistoryChangeListener(l: OnHistoryChangeListener?) {
|
|
onHistoryChangeListener = l
|
|
}
|
|
|
|
fun retrieveClipboardContent(): CharSequence {
|
|
val clipData = clipboardManager.primaryClip ?: return ""
|
|
if (clipData.itemCount == 0) return ""
|
|
return clipData.getItemAt(0)?.coerceToText(latinIME) ?: ""
|
|
}
|
|
|
|
// pinned clips are stored in default shared preferences, not in device protected preferences!
|
|
private fun loadPinnedClips() {
|
|
val pinnedClipString = Settings.readPinnedClipString(latinIME)
|
|
if (pinnedClipString.isEmpty()) return
|
|
val pinnedClips: List<ClipboardHistoryEntry> = Json.decodeFromString(pinnedClipString)
|
|
latinIME.mHandler.postUpdateClipboardPinnedClips(pinnedClips)
|
|
}
|
|
|
|
private fun savePinnedClips() {
|
|
val pinnedClips = Json.encodeToString(historyEntries.filter { it.isPinned })
|
|
Settings.writePinnedClipString(latinIME, pinnedClips)
|
|
}
|
|
|
|
interface OnHistoryChangeListener {
|
|
fun onClipboardHistoryEntryAdded(at: Int)
|
|
fun onClipboardHistoryEntriesRemoved(pos: Int, count: Int)
|
|
fun onClipboardHistoryEntryMoved(from: Int, to: Int)
|
|
}
|
|
|
|
companion object {
|
|
// store pinned clips in companion object so they survive a keyboard switch (which destroys the current instance)
|
|
private val historyEntries: MutableList<ClipboardHistoryEntry> = ArrayList()
|
|
}
|
|
}
|