/* * Nextcloud Talk application * * @author Andy Scherzinger * @author Mario Danic * @author Tim Krüger * @author Ezhil Shanmugham * Copyright (C) 2021 Tim Krüger * Copyright (C) 2021-2022 Andy Scherzinger * Copyright (C) 2017 Mario Danic (mario@lovelyhq.com) * Copyright (C) 2023 Ezhil Shanmugham * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.nextcloud.talk.settings import android.animation.Animator import android.animation.AnimatorListenerAdapter import android.annotation.SuppressLint import android.app.KeyguardManager import android.content.Context import android.content.DialogInterface import android.content.DialogInterface.OnShowListener import android.content.Intent import android.content.pm.PackageManager import android.content.res.ColorStateList import android.graphics.PorterDuff import android.graphics.drawable.ColorDrawable import android.media.RingtoneManager import android.net.Uri import android.os.Build import android.os.Bundle import android.provider.Settings import android.security.KeyChain import android.text.Editable import android.text.InputType import android.text.TextUtils import android.text.TextWatcher import android.util.Log import android.view.View import android.view.WindowManager import android.widget.EditText import android.widget.LinearLayout import android.widget.Toast import androidx.appcompat.app.AlertDialog import androidx.appcompat.view.ContextThemeWrapper import androidx.core.content.ContextCompat import androidx.core.view.ViewCompat import androidx.work.OneTimeWorkRequest import androidx.work.WorkInfo import androidx.work.WorkManager import autodagger.AutoInjector import com.google.android.material.dialog.MaterialAlertDialogBuilder import com.google.android.material.snackbar.Snackbar import com.google.android.material.textfield.TextInputLayout import com.nextcloud.android.common.ui.theme.utils.ColorRole import com.nextcloud.talk.BuildConfig import com.nextcloud.talk.R import com.nextcloud.talk.activities.BaseActivity import com.nextcloud.talk.activities.MainActivity import com.nextcloud.talk.api.NcApi import com.nextcloud.talk.application.NextcloudTalkApplication import com.nextcloud.talk.application.NextcloudTalkApplication.Companion.setAppTheme import com.nextcloud.talk.data.user.model.User import com.nextcloud.talk.databinding.ActivitySettingsBinding import com.nextcloud.talk.jobs.AccountRemovalWorker import com.nextcloud.talk.jobs.CapabilitiesWorker import com.nextcloud.talk.jobs.ContactAddressBookWorker import com.nextcloud.talk.jobs.ContactAddressBookWorker.Companion.checkPermission import com.nextcloud.talk.jobs.ContactAddressBookWorker.Companion.deleteAll import com.nextcloud.talk.models.json.generic.GenericOverall import com.nextcloud.talk.models.json.userprofile.UserProfileOverall import com.nextcloud.talk.profile.ProfileActivity import com.nextcloud.talk.users.UserManager import com.nextcloud.talk.utils.ApiUtils import com.nextcloud.talk.utils.DisplayUtils import com.nextcloud.talk.utils.LoggingUtils.sendMailWithAttachment import com.nextcloud.talk.utils.NotificationUtils import com.nextcloud.talk.utils.NotificationUtils.getCallRingtoneUri import com.nextcloud.talk.utils.NotificationUtils.getMessageRingtoneUri import com.nextcloud.talk.utils.SecurityUtils import com.nextcloud.talk.utils.database.user.CapabilitiesUtilNew import com.nextcloud.talk.utils.database.user.CurrentUserProviderNew import com.nextcloud.talk.utils.preferences.AppPreferencesImpl import com.nextcloud.talk.utils.singletons.ApplicationWideMessageHolder import io.reactivex.Observer import io.reactivex.android.schedulers.AndroidSchedulers import io.reactivex.disposables.Disposable import io.reactivex.schedulers.Schedulers import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers import kotlinx.coroutines.ExperimentalCoroutinesApi import kotlinx.coroutines.flow.Flow import kotlinx.coroutines.launch import okhttp3.MediaType.Companion.toMediaTypeOrNull import okhttp3.RequestBody.Companion.toRequestBody import java.net.URI import java.net.URISyntaxException import java.util.Locale import javax.inject.Inject @Suppress("LargeClass", "TooManyFunctions") @AutoInjector(NextcloudTalkApplication::class) class SettingsActivity : BaseActivity() { private lateinit var binding: ActivitySettingsBinding @Inject lateinit var ncApi: NcApi @Inject lateinit var userManager: UserManager @Inject lateinit var currentUserProvider: CurrentUserProviderNew private var currentUser: User? = null private var credentials: String? = null private lateinit var proxyTypeFlow: Flow private lateinit var proxyCredentialFlow: Flow private lateinit var screenSecurityFlow: Flow private lateinit var screenLockFlow: Flow private lateinit var screenLockTimeoutFlow: Flow private lateinit var themeFlow: Flow private lateinit var readPrivacyFlow: Flow private lateinit var typingStatusFlow: Flow private lateinit var phoneBookIntegrationFlow: Flow private var profileQueryDisposable: Disposable? = null private var dbQueryDisposable: Disposable? = null override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) NextcloudTalkApplication.sharedApplication!!.componentApplication.inject(this) binding = ActivitySettingsBinding.inflate(layoutInflater) setupActionBar() setContentView(binding.root) setupSystemColors() binding.avatarImage.let { ViewCompat.setTransitionName(it, "userAvatar.transitionTag") } getCurrentUser() setupLicenceSetting() if (Build.VERSION.SDK_INT < Build.VERSION_CODES.O) { binding.settingsIncognitoKeyboard.visibility = View.GONE } binding.settingsScreenLockSummary.text = String.format( Locale.getDefault(), resources!!.getString(R.string.nc_settings_screen_lock_desc), resources!!.getString(R.string.nc_app_product_name) ) setupPrivacyUrl() setupSourceCodeUrl() binding.settingsVersionSummary.text = String.format("v" + BuildConfig.VERSION_NAME) setupSoundSettings() setupPhoneBookIntegration() setupClientCertView() } override fun onResume() { super.onResume() supportActionBar?.show() dispose(null) loadCapabilitiesAndUpdateSettings() binding.settingsVersion.setOnClickListener { sendLogs() } if (!TextUtils.isEmpty(currentUser!!.clientCertificate)) { binding.settingsClientCertTitle.setText(R.string.nc_client_cert_change) } else { binding.settingsClientCertTitle.setText(R.string.nc_client_cert_setup) } setupCheckables() setupScreenLockSetting() if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { binding.settingsNotificationsTitle.text = resources!!.getString( R.string.nc_settings_notification_sounds_post_oreo ) } val callRingtoneUri = getCallRingtoneUri(context, (appPreferences)) binding.callsRingtone.text = getRingtoneName(context, callRingtoneUri) val messageRingtoneUri = getMessageRingtoneUri(context, (appPreferences)) binding.messagesRingtone.text = getRingtoneName(context, messageRingtoneUri) setupProxyTypeSettings() setupProxyCredentialSettings() registerChangeListeners() if (currentUser != null) { binding.domainText.text = Uri.parse(currentUser!!.baseUrl).host setupServerAgeWarning() if (currentUser!!.displayName != null) { binding.nameText.text = currentUser!!.displayName } DisplayUtils.loadAvatarImage(currentUser, binding.avatarImage, false) setupProfileQueryDisposable() binding.settingsRemoveAccount.setOnClickListener { showRemoveAccountWarning() } } setupMessageView() binding.settingsName.visibility = View.VISIBLE binding.settingsName.setOnClickListener { val intent = Intent(this, ProfileActivity::class.java) startActivity(intent) } themeTitles() themeSwitchPreferences() } private fun loadCapabilitiesAndUpdateSettings() { val capabilitiesWork = OneTimeWorkRequest.Builder(CapabilitiesWorker::class.java).build() WorkManager.getInstance(context).enqueue(capabilitiesWork) WorkManager.getInstance(context).getWorkInfoByIdLiveData(capabilitiesWork.id) .observe(this) { workInfo -> if (workInfo?.state == WorkInfo.State.SUCCEEDED) { getCurrentUser() setupCheckables() } } } private fun setupActionBar() { setSupportActionBar(binding.settingsToolbar) binding.settingsToolbar.setNavigationOnClickListener { onBackPressedDispatcher.onBackPressed() } supportActionBar?.setDisplayHomeAsUpEnabled(true) supportActionBar?.setDisplayShowHomeEnabled(true) supportActionBar?.setIcon(ColorDrawable(resources!!.getColor(android.R.color.transparent, null))) supportActionBar?.title = context.getString(R.string.nc_settings) viewThemeUtils.material.themeToolbar(binding.settingsToolbar) } private fun getCurrentUser() { currentUser = currentUserProvider.currentUser.blockingGet() credentials = ApiUtils.getCredentials(currentUser!!.username, currentUser!!.token) } private fun setupPhoneBookIntegration() { if (CapabilitiesUtilNew.isPhoneBookIntegrationAvailable(currentUser!!)) { binding.settingsPhoneBookIntegration.visibility = View.VISIBLE } else { binding.settingsPhoneBookIntegration.visibility = View.GONE } } private fun setupSoundSettings() { if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { binding.settingsCallSound.setOnClickListener { val intent = Intent(Settings.ACTION_CHANNEL_NOTIFICATION_SETTINGS) intent.putExtra(Settings.EXTRA_APP_PACKAGE, BuildConfig.APPLICATION_ID) intent.putExtra( Settings.EXTRA_CHANNEL_ID, NotificationUtils.NotificationChannels.NOTIFICATION_CHANNEL_CALLS_V4.name ) startActivity(intent) } binding.settingsMessageSound.setOnClickListener { val intent = Intent(Settings.ACTION_CHANNEL_NOTIFICATION_SETTINGS) intent.putExtra(Settings.EXTRA_APP_PACKAGE, BuildConfig.APPLICATION_ID) intent.putExtra( Settings.EXTRA_CHANNEL_ID, NotificationUtils.NotificationChannels.NOTIFICATION_CHANNEL_MESSAGES_V4.name ) startActivity(intent) } } else { Log.e(TAG, "setupSoundSettings currently not supported for versions < Build.VERSION_CODES.O") } } private fun setupSourceCodeUrl() { if (!TextUtils.isEmpty(resources!!.getString(R.string.nc_source_code_url))) { binding.settingsSourceCode.setOnClickListener { startActivity( Intent( Intent.ACTION_VIEW, Uri.parse(resources!!.getString(R.string.nc_source_code_url)) ) ) } } else { binding.settingsSourceCode.visibility = View.GONE } } private fun setupPrivacyUrl() { if (!TextUtils.isEmpty(resources!!.getString(R.string.nc_privacy_url))) { binding.settingsPrivacy.setOnClickListener { startActivity( Intent( Intent.ACTION_VIEW, Uri.parse(resources!!.getString(R.string.nc_privacy_url)) ) ) } } else { binding.settingsPrivacy.visibility = View.GONE } } private fun setupLicenceSetting() { if (!TextUtils.isEmpty(resources!!.getString(R.string.nc_gpl3_url))) { binding.settingsLicence.setOnClickListener { startActivity( Intent( Intent.ACTION_VIEW, Uri.parse(resources!!.getString(R.string.nc_gpl3_url)) ) ) } } else { binding.settingsLicence.visibility = View.GONE } } private fun setupClientCertView() { var host: String? = null var port = -1 val uri: URI try { uri = URI(currentUser!!.baseUrl) host = uri.host port = uri.port Log.d(TAG, "uri is $uri") } catch (e: URISyntaxException) { Log.e(TAG, "Failed to create uri") } binding.settingsClientCert.setOnClickListener { KeyChain.choosePrivateKeyAlias( this, { alias: String? -> var finalAlias: String? = alias runOnUiThread { if (finalAlias != null) { binding.settingsClientCertTitle.setText(R.string.nc_client_cert_change) } else { binding.settingsClientCertTitle.setText(R.string.nc_client_cert_setup) } } if (finalAlias == null) { finalAlias = "" } Log.d(TAG, "host: $host and port: $port") currentUser!!.clientCertificate = finalAlias userManager.updateOrCreateUser(currentUser!!) }, arrayOf("RSA", "EC"), null, host, port, currentUser!!.clientCertificate ) } } @OptIn(ExperimentalCoroutinesApi::class) private fun registerChangeListeners() { val appPreferences = AppPreferencesImpl(context) proxyTypeFlow = appPreferences.readString(AppPreferencesImpl.PROXY_TYPE) proxyCredentialFlow = appPreferences.readBoolean(AppPreferencesImpl.PROXY_CRED) screenSecurityFlow = appPreferences.readBoolean(AppPreferencesImpl.SCREEN_SECURITY) screenLockFlow = appPreferences.readBoolean(AppPreferencesImpl.SCREEN_LOCK) screenLockTimeoutFlow = appPreferences.readString(AppPreferencesImpl.SCREEN_LOCK_TIMEOUT) val themeKey = context.resources.getString(R.string.nc_settings_theme_key) themeFlow = appPreferences.readString(themeKey) val privacyKey = context.resources.getString(R.string.nc_settings_read_privacy_key) readPrivacyFlow = appPreferences.readBoolean(privacyKey) typingStatusFlow = appPreferences.readBoolean(AppPreferencesImpl.TYPING_STATUS) phoneBookIntegrationFlow = appPreferences.readBoolean(AppPreferencesImpl.PHONE_BOOK_INTEGRATION) var pos = resources.getStringArray(R.array.screen_lock_timeout_entry_values).indexOf( appPreferences.screenLockTimeout ) binding.settingsScreenLockTimeoutLayoutDropdown.setText( resources.getStringArray(R.array.screen_lock_timeout_descriptions)[pos] ) binding.settingsScreenLockTimeoutLayoutDropdown.setSimpleItems(R.array.screen_lock_timeout_descriptions) binding.settingsScreenLockTimeoutLayoutDropdown.setOnItemClickListener { _, _, position, _ -> val entryVal: String = resources.getStringArray(R.array.screen_lock_timeout_entry_values)[position] appPreferences.screenLockTimeout = entryVal SecurityUtils.createKey(entryVal) } pos = resources.getStringArray(R.array.theme_entry_values).indexOf(appPreferences.theme) binding.settingsTheme.setText(resources.getStringArray(R.array.theme_descriptions)[pos]) binding.settingsTheme.setSimpleItems(R.array.theme_descriptions) binding.settingsTheme.setOnItemClickListener { _, _, position, _ -> val entryVal: String = resources.getStringArray(R.array.theme_entry_values)[position] appPreferences.theme = entryVal } observeProxyType() observeProxyCredential() observeScreenSecurity() observeScreenLock() observeTheme() observeReadPrivacy() observeTypingStatus() } fun sendLogs() { if (resources!!.getBoolean(R.bool.nc_is_debug)) { sendMailWithAttachment((context)) } } private fun showRemoveAccountWarning() { binding.messageText.context?.let { val materialAlertDialogBuilder = MaterialAlertDialogBuilder(it) .setTitle(R.string.nc_settings_remove_account) .setMessage(R.string.nc_settings_remove_confirmation) .setPositiveButton(R.string.nc_settings_remove) { _, _ -> removeCurrentAccount() } .setNegativeButton(R.string.nc_cancel) { _, _ -> // unused atm } viewThemeUtils.dialog.colorMaterialAlertDialogBackground( it, materialAlertDialogBuilder ) val dialog = materialAlertDialogBuilder.show() viewThemeUtils.platform.colorTextButtons( dialog.getButton(AlertDialog.BUTTON_POSITIVE), dialog.getButton(AlertDialog.BUTTON_NEGATIVE) ) } } @SuppressLint("CheckResult") private fun removeCurrentAccount() { userManager.scheduleUserForDeletionWithId(currentUser!!.id!!).blockingGet() val accountRemovalWork = OneTimeWorkRequest.Builder(AccountRemovalWorker::class.java).build() WorkManager.getInstance(applicationContext).enqueue(accountRemovalWork) WorkManager.getInstance(context).getWorkInfoByIdLiveData(accountRemovalWork.id) .observeForever { workInfo: WorkInfo -> when (workInfo.state) { WorkInfo.State.SUCCEEDED -> { val text = String.format( context.resources.getString(R.string.nc_deleted_user), currentUser!!.displayName ) Toast.makeText( context, text, Toast.LENGTH_LONG ).show() restartApp() } WorkInfo.State.FAILED, WorkInfo.State.CANCELLED -> { Toast.makeText( context, context.resources.getString(R.string.nc_common_error_sorry), Toast.LENGTH_LONG ).show() Log.e(TAG, "something went wrong when deleting user with id " + currentUser!!.userId) restartApp() } else -> {} } } } private fun restartApp() { val intent = Intent(context, MainActivity::class.java) intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) startActivity(intent) } private fun getRingtoneName(context: Context, ringtoneUri: Uri?): String { return if (ringtoneUri == null) { resources!!.getString(R.string.nc_settings_no_ringtone) } else if ((NotificationUtils.DEFAULT_CALL_RINGTONE_URI == ringtoneUri.toString()) || (NotificationUtils.DEFAULT_MESSAGE_RINGTONE_URI == ringtoneUri.toString()) ) { resources!!.getString(R.string.nc_settings_default_ringtone) } else { val r = RingtoneManager.getRingtone(context, ringtoneUri) r.getTitle(context) } } private fun themeSwitchPreferences() { binding.run { listOf( settingsScreenLockSwitch, settingsScreenSecuritySwitch, settingsIncognitoKeyboardSwitch, settingsPhoneBookIntegrationSwitch, settingsReadPrivacySwitch, settingsTypingStatusSwitch, settingsProxyUseCredentialsSwitch ).forEach(viewThemeUtils.talk::colorSwitch) } } private fun themeTitles() { binding.run { listOf( settingsNotificationsTitle, settingsAboutTitle, settingsAdvancedTitle, settingsAppearanceTitle, settingsPrivacyTitle ).forEach(viewThemeUtils.platform::colorTextView) } } private fun setupProxyTypeSettings() { if (appPreferences.proxyType == null) { appPreferences.proxyType = resources.getString(R.string.nc_no_proxy) } binding.settingsProxyChoice.setText(appPreferences.proxyType) binding.settingsProxyChoice.setSimpleItems(R.array.proxy_type_descriptions) binding.settingsProxyChoice.setOnItemClickListener { _, _, position, _ -> val entryVal = resources.getStringArray(R.array.proxy_type_descriptions)[position] appPreferences.proxyType = entryVal } binding.settingsProxyHostEdit.setText(appPreferences.proxyHost) binding.settingsProxyHostEdit.setOnFocusChangeListener { _, hasFocus -> if (!hasFocus) { appPreferences.proxyHost = binding.settingsProxyHostEdit.text.toString() } } binding.settingsProxyPortEdit.setText(appPreferences.proxyPort) binding.settingsProxyPortEdit.setOnFocusChangeListener { _, hasFocus -> if (!hasFocus) { appPreferences.proxyPort = binding.settingsProxyPortEdit.text.toString() } } binding.settingsProxyUsernameEdit.setText(appPreferences.proxyUsername) binding.settingsProxyUsernameEdit.setOnFocusChangeListener { _, hasFocus -> if (!hasFocus) { appPreferences.proxyUsername = binding.settingsProxyUsernameEdit.text.toString() } } binding.settingsProxyPasswordEdit.setText(appPreferences.proxyPassword) binding.settingsProxyPasswordEdit.setOnFocusChangeListener { _, hasFocus -> if (!hasFocus) { appPreferences.proxyPassword = binding.settingsProxyPasswordEdit.text.toString() } } if (((context.resources.getString(R.string.nc_no_proxy)) == appPreferences.proxyType) || appPreferences.proxyType == null ) { hideProxySettings() } else { showProxySettings() } } private fun setupProxyCredentialSettings() { if (appPreferences.proxyCredentials) { showProxyCredentials() } else { hideProxyCredentials() } } private fun setupMessageView() { if (ApplicationWideMessageHolder.getInstance().messageType != null) { when (ApplicationWideMessageHolder.getInstance().messageType) { ApplicationWideMessageHolder.MessageType.ACCOUNT_UPDATED_NOT_ADDED -> { binding.messageText.let { viewThemeUtils.platform.colorTextView(it, ColorRole.PRIMARY) it.text = resources!!.getString(R.string.nc_settings_account_updated) binding.messageText.visibility = View.VISIBLE } } ApplicationWideMessageHolder.MessageType.SERVER_WITHOUT_TALK -> { binding.messageText.let { it.setTextColor(resources!!.getColor(R.color.nc_darkRed, null)) it.text = resources!!.getString(R.string.nc_settings_wrong_account) binding.messageText.visibility = View.VISIBLE viewThemeUtils.platform.colorTextView(it, ColorRole.PRIMARY) it.text = resources!!.getString(R.string.nc_Server_account_imported) binding.messageText.visibility = View.VISIBLE } } ApplicationWideMessageHolder.MessageType.ACCOUNT_WAS_IMPORTED -> { binding.messageText.let { viewThemeUtils.platform.colorTextView(it, ColorRole.PRIMARY) it.text = resources!!.getString(R.string.nc_Server_account_imported) binding.messageText.visibility = View.VISIBLE } } ApplicationWideMessageHolder.MessageType.FAILED_TO_IMPORT_ACCOUNT -> { binding.messageText.let { it.setTextColor(resources!!.getColor(R.color.nc_darkRed, null)) it.text = resources!!.getString(R.string.nc_server_failed_to_import_account) binding.messageText.visibility = View.VISIBLE } } else -> binding.messageText.visibility = View.GONE } ApplicationWideMessageHolder.getInstance().messageType = null binding.messageText.animate() ?.translationY(0f) ?.alpha(0.0f) ?.setDuration(DURATION) ?.setStartDelay(START_DELAY) ?.setListener(object : AnimatorListenerAdapter() { override fun onAnimationEnd(animation: Animator) { super.onAnimationEnd(animation) binding.messageText.visibility = View.GONE } }) } else { binding.messageText.visibility = View.GONE } } private fun setupProfileQueryDisposable() { profileQueryDisposable = ncApi.getUserProfile( credentials, ApiUtils.getUrlForUserProfile(currentUser!!.baseUrl) ) .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe( { userProfileOverall: UserProfileOverall -> var displayName: String? = null if (!TextUtils.isEmpty( userProfileOverall.ocs!!.data!!.displayName ) ) { displayName = userProfileOverall.ocs!!.data!!.displayName } else if (!TextUtils.isEmpty( userProfileOverall.ocs!!.data!!.displayNameAlt ) ) { displayName = userProfileOverall.ocs!!.data!!.displayNameAlt } if ((!TextUtils.isEmpty(displayName) && !(displayName == currentUser!!.displayName))) { currentUser!!.displayName = displayName userManager.updateOrCreateUser(currentUser!!) binding.nameText.text = currentUser!!.displayName } }, { dispose(profileQueryDisposable) }, { dispose(profileQueryDisposable) } ) } private fun setupServerAgeWarning() { when { CapabilitiesUtilNew.isServerEOL(currentUser!!.capabilities) -> { binding.serverAgeWarningText.setTextColor(ContextCompat.getColor((context), R.color.nc_darkRed)) binding.serverAgeWarningText.setText(R.string.nc_settings_server_eol) binding.serverAgeWarningIcon.setColorFilter( ContextCompat.getColor((context), R.color.nc_darkRed), PorterDuff.Mode.SRC_IN ) } CapabilitiesUtilNew.isServerAlmostEOL(currentUser!!) -> { binding.serverAgeWarningText.setTextColor( ContextCompat.getColor((context), R.color.nc_darkYellow) ) binding.serverAgeWarningText.setText(R.string.nc_settings_server_almost_eol) binding.serverAgeWarningIcon.setColorFilter( ContextCompat.getColor((context), R.color.nc_darkYellow), PorterDuff.Mode.SRC_IN ) } else -> { binding.serverAgeWarningTextCard.visibility = View.GONE } } } private fun setupCheckables() { binding.settingsScreenSecuritySwitch.isChecked = appPreferences.isScreenSecured if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { binding.settingsIncognitoKeyboardSwitch.isChecked = appPreferences.isKeyboardIncognito } else { binding.settingsIncognitoKeyboardSwitch.visibility = View.GONE } if (CapabilitiesUtilNew.isReadStatusAvailable(currentUser!!)) { binding.settingsReadPrivacySwitch.isChecked = !CapabilitiesUtilNew.isReadStatusPrivate(currentUser!!) } else { binding.settingsReadPrivacy.visibility = View.GONE } setupTypingStatusSetting() binding.settingsPhoneBookIntegrationSwitch.isChecked = appPreferences.isPhoneBookIntegrationEnabled binding.settingsProxyUseCredentialsSwitch.isChecked = appPreferences.proxyCredentials binding.settingsProxyUseCredentials.setOnClickListener { val isChecked = binding.settingsProxyUseCredentialsSwitch.isChecked binding.settingsProxyUseCredentialsSwitch.isChecked = !isChecked appPreferences.setProxyNeedsCredentials(!isChecked) } binding.settingsScreenLockSwitch.isChecked = appPreferences.isScreenLocked binding.settingsScreenLock.setOnClickListener { val isChecked = binding.settingsScreenLockSwitch.isChecked binding.settingsScreenLockSwitch.isChecked = !isChecked appPreferences.setScreenLock(!isChecked) } binding.settingsReadPrivacy.setOnClickListener { val isChecked = binding.settingsReadPrivacySwitch.isChecked binding.settingsReadPrivacySwitch.isChecked = !isChecked appPreferences.setReadPrivacy(!isChecked) } binding.settingsIncognitoKeyboard.setOnClickListener { val isChecked = binding.settingsIncognitoKeyboardSwitch.isChecked binding.settingsIncognitoKeyboardSwitch.isChecked = !isChecked appPreferences.setIncognitoKeyboard(!isChecked) } binding.settingsPhoneBookIntegration.setOnClickListener { val isChecked = binding.settingsPhoneBookIntegrationSwitch.isChecked binding.settingsPhoneBookIntegrationSwitch.isChecked = !isChecked appPreferences.setPhoneBookIntegration(!isChecked) if (!isChecked) { if (checkPermission(this@SettingsActivity, (context))) { checkForPhoneNumber() } } else { deleteAll() } } binding.settingsScreenSecurity.setOnClickListener { val isChecked = binding.settingsScreenSecuritySwitch.isChecked binding.settingsScreenSecuritySwitch.isChecked = !isChecked appPreferences.setScreenSecurity(!isChecked) } binding.settingsTypingStatus.setOnClickListener { val isChecked = binding.settingsTypingStatusSwitch.isChecked binding.settingsTypingStatusSwitch.isChecked = !isChecked appPreferences.setTypingStatus(!isChecked) } } private fun setupTypingStatusSetting() { if (currentUser!!.externalSignalingServer?.externalSignalingServer?.isNotEmpty() == true) { binding.settingsTypingStatusOnlyWithHpb.visibility = View.GONE Log.i(TAG, "Typing Status Available: ${CapabilitiesUtilNew.isTypingStatusAvailable(currentUser!!)}") if (CapabilitiesUtilNew.isTypingStatusAvailable(currentUser!!)) { binding.settingsTypingStatusSwitch.isChecked = !CapabilitiesUtilNew.isTypingStatusPrivate(currentUser!!) } else { binding.settingsTypingStatus.visibility = View.GONE } } else { Log.i(TAG, "Typing Status not Available") binding.settingsTypingStatusSwitch.isChecked = false binding.settingsTypingStatusOnlyWithHpb.visibility = View.VISIBLE binding.settingsTypingStatus.isEnabled = false binding.settingsTypingStatusOnlyWithHpb.alpha = DISABLED_ALPHA binding.settingsTypingStatus.alpha = DISABLED_ALPHA } } private fun setupScreenLockSetting() { val keyguardManager = context.getSystemService(Context.KEYGUARD_SERVICE) as KeyguardManager if (keyguardManager.isKeyguardSecure) { binding.settingsScreenLock.isEnabled = true binding.settingsScreenLockTimeout.isEnabled = true binding.settingsScreenLockSwitch.isChecked = appPreferences.isScreenLocked binding.settingsScreenLockTimeoutLayoutDropdown.isEnabled = appPreferences.isScreenLocked if (appPreferences.isScreenLocked) { binding.settingsScreenLockTimeout.alpha = ENABLED_ALPHA } else { binding.settingsScreenLockTimeout.alpha = DISABLED_ALPHA } binding.settingsScreenLock.alpha = ENABLED_ALPHA } else { binding.settingsScreenLock.isEnabled = false binding.settingsScreenLockTimeoutLayoutDropdown.isEnabled = false appPreferences.removeScreenLock() appPreferences.removeScreenLockTimeout() binding.settingsScreenLockSwitch.isChecked = false binding.settingsScreenLock.alpha = DISABLED_ALPHA binding.settingsScreenLockTimeout.alpha = DISABLED_ALPHA } } public override fun onDestroy() { // appPreferences.unregisterProxyTypeListener(proxyTypeChangeListener) // appPreferences.unregisterProxyCredentialsListener(proxyCredentialsChangeListener) // appPreferences.unregisterScreenSecurityListener(screenSecurityChangeListener) // appPreferences.unregisterScreenLockListener(screenLockChangeListener) // appPreferences.unregisterScreenLockTimeoutListener(screenLockTimeoutChangeListener) // appPreferences.unregisterThemeChangeListener(themeChangeListener) // appPreferences.unregisterReadPrivacyChangeListener(readPrivacyChangeListener) // appPreferences.unregisterTypingStatusChangeListener(typingStatusChangeListener) // appPreferences.unregisterPhoneBookIntegrationChangeListener(phoneBookIntegrationChangeListener) super.onDestroy() } private fun hideProxySettings() { appPreferences.removeProxyHost() appPreferences.removeProxyPort() appPreferences.removeProxyCredentials() appPreferences.removeProxyUsername() appPreferences.removeProxyPassword() binding.settingsProxyHostLayout.visibility = View.GONE binding.settingsProxyPortLayout.visibility = View.GONE binding.settingsProxyUseCredentials.visibility = View.GONE hideProxyCredentials() } private fun showProxySettings() { binding.settingsProxyHostLayout.visibility = View.VISIBLE binding.settingsProxyPortLayout.visibility = View.VISIBLE binding.settingsProxyUseCredentials.visibility = View.VISIBLE if (binding.settingsProxyUseCredentialsSwitch.isChecked) showProxyCredentials() } private fun showProxyCredentials() { binding.settingsProxyUsernameLayout.visibility = View.VISIBLE binding.settingsProxyPasswordLayout.visibility = View.VISIBLE } private fun hideProxyCredentials() { appPreferences.removeProxyUsername() appPreferences.removeProxyPassword() binding.settingsProxyUsernameLayout.visibility = View.GONE binding.settingsProxyPasswordLayout.visibility = View.GONE } private fun dispose(disposable: Disposable?) { if (disposable != null && !disposable.isDisposed) { disposable.dispose() } else if (disposable == null) { disposeProfileQueryDisposable() disposeDbQueryDisposable() } } private fun disposeDbQueryDisposable() { if (dbQueryDisposable != null && !dbQueryDisposable!!.isDisposed) { dbQueryDisposable!!.dispose() dbQueryDisposable = null } else if (dbQueryDisposable != null) { dbQueryDisposable = null } } private fun disposeProfileQueryDisposable() { if (profileQueryDisposable != null && !profileQueryDisposable!!.isDisposed) { profileQueryDisposable!!.dispose() profileQueryDisposable = null } else if (profileQueryDisposable != null) { profileQueryDisposable = null } } override fun onRequestPermissionsResult(requestCode: Int, permissions: Array, grantResults: IntArray) { super.onRequestPermissionsResult(requestCode, permissions, grantResults) if (requestCode == ContactAddressBookWorker.REQUEST_PERMISSION && grantResults.isNotEmpty() && grantResults[0] == PackageManager.PERMISSION_GRANTED ) { WorkManager .getInstance(this) .enqueue(OneTimeWorkRequest.Builder(ContactAddressBookWorker::class.java).build()) checkForPhoneNumber() } else { appPreferences.setPhoneBookIntegration(false) binding.settingsPhoneBookIntegrationSwitch.isChecked = appPreferences.isPhoneBookIntegrationEnabled Snackbar.make( binding.root, context.resources.getString(R.string.no_phone_book_integration_due_to_permissions), Snackbar.LENGTH_LONG ).show() } } private fun observeScreenLock() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.isScreenLocked screenLockFlow.collect { newBoolean -> if (newBoolean != state) { state = newBoolean binding.settingsScreenLockTimeout.isEnabled = newBoolean if (newBoolean) { binding.settingsScreenLockTimeout.alpha = ENABLED_ALPHA } else { binding.settingsScreenLockTimeout.alpha = DISABLED_ALPHA } } } } } private fun observeScreenSecurity() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.isScreenSecured screenSecurityFlow.collect { newBoolean -> if (newBoolean != state) { state = newBoolean if (newBoolean) { window.addFlags(WindowManager.LayoutParams.FLAG_SECURE) } else { window.clearFlags(WindowManager.LayoutParams.FLAG_SECURE) } } } } } private fun observeProxyCredential() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.proxyCredentials proxyCredentialFlow.collect { newBoolean -> if (newBoolean != state) { state = newBoolean if (newBoolean) { showProxyCredentials() } else { hideProxyCredentials() } } } } } private fun observeProxyType() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.proxyType proxyTypeFlow.collect { newString -> if (newString != state) { state = newString if (((context.resources.getString(R.string.nc_no_proxy)) == newString) || newString.isEmpty()) { hideProxySettings() } else { when (newString) { "HTTP" -> { binding.settingsProxyPortEdit.setText(getString(R.string.nc_settings_http_value)) appPreferences.proxyPort = getString(R.string.nc_settings_http_value) } "DIRECT" -> { binding.settingsProxyPortEdit.setText(getString(R.string.nc_settings_direct_value)) appPreferences.proxyPort = getString(R.string.nc_settings_direct_value) } "SOCKS" -> { binding.settingsProxyPortEdit.setText(getString(R.string.nc_settings_socks_value)) appPreferences.proxyPort = getString(R.string.nc_settings_socks_value) } else -> { } } showProxySettings() } } } } } private fun observeTheme() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.theme themeFlow.collect { newString -> if (newString != state) { state = newString setAppTheme(newString) } } } } private fun checkForPhoneNumber() { ncApi.getUserData( ApiUtils.getCredentials(currentUser!!.username, currentUser!!.token), ApiUtils.getUrlForUserProfile(currentUser!!.baseUrl) ).subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(object : Observer { override fun onSubscribe(d: Disposable) { // unused atm } override fun onNext(userProfileOverall: UserProfileOverall) { if (userProfileOverall.ocs!!.data!!.phone?.isEmpty() == true) { askForPhoneNumber() } else { Log.d(TAG, "phone number already set") } } override fun onError(e: Throwable) { // unused atm } override fun onComplete() { // unused atm } }) } private fun askForPhoneNumber() { val phoneNumberLayoutWrapper = LinearLayout(context) phoneNumberLayoutWrapper.orientation = LinearLayout.VERTICAL phoneNumberLayoutWrapper.setPadding(PHONE_NUMBER_SIDE_PADDING, 0, PHONE_NUMBER_SIDE_PADDING, 0) val phoneNumberInputLayout = TextInputLayout(ContextThemeWrapper(this, R.style.TextInputLayoutTheme)) val phoneNumberField = EditText(context) phoneNumberInputLayout.setHelperTextColor( ColorStateList.valueOf(resources!!.getColor(R.color.nc_darkRed, null)) ) phoneNumberField.inputType = InputType.TYPE_CLASS_PHONE phoneNumberField.setText("+") phoneNumberField.addTextChangedListener(object : TextWatcher { override fun afterTextChanged(s: Editable) { // unused atm } override fun beforeTextChanged(s: CharSequence, start: Int, count: Int, after: Int) { // unused atm } override fun onTextChanged(s: CharSequence, start: Int, before: Int, count: Int) { phoneNumberInputLayout.helperText = "" } }) phoneNumberInputLayout.addView(phoneNumberField) phoneNumberLayoutWrapper.addView(phoneNumberInputLayout) val dialogBuilder = MaterialAlertDialogBuilder(phoneNumberInputLayout.context) .setTitle(R.string.nc_settings_phone_book_integration_phone_number_dialog_title) .setMessage(R.string.nc_settings_phone_book_integration_phone_number_dialog_description) .setView(phoneNumberLayoutWrapper) .setPositiveButton(context.resources.getString(R.string.nc_common_set), null) .setNegativeButton(context.resources.getString(R.string.nc_common_skip), null) viewThemeUtils.dialog.colorMaterialAlertDialogBackground(phoneNumberInputLayout.context, dialogBuilder) val dialog = dialogBuilder.create() dialog.setOnShowListener(object : OnShowListener { override fun onShow(dialogInterface: DialogInterface) { val button = dialog.getButton(AlertDialog.BUTTON_POSITIVE) button.setOnClickListener(object : View.OnClickListener { override fun onClick(view: View) { setPhoneNumber(phoneNumberInputLayout, dialog) } }) } }) dialog.show() viewThemeUtils.platform.colorTextButtons( dialog.getButton(AlertDialog.BUTTON_POSITIVE), dialog.getButton(AlertDialog.BUTTON_NEGATIVE) ) } private fun setPhoneNumber(textInputLayout: TextInputLayout, dialog: AlertDialog) { val phoneNumber = textInputLayout.editText!!.text.toString() ncApi.setUserData( ApiUtils.getCredentials(currentUser!!.username, currentUser!!.token), ApiUtils.getUrlForUserData(currentUser!!.baseUrl, currentUser!!.userId), "phone", phoneNumber ).subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(object : Observer { override fun onSubscribe(d: Disposable) { // unused atm } override fun onNext(genericOverall: GenericOverall) { when (val statusCode = genericOverall.ocs?.meta?.statusCode) { HTTP_CODE_OK -> { dialog.dismiss() Snackbar.make( binding.root, context.resources.getString( R.string.nc_settings_phone_book_integration_phone_number_dialog_success ), Snackbar.LENGTH_LONG ).show() } else -> { textInputLayout.helperText = context.resources.getString( R.string.nc_settings_phone_book_integration_phone_number_dialog_invalid ) Log.d(TAG, "failed to set phoneNumber. statusCode=$statusCode") } } } override fun onError(e: Throwable) { textInputLayout.helperText = context.resources.getString( R.string.nc_settings_phone_book_integration_phone_number_dialog_invalid ) Log.e(TAG, "setPhoneNumber error", e) } override fun onComplete() { // unused atm } }) } private fun observeReadPrivacy() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.readPrivacy readPrivacyFlow.collect { newBoolean -> if (state != newBoolean) { state = newBoolean val booleanValue = if (newBoolean) "0" else "1" val json = "{\"key\": \"read_status_privacy\", \"value\" : $booleanValue}" ncApi.setReadStatusPrivacy( ApiUtils.getCredentials(currentUser!!.username, currentUser!!.token), ApiUtils.getUrlForUserSettings(currentUser!!.baseUrl), json.toRequestBody("application/json".toMediaTypeOrNull()) ) .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(object : Observer { override fun onSubscribe(d: Disposable) { // unused atm } override fun onNext(genericOverall: GenericOverall) { // unused atm } override fun onError(e: Throwable) { appPreferences.setReadPrivacy(!newBoolean) binding.settingsReadPrivacySwitch.isChecked = !newBoolean } override fun onComplete() { // unused atm } }) } } } } private fun observeTypingStatus() { CoroutineScope(Dispatchers.Main).launch { var state = appPreferences.typingStatus typingStatusFlow.collect { newBoolean -> if (state != newBoolean) { state = newBoolean val booleanValue = if (newBoolean) "0" else "1" val json = "{\"key\": \"typing_privacy\", \"value\" : $booleanValue}" ncApi.setTypingStatusPrivacy( ApiUtils.getCredentials(currentUser!!.username, currentUser!!.token), ApiUtils.getUrlForUserSettings(currentUser!!.baseUrl), json.toRequestBody("application/json".toMediaTypeOrNull()) ) .subscribeOn(Schedulers.io()) .observeOn(AndroidSchedulers.mainThread()) .subscribe(object : Observer { override fun onSubscribe(d: Disposable) { // unused atm } override fun onNext(genericOverall: GenericOverall) { loadCapabilitiesAndUpdateSettings() Log.i(TAG, "onNext called typing status set") } override fun onError(e: Throwable) { appPreferences.typingStatus = !newBoolean binding.settingsTypingStatusSwitch.isChecked = !newBoolean } override fun onComplete() { // unused atm } }) } } } } companion object { private val TAG = SettingsActivity::class.java.simpleName private const val DURATION: Long = 2500 private const val START_DELAY: Long = 5000 private const val DISABLED_ALPHA: Float = 0.38f private const val ENABLED_ALPHA: Float = 1.0f private const val HTTP_CODE_OK: Int = 200 private const val PHONE_NUMBER_SIDE_PADDING: Int = 50 } }