2020-01-23 05:46:41 +00:00
|
|
|
package com.geeksville.mesh
|
2020-01-20 23:53:22 +00:00
|
|
|
|
2020-01-21 21:12:01 +00:00
|
|
|
import android.Manifest
|
2020-01-24 20:49:27 +00:00
|
|
|
import android.bluetooth.BluetoothAdapter
|
|
|
|
import android.bluetooth.BluetoothManager
|
2020-02-09 13:52:17 +00:00
|
|
|
import android.content.*
|
2020-01-21 21:12:01 +00:00
|
|
|
import android.content.pm.PackageManager
|
2020-01-20 23:53:22 +00:00
|
|
|
import android.os.Bundle
|
2020-01-23 16:09:50 +00:00
|
|
|
import android.os.IBinder
|
2020-01-20 23:53:22 +00:00
|
|
|
import android.view.Menu
|
|
|
|
import android.view.MenuItem
|
2020-01-22 21:02:24 +00:00
|
|
|
import android.widget.Toast
|
2020-01-24 20:49:27 +00:00
|
|
|
import androidx.appcompat.app.AppCompatActivity
|
2020-01-21 21:12:01 +00:00
|
|
|
import androidx.core.app.ActivityCompat
|
|
|
|
import androidx.core.content.ContextCompat
|
2020-01-22 22:48:06 +00:00
|
|
|
import androidx.ui.core.setContent
|
2020-01-22 22:27:22 +00:00
|
|
|
import com.geeksville.android.Logging
|
2020-02-10 23:31:56 +00:00
|
|
|
import com.geeksville.mesh.service.*
|
2020-02-10 18:32:12 +00:00
|
|
|
import com.geeksville.mesh.ui.MeshApp
|
|
|
|
import com.geeksville.mesh.ui.TextMessage
|
|
|
|
import com.geeksville.mesh.ui.UIState
|
2020-02-05 05:23:52 +00:00
|
|
|
import com.geeksville.util.exceptionReporter
|
2020-02-14 15:47:20 +00:00
|
|
|
import com.google.android.gms.auth.api.signin.GoogleSignIn
|
|
|
|
import com.google.android.gms.auth.api.signin.GoogleSignInAccount
|
|
|
|
import com.google.android.gms.auth.api.signin.GoogleSignInOptions
|
|
|
|
import com.google.android.gms.common.api.ApiException
|
|
|
|
import com.google.android.gms.tasks.Task
|
2020-02-09 13:52:17 +00:00
|
|
|
import java.nio.charset.Charset
|
|
|
|
import java.util.*
|
2020-01-20 23:53:22 +00:00
|
|
|
|
2020-01-21 17:37:39 +00:00
|
|
|
|
2020-02-11 04:17:42 +00:00
|
|
|
class MainActivity : AppCompatActivity(), Logging,
|
|
|
|
ActivityCompat.OnRequestPermissionsResultCallback {
|
2020-01-20 23:53:22 +00:00
|
|
|
|
2020-01-21 17:37:39 +00:00
|
|
|
companion object {
|
|
|
|
const val REQUEST_ENABLE_BT = 10
|
2020-01-21 21:12:01 +00:00
|
|
|
const val DID_REQUEST_PERM = 11
|
2020-02-14 15:47:20 +00:00
|
|
|
const val RC_SIGN_IN = 12 // google signin completed
|
2020-01-21 17:37:39 +00:00
|
|
|
}
|
|
|
|
|
2020-02-09 15:28:24 +00:00
|
|
|
|
|
|
|
private val utf8 = Charset.forName("UTF-8")
|
2020-01-24 20:49:27 +00:00
|
|
|
|
2020-01-22 21:02:24 +00:00
|
|
|
private val bluetoothAdapter: BluetoothAdapter? by lazy(LazyThreadSafetyMode.NONE) {
|
2020-01-21 17:37:39 +00:00
|
|
|
val bluetoothManager = getSystemService(Context.BLUETOOTH_SERVICE) as BluetoothManager
|
2020-01-22 21:02:24 +00:00
|
|
|
bluetoothManager.adapter
|
2020-01-21 17:37:39 +00:00
|
|
|
}
|
|
|
|
|
2020-02-11 04:17:42 +00:00
|
|
|
private fun requestPermission() {
|
2020-01-22 22:27:22 +00:00
|
|
|
debug("Checking permissions")
|
|
|
|
|
2020-01-23 00:45:27 +00:00
|
|
|
val perms = arrayOf(
|
|
|
|
Manifest.permission.ACCESS_FINE_LOCATION,
|
2020-01-21 21:12:01 +00:00
|
|
|
Manifest.permission.ACCESS_BACKGROUND_LOCATION,
|
|
|
|
Manifest.permission.BLUETOOTH,
|
|
|
|
Manifest.permission.BLUETOOTH_ADMIN,
|
2020-01-25 01:47:32 +00:00
|
|
|
Manifest.permission.WAKE_LOCK,
|
2020-02-14 12:41:20 +00:00
|
|
|
Manifest.permission.WRITE_EXTERNAL_STORAGE
|
2020-01-23 00:45:27 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
val missingPerms = perms.filter {
|
|
|
|
ContextCompat.checkSelfPermission(
|
|
|
|
this,
|
|
|
|
it
|
|
|
|
) != PackageManager.PERMISSION_GRANTED
|
|
|
|
}
|
2020-01-21 21:12:01 +00:00
|
|
|
if (missingPerms.isNotEmpty()) {
|
|
|
|
missingPerms.forEach {
|
|
|
|
// Permission is not granted
|
|
|
|
// Should we show an explanation?
|
|
|
|
if (ActivityCompat.shouldShowRequestPermissionRationale(this, it)) {
|
|
|
|
// FIXME
|
|
|
|
// Show an explanation to the user *asynchronously* -- don't block
|
|
|
|
// this thread waiting for the user's response! After the user
|
|
|
|
// sees the explanation, try again to request the permission.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ask for all the missing perms
|
|
|
|
ActivityCompat.requestPermissions(this, missingPerms.toTypedArray(), DID_REQUEST_PERM)
|
|
|
|
|
|
|
|
// DID_REQUEST_PERM is an
|
|
|
|
// app-defined int constant. The callback method gets the
|
|
|
|
// result of the request.
|
|
|
|
} else {
|
|
|
|
// Permission has already been granted
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-11 04:17:42 +00:00
|
|
|
override fun onRequestPermissionsResult(
|
|
|
|
requestCode: Int,
|
|
|
|
permissions: Array<out String>,
|
|
|
|
grantResults: IntArray
|
|
|
|
) {
|
|
|
|
super.onRequestPermissionsResult(requestCode, permissions, grantResults)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
private fun setOwner() {
|
2020-02-14 15:47:20 +00:00
|
|
|
// Note: we are careful to not set a new unique ID
|
|
|
|
meshService!!.setOwner(null, "Kevin Xter", "kx")
|
|
|
|
}
|
|
|
|
|
|
|
|
private fun sendTestPackets() {
|
|
|
|
exceptionReporter {
|
|
|
|
val m = meshService!!
|
|
|
|
|
|
|
|
// Do some test operations
|
|
|
|
val testPayload = "hello world".toByteArray()
|
|
|
|
m.sendData(
|
|
|
|
"+16508675310",
|
|
|
|
testPayload,
|
|
|
|
MeshProtos.Data.Type.SIGNAL_OPAQUE_VALUE
|
|
|
|
)
|
|
|
|
m.sendData(
|
|
|
|
"+16508675310",
|
|
|
|
testPayload,
|
|
|
|
MeshProtos.Data.Type.CLEAR_TEXT_VALUE
|
|
|
|
)
|
|
|
|
}
|
|
|
|
}
|
2020-02-11 04:17:42 +00:00
|
|
|
|
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onCreate(savedInstanceState: Bundle?) {
|
|
|
|
super.onCreate(savedInstanceState)
|
|
|
|
|
|
|
|
setContent {
|
|
|
|
MeshApp()
|
2020-02-11 04:17:42 +00:00
|
|
|
}
|
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// Ensures Bluetooth is available on the device and it is enabled. If not,
|
|
|
|
// displays a dialog requesting user permission to enable Bluetooth.
|
|
|
|
if (bluetoothAdapter != null) {
|
|
|
|
bluetoothAdapter!!.takeIf { !it.isEnabled }?.apply {
|
|
|
|
val enableBtIntent = Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE)
|
|
|
|
startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT)
|
2020-02-14 12:41:20 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
} else {
|
|
|
|
Toast.makeText(this, "Error - this app requires bluetooth", Toast.LENGTH_LONG)
|
|
|
|
.show()
|
2020-02-05 05:23:52 +00:00
|
|
|
}
|
2020-01-28 03:23:34 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
requestPermission()
|
2020-02-10 15:40:45 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// Configure sign-in to request the user's ID, email address, and basic
|
|
|
|
// profile. ID and basic profile are included in DEFAULT_SIGN_IN.
|
|
|
|
// Configure sign-in to request the user's ID, email address, and basic
|
|
|
|
// profile. ID and basic profile are included in DEFAULT_SIGN_IN.
|
|
|
|
val gso =
|
|
|
|
GoogleSignInOptions.Builder(GoogleSignInOptions.DEFAULT_SIGN_IN)
|
|
|
|
.requestEmail()
|
|
|
|
.build()
|
2020-01-25 18:00:57 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// Build a GoogleSignInClient with the options specified by gso.
|
|
|
|
UIState.googleSignInClient = GoogleSignIn.getClient(this, gso);
|
|
|
|
}
|
2020-01-21 17:37:39 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onDestroy() {
|
|
|
|
unregisterMeshReceiver()
|
|
|
|
super.onDestroy()
|
|
|
|
}
|
2020-02-14 12:41:20 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
/**
|
|
|
|
* Dispatch incoming result to the correct fragment.
|
|
|
|
*/
|
|
|
|
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
|
|
|
|
super.onActivityResult(requestCode, resultCode, data)
|
|
|
|
|
|
|
|
// Result returned from launching the Intent from GoogleSignInClient.getSignInIntent(...);
|
|
|
|
// Result returned from launching the Intent from GoogleSignInClient.getSignInIntent(...);
|
|
|
|
if (requestCode === RC_SIGN_IN) {
|
|
|
|
// The Task returned from this call is always completed, no need to attach
|
|
|
|
// a listener.
|
|
|
|
val task: Task<GoogleSignInAccount> =
|
|
|
|
GoogleSignIn.getSignedInAccountFromIntent(data)
|
|
|
|
handleSignInResult(task)
|
2020-01-20 23:53:22 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private fun handleSignInResult(completedTask: Task<GoogleSignInAccount>) {
|
|
|
|
try {
|
|
|
|
val account =
|
|
|
|
completedTask.getResult(ApiException::class.java)
|
|
|
|
// Signed in successfully, show authenticated UI.
|
|
|
|
//updateUI(account)
|
|
|
|
} catch (e: ApiException) { // The ApiException status code indicates the detailed failure reason.
|
|
|
|
// Please refer to the GoogleSignInStatusCodes class reference for more information.
|
|
|
|
warn("signInResult:failed code=" + e.statusCode)
|
|
|
|
//updateUI(null)
|
2020-02-14 12:41:20 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-11 04:17:42 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private var receiverRegistered = false
|
2020-02-11 04:17:42 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private fun registerMeshReceiver() {
|
|
|
|
logAssert(!receiverRegistered)
|
|
|
|
val filter = IntentFilter()
|
|
|
|
filter.addAction(MeshService.ACTION_MESH_CONNECTED)
|
|
|
|
filter.addAction(MeshService.ACTION_NODE_CHANGE)
|
|
|
|
filter.addAction(MeshService.ACTION_RECEIVED_DATA)
|
|
|
|
registerReceiver(meshServiceReceiver, filter)
|
2020-02-11 04:17:42 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private fun unregisterMeshReceiver() {
|
|
|
|
if (receiverRegistered) {
|
|
|
|
receiverRegistered = false
|
|
|
|
unregisterReceiver(meshServiceReceiver)
|
2020-02-11 04:17:42 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
/// Called when we gain/lose a connection to our mesh radio
|
|
|
|
private fun onMeshConnectionChanged(connected: Boolean) {
|
|
|
|
UIState.isConnected.value = connected
|
|
|
|
debug("connchange ${UIState.isConnected.value}")
|
|
|
|
if (connected) {
|
|
|
|
// everytime the radio reconnects, we slam in our current owner data
|
|
|
|
setOwner()
|
2020-02-11 04:17:42 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private val meshServiceReceiver = object : BroadcastReceiver() {
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onReceive(context: Context, intent: Intent) = exceptionReporter {
|
|
|
|
debug("Received from mesh service $intent")
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
when (intent.action) {
|
|
|
|
MeshService.ACTION_NODE_CHANGE -> {
|
|
|
|
val info: NodeInfo = intent.getParcelableExtra(EXTRA_NODEINFO)!!
|
|
|
|
debug("UI nodechange $info")
|
2020-02-09 13:52:17 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// We only care about nodes that have user info
|
|
|
|
info.user?.id?.let {
|
|
|
|
val newnodes = UIState.nodes.value.toMutableMap()
|
|
|
|
newnodes[it] = info
|
|
|
|
UIState.nodes.value = newnodes
|
2020-02-09 13:52:17 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-02-09 15:28:24 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
MeshService.ACTION_RECEIVED_DATA -> {
|
|
|
|
debug("TODO rxdata")
|
|
|
|
val sender = intent.getStringExtra(EXTRA_SENDER)!!
|
|
|
|
val payload = intent.getByteArrayExtra(EXTRA_PAYLOAD)!!
|
|
|
|
val typ = intent.getIntExtra(EXTRA_TYP, -1)
|
|
|
|
|
|
|
|
when (typ) {
|
|
|
|
MeshProtos.Data.Type.CLEAR_TEXT_VALUE -> {
|
|
|
|
// FIXME - use the real time from the packet
|
|
|
|
val modded = UIState.messages.value.toMutableList()
|
|
|
|
modded.add(TextMessage(Date(), sender, payload.toString(utf8)))
|
|
|
|
UIState.messages.value = modded
|
2020-02-09 13:52:17 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
else -> TODO()
|
2020-02-09 13:52:17 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
MeshService.ACTION_MESH_CONNECTED -> {
|
|
|
|
val connected = intent.getBooleanExtra(EXTRA_CONNECTED, false)
|
|
|
|
onMeshConnectionChanged(connected)
|
|
|
|
}
|
|
|
|
else -> TODO()
|
2020-02-09 13:52:17 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-01-20 23:53:22 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private var meshService: IMeshService? = null
|
|
|
|
private var isBound = false
|
2020-01-28 00:00:00 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private var serviceConnection = object : ServiceConnection {
|
|
|
|
override fun onServiceConnected(name: ComponentName, service: IBinder) =
|
|
|
|
exceptionReporter {
|
|
|
|
val m = IMeshService.Stub.asInterface(service)
|
|
|
|
meshService = m
|
2020-02-11 04:17:42 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// We don't start listening for packets until after we are connected to the service
|
|
|
|
registerMeshReceiver()
|
2020-01-28 03:23:34 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// We won't receive a notify for the initial state of connection, so we force an update here
|
|
|
|
onMeshConnectionChanged(m.isConnected)
|
2020-02-10 23:39:04 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
debug("connected to mesh service, isConnected=${UIState.isConnected.value}")
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
// make some placeholder nodeinfos
|
|
|
|
UIState.nodes.value =
|
|
|
|
m.nodes.toList().map {
|
|
|
|
it.user?.id!! to it
|
|
|
|
}.toMap()
|
2020-02-14 12:41:20 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
|
|
|
|
override fun onServiceDisconnected(name: ComponentName) {
|
|
|
|
warn("The mesh service has disconnected")
|
|
|
|
unregisterMeshReceiver()
|
|
|
|
meshService = null
|
2020-01-23 16:09:50 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private fun bindMeshService() {
|
|
|
|
debug("Binding to mesh service!")
|
|
|
|
// we bind using the well known name, to make sure 3rd party apps could also
|
|
|
|
logAssert(meshService == null)
|
2020-01-23 17:04:06 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
val intent = MeshService.startService(this)
|
|
|
|
if (intent != null) {
|
|
|
|
// ALSO bind so we can use the api
|
|
|
|
logAssert(bindService(intent, serviceConnection, Context.BIND_AUTO_CREATE))
|
|
|
|
isBound = true;
|
2020-02-04 21:24:04 +00:00
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
private fun unbindMeshService() {
|
|
|
|
// If we have received the service, and hence registered with
|
|
|
|
// it, then now is the time to unregister.
|
|
|
|
// if we never connected, do nothing
|
|
|
|
debug("Unbinding from mesh service!")
|
|
|
|
if (isBound)
|
|
|
|
unbindService(serviceConnection)
|
|
|
|
meshService = null
|
|
|
|
}
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onPause() {
|
|
|
|
unregisterMeshReceiver() // No point in receiving updates while the GUI is gone, we'll get them when the user launches the activity
|
|
|
|
unbindMeshService()
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
super.onPause()
|
|
|
|
}
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onResume() {
|
|
|
|
super.onResume()
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
bindMeshService()
|
|
|
|
}
|
2020-01-23 16:09:50 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onCreateOptionsMenu(menu: Menu): Boolean {
|
|
|
|
// Inflate the menu; this adds items to the action bar if it is present.
|
|
|
|
menuInflater.inflate(R.menu.menu_main, menu)
|
|
|
|
return true
|
|
|
|
}
|
2020-01-20 23:53:22 +00:00
|
|
|
|
2020-02-14 15:47:20 +00:00
|
|
|
override fun onOptionsItemSelected(item: MenuItem): Boolean {
|
|
|
|
// Handle action bar item clicks here. The action bar will
|
|
|
|
// automatically handle clicks on the Home/Up button, so long
|
|
|
|
// as you specify a parent activity in AndroidManifest.xml.
|
|
|
|
return when (item.itemId) {
|
|
|
|
R.id.action_settings -> true
|
|
|
|
else -> super.onOptionsItemSelected(item)
|
2020-01-20 23:53:22 +00:00
|
|
|
}
|
|
|
|
}
|
2020-02-14 15:47:20 +00:00
|
|
|
}
|
2020-01-21 17:37:39 +00:00
|
|
|
|
2020-02-09 18:18:26 +00:00
|
|
|
|