mirror of
https://github.com/oxen-io/session-android.git
synced 2025-03-26 21:00:51 +00:00
245 lines
11 KiB
Kotlin
245 lines
11 KiB
Kotlin
package org.thoughtcrime.securesms.loki.activities
|
|
|
|
import android.content.BroadcastReceiver
|
|
import android.content.Context
|
|
import android.content.Intent
|
|
import android.content.IntentFilter
|
|
import android.net.Uri
|
|
import android.os.Bundle
|
|
import android.os.Handler
|
|
import android.support.v4.content.LocalBroadcastManager
|
|
import android.util.AttributeSet
|
|
import android.util.TypedValue
|
|
import android.view.Gravity
|
|
import android.view.View
|
|
import android.widget.LinearLayout
|
|
import android.widget.RelativeLayout
|
|
import android.widget.TextView
|
|
import android.widget.Toast
|
|
import kotlinx.android.synthetic.main.activity_path.*
|
|
import network.loki.messenger.R
|
|
import org.thoughtcrime.securesms.PassphraseRequiredActionBarActivity
|
|
import org.thoughtcrime.securesms.loki.utilities.*
|
|
import org.whispersystems.signalservice.loki.api.Snode
|
|
import org.whispersystems.signalservice.loki.api.onionrequests.OnionRequestAPI
|
|
|
|
class PathActivity : PassphraseRequiredActionBarActivity() {
|
|
private val broadcastReceivers = mutableListOf<BroadcastReceiver>()
|
|
|
|
// region Lifecycle
|
|
override fun onCreate(savedInstanceState: Bundle?, isReady: Boolean) {
|
|
super.onCreate(savedInstanceState, isReady)
|
|
setContentView(R.layout.activity_path)
|
|
supportActionBar!!.title = resources.getString(R.string.activity_path_title)
|
|
learnMoreButton.setOnClickListener { learnMore() }
|
|
update(false)
|
|
registerObservers()
|
|
}
|
|
|
|
private fun registerObservers() {
|
|
val buildingPathsReceiver: BroadcastReceiver = object : BroadcastReceiver() {
|
|
|
|
override fun onReceive(context: Context, intent: Intent) {
|
|
handleBuildingPathsEvent()
|
|
}
|
|
}
|
|
broadcastReceivers.add(buildingPathsReceiver)
|
|
LocalBroadcastManager.getInstance(this).registerReceiver(buildingPathsReceiver, IntentFilter("buildingPaths"))
|
|
val pathsBuiltReceiver: BroadcastReceiver = object : BroadcastReceiver() {
|
|
|
|
override fun onReceive(context: Context, intent: Intent) {
|
|
handlePathsBuiltEvent()
|
|
}
|
|
}
|
|
broadcastReceivers.add(pathsBuiltReceiver)
|
|
LocalBroadcastManager.getInstance(this).registerReceiver(pathsBuiltReceiver, IntentFilter("pathsBuilt"))
|
|
val onionRequestPathCountriesLoadedReceiver: BroadcastReceiver = object : BroadcastReceiver() {
|
|
|
|
override fun onReceive(context: Context, intent: Intent) {
|
|
handleOnionRequestPathCountriesLoaded()
|
|
}
|
|
}
|
|
broadcastReceivers.add(onionRequestPathCountriesLoadedReceiver)
|
|
LocalBroadcastManager.getInstance(this).registerReceiver(onionRequestPathCountriesLoadedReceiver, IntentFilter("onionRequestPathCountriesLoaded"))
|
|
}
|
|
|
|
override fun onDestroy() {
|
|
for (receiver in broadcastReceivers) {
|
|
LocalBroadcastManager.getInstance(this).unregisterReceiver(receiver)
|
|
}
|
|
super.onDestroy()
|
|
}
|
|
// endregion
|
|
|
|
// region Updating
|
|
private fun handleBuildingPathsEvent() { update(false) }
|
|
private fun handlePathsBuiltEvent() { update(false) }
|
|
private fun handleOnionRequestPathCountriesLoaded() { update(false) }
|
|
|
|
private fun update(isAnimated: Boolean) {
|
|
pathRowsContainer.removeAllViews()
|
|
if (OnionRequestAPI.paths.count() >= OnionRequestAPI.pathCount) {
|
|
val path = OnionRequestAPI.paths.firstOrNull() ?: return finish()
|
|
val dotAnimationRepeatInterval = path.count().toLong() * 1000 + 1000
|
|
val pathRows = path.mapIndexed { index, snode ->
|
|
val isGuardSnode = (OnionRequestAPI.guardSnodes.contains(snode))
|
|
getPathRow(snode, LineView.Location.Middle, index.toLong() * 1000 + 2000, dotAnimationRepeatInterval, isGuardSnode)
|
|
}
|
|
val youRow = getPathRow("You", null, LineView.Location.Top, 1000, dotAnimationRepeatInterval)
|
|
val destinationRow = getPathRow("Destination", null, LineView.Location.Bottom, path.count().toLong() * 1000 + 2000, dotAnimationRepeatInterval)
|
|
val rows = listOf( youRow ) + pathRows + listOf( destinationRow )
|
|
for (row in rows) {
|
|
pathRowsContainer.addView(row)
|
|
}
|
|
if (isAnimated) {
|
|
spinner.fadeOut()
|
|
} else {
|
|
spinner.alpha = 0.0f
|
|
}
|
|
} else {
|
|
if (isAnimated) {
|
|
spinner.fadeIn()
|
|
} else {
|
|
spinner.alpha = 1.0f
|
|
}
|
|
}
|
|
}
|
|
// endregion
|
|
|
|
// region General
|
|
private fun getPathRow(title: String, subtitle: String?, location: LineView.Location, dotAnimationStartDelay: Long, dotAnimationRepeatInterval: Long): LinearLayout {
|
|
val mainContainer = LinearLayout(this)
|
|
mainContainer.orientation = LinearLayout.HORIZONTAL
|
|
mainContainer.gravity = Gravity.CENTER_VERTICAL
|
|
val mainContainerLayoutParams = LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT)
|
|
mainContainer.layoutParams = mainContainerLayoutParams
|
|
val lineView = LineView(this, location, dotAnimationStartDelay, dotAnimationRepeatInterval)
|
|
val lineViewLayoutParams = LinearLayout.LayoutParams(resources.getDimensionPixelSize(R.dimen.path_row_expanded_dot_size), resources.getDimensionPixelSize(R.dimen.path_row_height))
|
|
lineView.layoutParams = lineViewLayoutParams
|
|
mainContainer.addView(lineView)
|
|
val titleTextView = TextView(this)
|
|
titleTextView.setTextColor(resources.getColorWithID(R.color.text, theme))
|
|
titleTextView.setTextSize(TypedValue.COMPLEX_UNIT_PX, resources.getDimension(R.dimen.medium_font_size))
|
|
titleTextView.text = title
|
|
titleTextView.textAlignment = TextView.TEXT_ALIGNMENT_VIEW_START
|
|
val titleContainer = LinearLayout(this)
|
|
titleContainer.orientation = LinearLayout.VERTICAL
|
|
titleContainer.addView(titleTextView)
|
|
val titleContainerLayoutParams = LinearLayout.LayoutParams(LinearLayout.LayoutParams.WRAP_CONTENT, LinearLayout.LayoutParams.WRAP_CONTENT)
|
|
titleContainerLayoutParams.marginStart = resources.getDimensionPixelSize(R.dimen.large_spacing)
|
|
titleContainer.layoutParams = titleContainerLayoutParams
|
|
mainContainer.addView(titleContainer)
|
|
if (subtitle != null) {
|
|
val subtitleTextView = TextView(this)
|
|
subtitleTextView.setTextColor(resources.getColorWithID(R.color.text, theme))
|
|
subtitleTextView.setTextSize(TypedValue.COMPLEX_UNIT_PX, resources.getDimension(R.dimen.small_font_size))
|
|
subtitleTextView.text = subtitle
|
|
subtitleTextView.textAlignment = TextView.TEXT_ALIGNMENT_VIEW_START
|
|
titleContainer.addView(subtitleTextView)
|
|
}
|
|
return mainContainer
|
|
}
|
|
|
|
private fun getPathRow(snode: Snode, location: LineView.Location, dotAnimationStartDelay: Long, dotAnimationRepeatInterval: Long, isGuardSnode: Boolean): LinearLayout {
|
|
val title = if (isGuardSnode) resources.getString(R.string.activity_path_guard_node_row_title) else resources.getString(R.string.activity_path_service_node_row_title)
|
|
val subtitle = IP2Country.shared.countryNamesCache[snode.ip] ?: "Resolving..."
|
|
return getPathRow(title, subtitle, location, dotAnimationStartDelay, dotAnimationRepeatInterval)
|
|
}
|
|
// endregion
|
|
|
|
// region Interaction
|
|
private fun learnMore() {
|
|
try {
|
|
val url = "https://getsession.org/faq/#onion-routing"
|
|
val intent = Intent(Intent.ACTION_VIEW, Uri.parse(url))
|
|
startActivity(intent)
|
|
} catch (e: Exception) {
|
|
Toast.makeText(this, R.string.invalid_url, Toast.LENGTH_SHORT).show()
|
|
}
|
|
}
|
|
// endregion
|
|
|
|
// region Line View
|
|
private class LineView : RelativeLayout {
|
|
private lateinit var location: Location
|
|
private var dotAnimationStartDelay: Long = 0
|
|
private var dotAnimationRepeatInterval: Long = 0
|
|
|
|
private val dotView by lazy {
|
|
val result = View(context)
|
|
result.setBackgroundResource(R.drawable.accent_dot)
|
|
result
|
|
}
|
|
|
|
enum class Location {
|
|
Top, Middle, Bottom
|
|
}
|
|
|
|
constructor(context: Context, location: Location, dotAnimationStartDelay: Long, dotAnimationRepeatInterval: Long) : super(context) {
|
|
this.location = location
|
|
this.dotAnimationStartDelay = dotAnimationStartDelay
|
|
this.dotAnimationRepeatInterval = dotAnimationRepeatInterval
|
|
setUpViewHierarchy()
|
|
}
|
|
|
|
constructor(context: Context) : super(context) {
|
|
throw Exception("Use LineView(context:location:dotAnimationStartDelay:dotAnimationRepeatInterval:) instead.")
|
|
}
|
|
|
|
constructor(context: Context, attrs: AttributeSet) : super(context, attrs) {
|
|
throw Exception("Use LineView(context:location:dotAnimationStartDelay:dotAnimationRepeatInterval:) instead.")
|
|
}
|
|
|
|
constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int) : super(context, attrs, defStyleAttr) {
|
|
throw Exception("Use LineView(context:location:dotAnimationStartDelay:dotAnimationRepeatInterval:) instead.")
|
|
}
|
|
|
|
constructor(context: Context, attrs: AttributeSet, defStyleAttr: Int, defStyleRes: Int) : super(context, attrs, defStyleAttr, defStyleRes) {
|
|
throw Exception("Use LineView(context:location:dotAnimationStartDelay:dotAnimationRepeatInterval:) instead.")
|
|
}
|
|
|
|
private fun setUpViewHierarchy() {
|
|
val lineView = View(context)
|
|
lineView.setBackgroundColor(resources.getColorWithID(R.color.text, context.theme))
|
|
val lineViewHeight = when (location) {
|
|
Location.Top, Location.Bottom -> resources.getDimensionPixelSize(R.dimen.path_row_height) / 2
|
|
Location.Middle -> resources.getDimensionPixelSize(R.dimen.path_row_height)
|
|
}
|
|
val lineViewLayoutParams = LayoutParams(1, lineViewHeight)
|
|
when (location) {
|
|
Location.Top -> lineViewLayoutParams.addRule(ALIGN_PARENT_BOTTOM)
|
|
Location.Middle, Location.Bottom -> lineViewLayoutParams.addRule(ALIGN_PARENT_TOP)
|
|
}
|
|
lineViewLayoutParams.addRule(CENTER_HORIZONTAL)
|
|
lineView.layoutParams = lineViewLayoutParams
|
|
addView(lineView)
|
|
val dotViewSize = resources.getDimensionPixelSize(R.dimen.path_row_dot_size)
|
|
val dotViewLayoutParams = LayoutParams(dotViewSize, dotViewSize)
|
|
dotViewLayoutParams.addRule(CENTER_IN_PARENT)
|
|
dotView.layoutParams = dotViewLayoutParams
|
|
addView(dotView)
|
|
Handler().postDelayed({
|
|
performAnimation()
|
|
}, dotAnimationStartDelay)
|
|
}
|
|
|
|
private fun performAnimation() {
|
|
expand()
|
|
Handler().postDelayed({
|
|
collapse()
|
|
Handler().postDelayed({
|
|
performAnimation()
|
|
}, dotAnimationRepeatInterval)
|
|
}, 1000)
|
|
}
|
|
|
|
private fun expand() {
|
|
dotView.animateSizeChange(R.dimen.path_row_dot_size, R.dimen.path_row_expanded_dot_size)
|
|
}
|
|
|
|
private fun collapse() {
|
|
dotView.animateSizeChange(R.dimen.path_row_expanded_dot_size, R.dimen.path_row_dot_size)
|
|
}
|
|
}
|
|
// endregion
|
|
} |