MainActivity.kt 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377
  1. /*
  2. * Nextcloud Talk application
  3. *
  4. * @author Mario Danic
  5. * @author Andy Scherzinger
  6. * @author Marcel Hibbe
  7. * Copyright (C) 2023 Marcel Hibbe <dev@mhibbe.de>
  8. * Copyright (C) 2021 Andy Scherzinger (infoi@andy-scherzinger.de)
  9. * Copyright (C) 2017 Mario Danic (mario@lovelyhq.com)
  10. *
  11. * This program is free software: you can redistribute it and/or modify
  12. * it under the terms of the GNU General Public License as published by
  13. * the Free Software Foundation, either version 3 of the License, or
  14. * at your option) any later version.
  15. *
  16. * This program is distributed in the hope that it will be useful,
  17. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  19. * GNU General Public License for more details.
  20. *
  21. * You should have received a copy of the GNU General Public License
  22. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  23. */
  24. package com.nextcloud.talk.activities
  25. import android.app.KeyguardManager
  26. import android.content.Context
  27. import android.content.Intent
  28. import android.os.Bundle
  29. import android.provider.ContactsContract
  30. import android.text.TextUtils
  31. import android.util.Log
  32. import androidx.lifecycle.DefaultLifecycleObserver
  33. import androidx.lifecycle.LifecycleOwner
  34. import androidx.lifecycle.ProcessLifecycleOwner
  35. import autodagger.AutoInjector
  36. import com.bluelinelabs.conductor.Conductor
  37. import com.bluelinelabs.conductor.Router
  38. import com.bluelinelabs.conductor.RouterTransaction
  39. import com.bluelinelabs.conductor.changehandler.HorizontalChangeHandler
  40. import com.bluelinelabs.conductor.changehandler.VerticalChangeHandler
  41. import com.google.android.material.snackbar.Snackbar
  42. import com.nextcloud.talk.BuildConfig
  43. import com.nextcloud.talk.R
  44. import com.nextcloud.talk.api.NcApi
  45. import com.nextcloud.talk.application.NextcloudTalkApplication
  46. import com.nextcloud.talk.chat.ChatActivity
  47. import com.nextcloud.talk.controllers.ServerSelectionController
  48. import com.nextcloud.talk.controllers.WebViewLoginController
  49. import com.nextcloud.talk.controllers.base.providers.ActionBarProvider
  50. import com.nextcloud.talk.conversationlist.ConversationsListActivity
  51. import com.nextcloud.talk.data.user.model.User
  52. import com.nextcloud.talk.databinding.ActivityMainBinding
  53. import com.nextcloud.talk.lock.LockedActivity
  54. import com.nextcloud.talk.models.json.conversations.RoomOverall
  55. import com.nextcloud.talk.users.UserManager
  56. import com.nextcloud.talk.utils.ApiUtils
  57. import com.nextcloud.talk.utils.SecurityUtils
  58. import com.nextcloud.talk.utils.bundle.BundleKeys
  59. import com.nextcloud.talk.utils.bundle.BundleKeys.ADD_ACCOUNT
  60. import com.nextcloud.talk.utils.bundle.BundleKeys.KEY_ACTIVE_CONVERSATION
  61. import com.nextcloud.talk.utils.bundle.BundleKeys.KEY_ROOM_ID
  62. import com.nextcloud.talk.utils.bundle.BundleKeys.KEY_ROOM_TOKEN
  63. import com.nextcloud.talk.utils.bundle.BundleKeys.KEY_USER_ENTITY
  64. import io.reactivex.Observer
  65. import io.reactivex.SingleObserver
  66. import io.reactivex.android.schedulers.AndroidSchedulers
  67. import io.reactivex.disposables.Disposable
  68. import io.reactivex.schedulers.Schedulers
  69. import org.parceler.Parcels
  70. import javax.inject.Inject
  71. @AutoInjector(NextcloudTalkApplication::class)
  72. class MainActivity : BaseActivity(), ActionBarProvider {
  73. lateinit var binding: ActivityMainBinding
  74. @Inject
  75. lateinit var ncApi: NcApi
  76. @Inject
  77. lateinit var userManager: UserManager
  78. private var router: Router? = null
  79. @Suppress("Detekt.TooGenericExceptionCaught")
  80. override fun onCreate(savedInstanceState: Bundle?) {
  81. Log.d(TAG, "onCreate: Activity: " + System.identityHashCode(this).toString())
  82. super.onCreate(savedInstanceState)
  83. ProcessLifecycleOwner.get().lifecycle.addObserver(object : DefaultLifecycleObserver {
  84. override fun onStart(owner: LifecycleOwner) {
  85. lockScreenIfConditionsApply()
  86. }
  87. })
  88. // Set the default theme to replace the launch screen theme.
  89. setTheme(R.style.AppTheme)
  90. binding = ActivityMainBinding.inflate(layoutInflater)
  91. setContentView(binding.root)
  92. NextcloudTalkApplication.sharedApplication!!.componentApplication.inject(this)
  93. setSupportActionBar(binding.toolbar)
  94. router = Conductor.attachRouter(this, binding.controllerContainer, savedInstanceState)
  95. if (intent.hasExtra(ADD_ACCOUNT) && intent.getBooleanExtra(ADD_ACCOUNT, false)) {
  96. addAccount()
  97. } else if (intent.hasExtra(BundleKeys.KEY_FROM_NOTIFICATION_START_CALL)) {
  98. onNewIntent(intent)
  99. } else if (!router!!.hasRootController()) {
  100. if (!appPreferences.isDbRoomMigrated) {
  101. appPreferences.isDbRoomMigrated = true
  102. }
  103. userManager.users.subscribe(object : SingleObserver<List<User>> {
  104. override fun onSubscribe(d: Disposable) {
  105. // unused atm
  106. }
  107. override fun onSuccess(users: List<User>) {
  108. if (users.isNotEmpty()) {
  109. runOnUiThread {
  110. openConversationList()
  111. }
  112. } else {
  113. runOnUiThread {
  114. launchLoginScreen()
  115. }
  116. }
  117. }
  118. override fun onError(e: Throwable) {
  119. Log.e(TAG, "Error loading existing users", e)
  120. }
  121. })
  122. }
  123. }
  124. fun lockScreenIfConditionsApply() {
  125. val keyguardManager = getSystemService(Context.KEYGUARD_SERVICE) as KeyguardManager
  126. if (keyguardManager.isKeyguardSecure && appPreferences.isScreenLocked) {
  127. if (!SecurityUtils.checkIfWeAreAuthenticated(appPreferences.screenLockTimeout)) {
  128. val lockIntent = Intent(context, LockedActivity::class.java)
  129. startActivity(lockIntent)
  130. }
  131. }
  132. }
  133. private fun launchLoginScreen() {
  134. if (!TextUtils.isEmpty(resources.getString(R.string.weblogin_url))) {
  135. router!!.pushController(
  136. RouterTransaction.with(
  137. WebViewLoginController(resources.getString(R.string.weblogin_url), false)
  138. )
  139. .pushChangeHandler(HorizontalChangeHandler())
  140. .popChangeHandler(HorizontalChangeHandler())
  141. )
  142. } else {
  143. router!!.setRoot(
  144. RouterTransaction.with(ServerSelectionController())
  145. .pushChangeHandler(HorizontalChangeHandler())
  146. .popChangeHandler(HorizontalChangeHandler())
  147. )
  148. }
  149. }
  150. override fun onStart() {
  151. Log.d(TAG, "onStart: Activity: " + System.identityHashCode(this).toString())
  152. super.onStart()
  153. logRouterBackStack(router!!)
  154. }
  155. override fun onResume() {
  156. Log.d(TAG, "onResume: Activity: " + System.identityHashCode(this).toString())
  157. super.onResume()
  158. if (appPreferences.isScreenLocked) {
  159. SecurityUtils.createKey(appPreferences.screenLockTimeout)
  160. }
  161. }
  162. override fun onPause() {
  163. Log.d(TAG, "onPause: Activity: " + System.identityHashCode(this).toString())
  164. super.onPause()
  165. }
  166. override fun onStop() {
  167. Log.d(TAG, "onStop: Activity: " + System.identityHashCode(this).toString())
  168. super.onStop()
  169. }
  170. private fun openConversationList() {
  171. val intent = Intent(this, ConversationsListActivity::class.java)
  172. intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP)
  173. intent.putExtras(Bundle())
  174. startActivity(intent)
  175. }
  176. fun addAccount() {
  177. router!!.pushController(
  178. RouterTransaction.with(ServerSelectionController())
  179. .pushChangeHandler(VerticalChangeHandler())
  180. .popChangeHandler(VerticalChangeHandler())
  181. )
  182. }
  183. private fun handleActionFromContact(intent: Intent) {
  184. if (intent.action == Intent.ACTION_VIEW && intent.data != null) {
  185. val cursor = contentResolver.query(intent.data!!, null, null, null, null)
  186. var userId = ""
  187. if (cursor != null) {
  188. if (cursor.moveToFirst()) {
  189. // userId @ server
  190. userId = cursor.getString(cursor.getColumnIndexOrThrow(ContactsContract.Data.DATA1))
  191. }
  192. cursor.close()
  193. }
  194. when (intent.type) {
  195. "vnd.android.cursor.item/vnd.com.nextcloud.talk2.chat" -> {
  196. val user = userId.substringBeforeLast("@")
  197. val baseUrl = userId.substringAfterLast("@")
  198. if (userManager.currentUser.blockingGet()?.baseUrl?.endsWith(baseUrl) == true) {
  199. startConversation(user)
  200. } else {
  201. Snackbar.make(
  202. binding.controllerContainer,
  203. R.string.nc_phone_book_integration_account_not_found,
  204. Snackbar.LENGTH_LONG
  205. ).show()
  206. }
  207. }
  208. }
  209. }
  210. }
  211. private fun startConversation(userId: String) {
  212. val roomType = "1"
  213. val currentUser = userManager.currentUser.blockingGet()
  214. val apiVersion = ApiUtils.getConversationApiVersion(currentUser, intArrayOf(ApiUtils.APIv4, 1))
  215. val credentials = ApiUtils.getCredentials(currentUser?.username, currentUser?.token)
  216. val retrofitBucket = ApiUtils.getRetrofitBucketForCreateRoom(
  217. apiVersion,
  218. currentUser?.baseUrl,
  219. roomType,
  220. null,
  221. userId,
  222. null
  223. )
  224. ncApi.createRoom(
  225. credentials,
  226. retrofitBucket.url,
  227. retrofitBucket.queryMap
  228. )
  229. .subscribeOn(Schedulers.io())
  230. .observeOn(AndroidSchedulers.mainThread())
  231. .subscribe(object : Observer<RoomOverall> {
  232. override fun onSubscribe(d: Disposable) {
  233. // unused atm
  234. }
  235. override fun onNext(roomOverall: RoomOverall) {
  236. val bundle = Bundle()
  237. bundle.putParcelable(KEY_USER_ENTITY, currentUser)
  238. bundle.putString(KEY_ROOM_TOKEN, roomOverall.ocs!!.data!!.token)
  239. bundle.putString(KEY_ROOM_ID, roomOverall.ocs!!.data!!.roomId)
  240. // FIXME once APIv2 or later is used only, the createRoom already returns all the data
  241. ncApi.getRoom(
  242. credentials,
  243. ApiUtils.getUrlForRoom(
  244. apiVersion,
  245. currentUser?.baseUrl,
  246. roomOverall.ocs!!.data!!.token
  247. )
  248. )
  249. .subscribeOn(Schedulers.io())
  250. .observeOn(AndroidSchedulers.mainThread())
  251. .subscribe(object : Observer<RoomOverall> {
  252. override fun onSubscribe(d: Disposable) {
  253. // unused atm
  254. }
  255. override fun onNext(roomOverall: RoomOverall) {
  256. bundle.putParcelable(
  257. KEY_ACTIVE_CONVERSATION,
  258. Parcels.wrap(roomOverall.ocs!!.data)
  259. )
  260. val chatIntent = Intent(context, ChatActivity::class.java)
  261. chatIntent.putExtras(bundle)
  262. startActivity(chatIntent)
  263. }
  264. override fun onError(e: Throwable) {
  265. // unused atm
  266. }
  267. override fun onComplete() {
  268. // unused atm
  269. }
  270. })
  271. }
  272. override fun onError(e: Throwable) {
  273. // unused atm
  274. }
  275. override fun onComplete() {
  276. // unused atm
  277. }
  278. })
  279. }
  280. override fun onNewIntent(intent: Intent) {
  281. super.onNewIntent(intent)
  282. Log.d(TAG, "onNewIntent Activity: " + System.identityHashCode(this).toString())
  283. val user = intent.getParcelableExtra<User>(KEY_USER_ENTITY)
  284. if (user != null && userManager.setUserAsActive(user).blockingGet()) {
  285. handleIntent(intent)
  286. }
  287. }
  288. private fun handleIntent(intent: Intent) {
  289. handleActionFromContact(intent)
  290. if (intent.hasExtra(BundleKeys.KEY_FROM_NOTIFICATION_START_CALL)) {
  291. if (intent.getBooleanExtra(BundleKeys.KEY_FROM_NOTIFICATION_START_CALL, false)) {
  292. if (!router!!.hasRootController()) {
  293. openConversationList()
  294. }
  295. val callNotificationIntent = Intent(this, CallNotificationActivity::class.java)
  296. intent.extras?.let { callNotificationIntent.putExtras(it) }
  297. startActivity(callNotificationIntent)
  298. } else {
  299. logRouterBackStack(router!!)
  300. val chatIntent = Intent(context, ChatActivity::class.java)
  301. chatIntent.putExtras(intent.extras!!)
  302. startActivity(chatIntent)
  303. logRouterBackStack(router!!)
  304. }
  305. }
  306. }
  307. override fun onBackPressed() {
  308. if (!router!!.handleBack()) {
  309. super.onBackPressed()
  310. }
  311. }
  312. private fun logRouterBackStack(router: Router) {
  313. if (BuildConfig.DEBUG) {
  314. val backstack = router.backstack
  315. var routerTransaction: RouterTransaction?
  316. Log.d(TAG, " backstack size: " + router.backstackSize)
  317. for (i in 0 until router.backstackSize) {
  318. routerTransaction = backstack[i]
  319. Log.d(TAG, " controller: " + routerTransaction.controller)
  320. }
  321. }
  322. }
  323. companion object {
  324. private const val TAG = "MainActivity"
  325. }
  326. }