package com.geeksville.mesh import android.Manifest import android.annotation.SuppressLint import android.app.Activity import android.bluetooth.BluetoothAdapter import android.bluetooth.BluetoothDevice import android.bluetooth.BluetoothManager import android.companion.CompanionDeviceManager import android.content.BroadcastReceiver import android.content.Context import android.content.Intent import android.content.IntentFilter import android.content.pm.PackageInfo import android.content.pm.PackageManager import android.hardware.usb.UsbDevice import android.hardware.usb.UsbManager import android.net.Uri import android.os.Build import android.os.Bundle import android.os.RemoteException import android.view.Menu import android.view.MenuItem import android.view.MotionEvent import android.view.View import android.widget.Toast import androidx.activity.viewModels import androidx.appcompat.app.AppCompatActivity import androidx.appcompat.widget.Toolbar import androidx.core.app.ActivityCompat import androidx.core.content.ContextCompat import androidx.fragment.app.Fragment import androidx.lifecycle.Observer import androidx.viewpager2.adapter.FragmentStateAdapter import com.geeksville.android.GeeksvilleApplication import com.geeksville.android.Logging import com.geeksville.android.ServiceClient import com.geeksville.concurrent.handledLaunch import com.geeksville.mesh.model.Channel import com.geeksville.mesh.model.UIViewModel import com.geeksville.mesh.service.* import com.geeksville.mesh.ui.* import com.geeksville.util.Exceptions import com.geeksville.util.exceptionReporter import com.google.android.gms.auth.api.signin.GoogleSignIn import com.google.android.gms.auth.api.signin.GoogleSignInAccount import com.google.android.gms.common.ConnectionResult import com.google.android.gms.common.GoogleApiAvailability import com.google.android.gms.tasks.Task import com.google.android.material.dialog.MaterialAlertDialogBuilder import com.google.android.material.tabs.TabLayoutMediator import com.google.protobuf.InvalidProtocolBufferException import com.vorlonsoft.android.rate.AppRate import com.vorlonsoft.android.rate.StoreType import kotlinx.android.synthetic.main.activity_main.* import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.Job import kotlinx.coroutines.cancel import java.nio.charset.Charset /* UI design material setup instructions: https://material.io/develop/android/docs/getting-started/ dark theme (or use system eventually) https://material.io/develop/android/theming/dark/ NavDrawer is a standard draw which can be dragged in from the left or the menu icon inside the app title. Fragments: SettingsFragment shows "Settings" username shortname bluetooth pairing list (eventually misc device settings that are not channel related) Channel fragment "Channel" qr code, copy link button ch number misc other settings (eventually a way of choosing between past channels) ChatFragment "Messages" a text box to enter new texts a scrolling list of rows. each row is a text and a sender info layout NodeListFragment "Users" a node info row for every node ViewModels: BTScanModel starts/stops bt scan and provides list of devices (manages entire scan lifecycle) MeshModel contains: (manages entire service relationship) current received texts current radio macaddr current node infos (updated dynamically) eventually use bottom navigation bar to switch between, Members, Chat, Channel, Settings. https://material.io/develop/android/components/bottom-navigation-view/ use numbers of # chat messages and # of members in the badges. (per this recommendation to not use top tabs: https://ux.stackexchange.com/questions/102439/android-ux-when-to-use-bottom-navigation-and-when-to-use-tabs ) eventually: make a custom theme: https://github.com/material-components/material-components-android/tree/master/material-theme-builder */ val utf8 = Charset.forName("UTF-8") class MainActivity : AppCompatActivity(), Logging, ActivityCompat.OnRequestPermissionsResultCallback { companion object { const val REQUEST_ENABLE_BT = 10 const val DID_REQUEST_PERM = 11 const val RC_SIGN_IN = 12 // google signin completed const val RC_SELECT_DEVICE = 13 // seems to be hardwired in CompanionDeviceManager to add 65536 } // Used to schedule a coroutine in the GUI thread private val mainScope = CoroutineScope(Dispatchers.Main + Job()) private val bluetoothAdapter: BluetoothAdapter? by lazy(LazyThreadSafetyMode.NONE) { val bluetoothManager = getSystemService(Context.BLUETOOTH_SERVICE) as BluetoothManager bluetoothManager.adapter } val model: UIViewModel by viewModels() data class TabInfo(val text: String, val icon: Int, val content: Fragment) // private val tabIndexes = generateSequence(0) { it + 1 } FIXME, instead do withIndex or zip? to get the ids below, also stop duplicating strings private val tabInfos = arrayOf( TabInfo( "Messages", R.drawable.ic_twotone_message_24, MessagesFragment() ), TabInfo( "Users", R.drawable.ic_twotone_people_24, UsersFragment() ), TabInfo( "Map", R.drawable.ic_twotone_map_24, MapFragment() ), TabInfo( "Channel", R.drawable.ic_twotone_contactless_24, ChannelFragment() ), TabInfo( "Settings", R.drawable.ic_twotone_settings_applications_24, SettingsFragment() ) ) private val tabsAdapter = object : FragmentStateAdapter(this) { override fun getItemCount(): Int = tabInfos.size override fun createFragment(position: Int): Fragment { // Return a NEW fragment instance in createFragment(int) /* fragment.arguments = Bundle().apply { // Our object is just an integer :-P putInt(ARG_OBJECT, position + 1) } */ return tabInfos[position].content } } private val btStateReceiver = BluetoothStateReceiver { _ -> updateBluetoothEnabled() } /** * Don't tell our app we have bluetooth until we have bluetooth _and_ location access */ private fun updateBluetoothEnabled() { var enabled = false // assume failure val requiredPerms = listOf( Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.BLUETOOTH, Manifest.permission.BLUETOOTH_ADMIN ) if (getMissingPermissions(requiredPerms).isEmpty()) { /// ask the adapter if we have access bluetoothAdapter?.apply { enabled = isEnabled } } else errormsg("Still missing needed bluetooth permissions") debug("Detected our bluetooth access=$enabled") model.bluetoothEnabled.value = enabled } /** * return a list of the permissions we don't have */ private fun getMissingPermissions(perms: List) = perms.filter { ContextCompat.checkSelfPermission( this, it ) != PackageManager.PERMISSION_GRANTED } private fun getMissingPermissions(): List { val perms = mutableListOf( Manifest.permission.ACCESS_COARSE_LOCATION, Manifest.permission.ACCESS_FINE_LOCATION, Manifest.permission.BLUETOOTH, Manifest.permission.BLUETOOTH_ADMIN, Manifest.permission.WAKE_LOCK // We only need this for logging to capture files for the simulator - turn off for most users // Manifest.permission.WRITE_EXTERNAL_STORAGE ) if (Build.VERSION.SDK_INT >= 29) // only added later perms.add(Manifest.permission.ACCESS_BACKGROUND_LOCATION) // Some old phones complain about requesting perms they don't understand if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { perms.add(Manifest.permission.REQUEST_COMPANION_RUN_IN_BACKGROUND) perms.add(Manifest.permission.REQUEST_COMPANION_USE_DATA_IN_BACKGROUND) } return getMissingPermissions(perms) } private fun requestPermission() { debug("Checking permissions") val missingPerms = getMissingPermissions() 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 } } /** * Remind user he's disabled permissions we need */ private fun warnMissingPermissions() { // Older versions of android don't know about these permissions - ignore failure to grant val ignoredPermissions = setOf( Manifest.permission.ACCESS_COARSE_LOCATION, Manifest.permission.REQUEST_COMPANION_RUN_IN_BACKGROUND, Manifest.permission.REQUEST_COMPANION_USE_DATA_IN_BACKGROUND ) val deniedPermissions = getMissingPermissions().filter { name -> !ignoredPermissions.contains(name) } if (deniedPermissions.isNotEmpty()) { errormsg("Denied permissions: ${deniedPermissions.joinToString(",")}") Toast.makeText( this, getString(R.string.permission_missing), Toast.LENGTH_LONG ).show() } } override fun onRequestPermissionsResult( requestCode: Int, permissions: Array, grantResults: IntArray ) { super.onRequestPermissionsResult(requestCode, permissions, grantResults) updateBluetoothEnabled() } private fun sendTestPackets() { exceptionReporter { val m = model.meshService!! // Do some test operations val testPayload = "hello world".toByteArray() m.send( DataPacket( "+16508675310", testPayload, MeshProtos.Data.Type.OPAQUE_VALUE ) ) m.send( DataPacket( "+16508675310", testPayload, MeshProtos.Data.Type.CLEAR_TEXT_VALUE ) ) } } /// Ask user to rate in play store private fun askToRate() { exceptionReporter { // Got one IllegalArgumentException from inside this lib, but we don't want to crash our app because of bugs in this optional feature val hasGooglePlay = GoogleApiAvailability.getInstance() .isGooglePlayServicesAvailable(this) != ConnectionResult.SERVICE_MISSING val rater = AppRate.with(this) .setInstallDays(10.toByte()) // default is 10, 0 means install day, 10 means app is launched 10 or more days later than installation .setLaunchTimes(10.toByte()) // default is 10, 3 means app is launched 3 or more times .setRemindInterval(1.toByte()) // default is 1, 1 means app is launched 1 or more days after neutral button clicked .setRemindLaunchesNumber(1.toByte()) // default is 0, 1 means app is launched 1 or more times after neutral button clicked .setStoreType(if (hasGooglePlay) StoreType.GOOGLEPLAY else StoreType.AMAZON) rater.monitor() // Monitors the app launch times // Only ask to rate if the user has a suitable store AppRate.showRateDialogIfMeetsConditions(this); // Shows the Rate Dialog when conditions are met } } private val isInTestLab: Boolean by lazy { (application as GeeksvilleApplication).isInTestLab } override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) val prefs = UIViewModel.getPreferences(this) model.ownerName.value = prefs.getString("owner", "")!! /// Set initial bluetooth state updateBluetoothEnabled() /// We now want to be informed of bluetooth state registerReceiver(btStateReceiver, btStateReceiver.intent) // if (!isInTestLab) - very important - even in test lab we must request permissions because we need location perms for some of our tests to pass requestPermission() /* not yet working // 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() // Build a GoogleSignInClient with the options specified by gso. UIState.googleSignInClient = GoogleSignIn.getClient(this, gso); */ /* setContent { MeshApp() } */ setContentView(R.layout.activity_main) initToolbar() pager.adapter = tabsAdapter pager.isUserInputEnabled = false // Gestures for screen switching doesn't work so good with the map view // pager.offscreenPageLimit = 0 // Don't keep any offscreen pages around, because we want to make sure our bluetooth scanning stops TabLayoutMediator(tab_layout, pager) { tab, position -> // tab.text = tabInfos[position].text // I think it looks better with icons only tab.icon = getDrawable(tabInfos[position].icon) }.attach() model.isConnected.observe(this, Observer { connected -> updateConnectionStatusImage(connected) }) // Handle any intent handleIntent(intent) askToRate() } private fun initToolbar() { val toolbar = findViewById(R.id.toolbar) as Toolbar setSupportActionBar(toolbar) supportActionBar?.setDisplayShowTitleEnabled(false) } private fun updateConnectionStatusImage(connected: MeshService.ConnectionState) { if (model.actionBarMenu == null) return val (image, tooltip) = when (connected) { MeshService.ConnectionState.CONNECTED -> Pair(R.drawable.cloud_on, R.string.connected) MeshService.ConnectionState.DEVICE_SLEEP -> Pair( R.drawable.ic_twotone_cloud_upload_24, R.string.device_sleeping ) MeshService.ConnectionState.DISCONNECTED -> Pair( R.drawable.cloud_off, R.string.disconnected ) } val item = model.actionBarMenu?.findItem(R.id.connectStatusImage) if (item != null) { item.setIcon(image) item.setTitle(tooltip) } } override fun onNewIntent(intent: Intent) { super.onNewIntent(intent) handleIntent(intent) } private var requestedChannelUrl: Uri? = null /** We keep the usb device here, so later we can give it to our service */ private var usbDevice: UsbDevice? = null /// Handle any itents that were passed into us private fun handleIntent(intent: Intent) { val appLinkAction = intent.action val appLinkData: Uri? = intent.data when (appLinkAction) { Intent.ACTION_VIEW -> { debug("Asked to open a channel URL - ask user if they want to switch to that channel. If so send the config to the radio") requestedChannelUrl = appLinkData // if the device is connected already, process it now if (model.isConnected.value == MeshService.ConnectionState.CONNECTED) perhapsChangeChannel() // We now wait for the device to connect, once connected, we ask the user if they want to switch to the new channel } UsbManager.ACTION_USB_DEVICE_ATTACHED -> { val device: UsbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE)!! debug("Handle USB device attached! $device") usbDevice = device } Intent.ACTION_MAIN -> { } else -> { warn("Unexpected action $appLinkAction") } } } override fun onDestroy() { unregisterReceiver(btStateReceiver) unregisterMeshReceiver() mainScope.cancel("Activity going away") super.onDestroy() } /** * Dispatch incoming result to the correct fragment. */ @SuppressLint("InlinedApi") 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(...); when (requestCode) { RC_SIGN_IN -> { // The Task returned from this call is always completed, no need to attach // a listener. val task: Task = GoogleSignIn.getSignedInAccountFromIntent(data) handleSignInResult(task) } (65536 + RC_SELECT_DEVICE) -> when (resultCode) { Activity.RESULT_OK -> { // User has chosen to pair with the Bluetooth device. val device: BluetoothDevice = data?.getParcelableExtra(CompanionDeviceManager.EXTRA_DEVICE)!! debug("Received BLE pairing ${device.address}") if (device.bondState != BluetoothDevice.BOND_BONDED) { device.createBond() // FIXME - wait for bond to complete } // ... Continue interacting with the paired device. model.meshService?.let { service -> MeshService.changeDeviceAddress(this@MainActivity, service, device.address) } } else -> warn("BLE device select intent failed") } } } private fun handleSignInResult(completedTask: Task) { /* 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) } */ } private var receiverRegistered = false private fun registerMeshReceiver() { unregisterMeshReceiver() val filter = IntentFilter() filter.addAction(MeshService.ACTION_MESH_CONNECTED) filter.addAction(MeshService.ACTION_NODE_CHANGE) filter.addAction(MeshService.ACTION_RECEIVED_DATA) filter.addAction((MeshService.ACTION_MESSAGE_STATUS)) registerReceiver(meshServiceReceiver, filter) receiverRegistered = true; } private fun unregisterMeshReceiver() { if (receiverRegistered) { receiverRegistered = false unregisterReceiver(meshServiceReceiver) } } /// Pull our latest node db from the device private fun updateNodesFromDevice() { model.meshService?.let { service -> // Update our nodeinfos based on data from the device val nodes = service.nodes.map { it.user?.id!! to it }.toMap() model.nodeDB.nodes.value = nodes try { // Pull down our real node ID - This must be done AFTER reading the nodedb because we need the DB to find our nodeinof object model.nodeDB.myId.value = service.myId val ourNodeInfo = model.nodeDB.ourNodeInfo model.ownerName.value = ourNodeInfo?.user?.longName } catch (ex: Exception) { warn("Ignoring failure to get myId, service is probably just uninited... ${ex.message}") } } } /// Called when we gain/lose a connection to our mesh radio private fun onMeshConnectionChanged(connected: MeshService.ConnectionState) { model.isConnected.value = connected debug("connchange ${model.isConnected.value}") if (connected == MeshService.ConnectionState.CONNECTED) { model.meshService?.let { service -> debug("Getting latest radioconfig from service") model.radioConfig.value = MeshProtos.RadioConfig.parseFrom(service.radioConfig) val info = service.myNodeInfo model.myNodeInfo.value = info val isOld = info.minAppVersion > BuildConfig.VERSION_CODE if (isOld) MaterialAlertDialogBuilder(this) .setTitle(getString(R.string.app_too_old)) .setMessage(getString(R.string.must_update)) .setPositiveButton("Okay") { _, _ -> info("User acknowledged app is old") } .show() updateNodesFromDevice() // we have a connection to our device now, do the channel change perhapsChangeChannel() } } } private fun perhapsChangeChannel() { // If the is opening a channel URL, handle it now requestedChannelUrl?.let { url -> try { val channel = Channel(url) requestedChannelUrl = null MaterialAlertDialogBuilder(this) .setTitle(R.string.new_channel_rcvd) .setMessage(getString(R.string.do_you_want_switch).format(channel.name)) .setNeutralButton(R.string.cancel) { _, _ -> // Do nothing } .setPositiveButton(R.string.accept) { _, _ -> debug("Setting channel from URL") try { model.setChannel(channel.settings) } catch (ex: RemoteException) { errormsg("Couldn't change channel ${ex.message}") Toast.makeText( this, "Couldn't change channel, because radio is not yet connected. Please try again.", Toast.LENGTH_SHORT ).show() } } .show() } catch (ex: InvalidProtocolBufferException) { Toast.makeText( this, R.string.channel_invalid, Toast.LENGTH_LONG ).show() } } } override fun dispatchTouchEvent(ev: MotionEvent?): Boolean { return try { super.dispatchTouchEvent(ev) } catch (ex: Throwable) { Exceptions.report( ex, "dispatchTouchEvent" ) // hide this Compose error from the user but report to the mothership false } } private val meshServiceReceiver = object : BroadcastReceiver() { override fun onReceive(context: Context, intent: Intent) = exceptionReporter { debug("Received from mesh service $intent") when (intent.action) { MeshService.ACTION_NODE_CHANGE -> { val info: NodeInfo = intent.getParcelableExtra(EXTRA_NODEINFO)!! debug("UI nodechange $info") // We only care about nodes that have user info info.user?.id?.let { val newnodes = model.nodeDB.nodes.value!! + Pair(it, info) model.nodeDB.nodes.value = newnodes } } MeshService.ACTION_RECEIVED_DATA -> { debug("received new data from service") val payload = intent.getParcelableExtra(EXTRA_PAYLOAD)!! when (payload.dataType) { MeshProtos.Data.Type.CLEAR_TEXT_VALUE -> { model.messagesState.addMessage(payload) } else -> errormsg("Unhandled dataType ${payload.dataType}") } } MeshService.ACTION_MESSAGE_STATUS -> { debug("received message status from service") val id = intent.getIntExtra(EXTRA_PACKET_ID, 0) val status = intent.getParcelableExtra(EXTRA_STATUS)!! model.messagesState.updateStatus(id, status) } MeshService.ACTION_MESH_CONNECTED -> { val connected = MeshService.ConnectionState.valueOf( intent.getStringExtra( EXTRA_CONNECTED )!! ) onMeshConnectionChanged(connected) } else -> TODO() } } } private val mesh = object : ServiceClient({ com.geeksville.mesh.IMeshService.Stub.asInterface(it) }) { override fun onConnected(service: com.geeksville.mesh.IMeshService) { /* Note: we must call this callback in a coroutine. Because apparently there is only a single activity looper thread. and if that onConnected override also tries to do a service operation we can deadlock. Old buggy stack trace: at sun.misc.Unsafe.park (Unsafe.java) - waiting on an unknown object at java.util.concurrent.locks.LockSupport.park (LockSupport.java:190) at java.util.concurrent.locks.AbstractQueuedSynchronizer$ConditionObject.await (AbstractQueuedSynchronizer.java:2067) at com.geeksville.android.ServiceClient.waitConnect (ServiceClient.java:46) at com.geeksville.android.ServiceClient.getService (ServiceClient.java:27) at com.geeksville.mesh.service.MeshService$binder$1$setDeviceAddress$1.invoke (MeshService.java:1519) at com.geeksville.mesh.service.MeshService$binder$1$setDeviceAddress$1.invoke (MeshService.java:1514) at com.geeksville.util.ExceptionsKt.toRemoteExceptions (ExceptionsKt.java:56) at com.geeksville.mesh.service.MeshService$binder$1.setDeviceAddress (MeshService.java:1516) at com.geeksville.mesh.MainActivity$mesh$1$onConnected$1.invoke (MainActivity.java:743) at com.geeksville.mesh.MainActivity$mesh$1$onConnected$1.invoke (MainActivity.java:734) at com.geeksville.util.ExceptionsKt.exceptionReporter (ExceptionsKt.java:34) at com.geeksville.mesh.MainActivity$mesh$1.onConnected (MainActivity.java:738) at com.geeksville.mesh.MainActivity$mesh$1.onConnected (MainActivity.java:734) at com.geeksville.android.ServiceClient$connection$1$onServiceConnected$1.invoke (ServiceClient.java:89) at com.geeksville.android.ServiceClient$connection$1$onServiceConnected$1.invoke (ServiceClient.java:84) at com.geeksville.util.ExceptionsKt.exceptionReporter (ExceptionsKt.java:34) at com.geeksville.android.ServiceClient$connection$1.onServiceConnected (ServiceClient.java:85) at android.app.LoadedApk$ServiceDispatcher.doConnected (LoadedApk.java:2067) at android.app.LoadedApk$ServiceDispatcher$RunConnection.run (LoadedApk.java:2099) at android.os.Handler.handleCallback (Handler.java:883) at android.os.Handler.dispatchMessage (Handler.java:100) at android.os.Looper.loop (Looper.java:237) at android.app.ActivityThread.main (ActivityThread.java:8016) at java.lang.reflect.Method.invoke (Method.java) at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run (RuntimeInit.java:493) at com.android.internal.os.ZygoteInit.main (ZygoteInit.java:1076) */ mainScope.handledLaunch { model.meshService = service usbDevice?.let { usb -> debug("Switching to USB radio ${usb.deviceName}") service.setDeviceAddress(SerialInterface.toInterfaceName(usb.deviceName)) usbDevice = null // Only switch once - thereafter it should be stored in settings } // We don't start listening for packets until after we are connected to the service registerMeshReceiver() // Init our messages table with the service's record of past text messages val msgs = service.oldMessages debug("Service provided ${msgs.size} messages") model.messagesState.setMessages(msgs) val connectionState = MeshService.ConnectionState.valueOf(service.connectionState()) // if we are not connected, onMeshConnectionChange won't fetch nodes from the service // in that case, we do it here - because the service certainly has a better idea of node db that we have if (connectionState != MeshService.ConnectionState.CONNECTED) updateNodesFromDevice() try { // We won't receive a notify for the initial state of connection, so we force an update here onMeshConnectionChanged(connectionState) } catch (ex: RemoteException) { // If we get an exception while reading our service config, the device might have gone away, double check to see if we are really connected errormsg("Device error during init ${ex.message}") model.isConnected.value = MeshService.ConnectionState.valueOf(service.connectionState()) } debug("connected to mesh service, isConnected=${model.isConnected.value}") } } override fun onDisconnected() { unregisterMeshReceiver() model.meshService = null } } private fun bindMeshService() { debug("Binding to mesh service!") // we bind using the well known name, to make sure 3rd party apps could also if (model.meshService != null) Exceptions.reportError("meshService was supposed to be null, ignoring (but reporting a bug)") try { MeshService.startService(this) // Start the service so it stays running even after we unbind } catch (ex: Exception) { // Old samsung phones have a race condition andthis might rarely fail. Which is probably find because the bind will be sufficient most of the time errormsg("Failed to start service from activity - but ignoring because bind will work ${ex.message}") } // ALSO bind so we can use the api mesh.connect(this, MeshService.intent, Context.BIND_AUTO_CREATE + Context.BIND_ABOVE_CLIENT) } 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!") mesh.close() model.meshService = null } override fun onStop() { unregisterMeshReceiver() // No point in receiving updates while the GUI is gone, we'll get them when the user launches the activity unbindMeshService() super.onStop() } override fun onStart() { super.onStart() warnMissingPermissions() // Ask to start bluetooth if no USB devices are visible val hasUSB = SerialInterface.findDrivers(this).isNotEmpty() if (!isInTestLab && !hasUSB) { bluetoothAdapter?.let { if (!it.isEnabled) { val enableBtIntent = Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE) startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT) } } } bindMeshService() val bonded = RadioInterfaceService.getBondedDeviceAddress(this) != null if (!bonded && usbDevice == null) // we will handle USB later showSettingsPage() } private fun showSettingsPage() { pager.currentItem = 5 } 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) model.actionBarMenu = menu updateConnectionStatusImage(model.isConnected.value!!) return true } 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.about -> { getVersionInfo() return true } R.id.connectStatusImage -> { Toast.makeText(applicationContext, item.title, Toast.LENGTH_SHORT).show() return true } else -> super.onOptionsItemSelected(item) } } private fun getVersionInfo() { try { val packageInfo: PackageInfo = packageManager.getPackageInfo(packageName, 0) val versionName = packageInfo.versionName Toast.makeText(applicationContext, versionName, Toast.LENGTH_LONG).show() } catch (e: PackageManager.NameNotFoundException) { errormsg("Can not find the version: ${e.message}") } } }