/* * Copyright (C) 2020 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package com.android.systemui.controls.ui import android.animation.Animator import android.animation.AnimatorListenerAdapter import android.animation.ObjectAnimator import android.app.Activity import android.app.ActivityOptions import android.app.Dialog import android.app.PendingIntent import android.content.ComponentName import android.content.Context import android.content.Intent import android.content.pm.PackageManager import android.graphics.drawable.Drawable import android.graphics.drawable.LayerDrawable import android.os.Trace import android.service.controls.Control import android.service.controls.ControlsProviderService import android.util.Log import android.view.ContextThemeWrapper import android.view.Gravity import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import android.view.animation.AccelerateInterpolator import android.view.animation.DecelerateInterpolator import android.widget.AdapterView import android.widget.ArrayAdapter import android.widget.BaseAdapter import android.widget.FrameLayout import android.widget.ImageView import android.widget.LinearLayout import android.widget.ListPopupWindow import android.widget.Space import android.widget.TextView import androidx.annotation.VisibleForTesting import com.android.systemui.Dumpable import com.android.systemui.R import com.android.systemui.controls.ControlsMetricsLogger import com.android.systemui.controls.ControlsServiceInfo import com.android.systemui.controls.CustomIconCache import com.android.systemui.controls.controller.ControlsController import com.android.systemui.controls.controller.StructureInfo import com.android.systemui.controls.controller.StructureInfo.Companion.EMPTY_COMPONENT import com.android.systemui.controls.controller.StructureInfo.Companion.EMPTY_STRUCTURE import com.android.systemui.controls.management.ControlAdapter import com.android.systemui.controls.management.ControlsEditingActivity import com.android.systemui.controls.management.ControlsFavoritingActivity import com.android.systemui.controls.management.ControlsListingController import com.android.systemui.controls.management.ControlsProviderSelectorActivity import com.android.systemui.controls.panels.AuthorizedPanelsRepository import com.android.systemui.controls.panels.SelectedComponentRepository import com.android.systemui.controls.settings.ControlsSettingsRepository import com.android.systemui.dagger.SysUISingleton import com.android.systemui.dagger.qualifiers.Background import com.android.systemui.dagger.qualifiers.Main import com.android.systemui.dump.DumpManager import com.android.systemui.flags.FeatureFlags import com.android.systemui.plugins.ActivityStarter import com.android.systemui.settings.UserTracker import com.android.systemui.statusbar.policy.KeyguardStateController import com.android.systemui.util.asIndenting import com.android.systemui.util.concurrency.DelayableExecutor import com.android.systemui.util.indentIfPossible import com.android.wm.shell.taskview.TaskViewFactory import dagger.Lazy import java.io.PrintWriter import java.text.Collator import java.util.Optional import java.util.function.Consumer import javax.inject.Inject private data class ControlKey(val componentName: ComponentName, val controlId: String) @SysUISingleton class ControlsUiControllerImpl @Inject constructor ( val controlsController: Lazy, val context: Context, private val packageManager: PackageManager, @Main val uiExecutor: DelayableExecutor, @Background val bgExecutor: DelayableExecutor, val controlsListingController: Lazy, private val controlActionCoordinator: ControlActionCoordinator, private val activityStarter: ActivityStarter, private val iconCache: CustomIconCache, private val controlsMetricsLogger: ControlsMetricsLogger, private val keyguardStateController: KeyguardStateController, private val userTracker: UserTracker, private val taskViewFactory: Optional, private val controlsSettingsRepository: ControlsSettingsRepository, private val authorizedPanelsRepository: AuthorizedPanelsRepository, private val selectedComponentRepository: SelectedComponentRepository, private val featureFlags: FeatureFlags, private val dialogsFactory: ControlsDialogsFactory, dumpManager: DumpManager ) : ControlsUiController, Dumpable { companion object { private const val FADE_IN_MILLIS = 200L private const val OPEN_APP_ID = 0L private const val ADD_CONTROLS_ID = 1L private const val ADD_APP_ID = 2L private const val EDIT_CONTROLS_ID = 3L private const val REMOVE_APP_ID = 4L } private var selectedItem: SelectedItem = SelectedItem.EMPTY_SELECTION private var selectionItem: SelectionItem? = null private lateinit var allStructures: List private val controlsById = mutableMapOf() private val controlViewsById = mutableMapOf() private lateinit var parent: ViewGroup private var popup: ListPopupWindow? = null private var hidden = true private lateinit var onDismiss: Runnable private val popupThemedContext = ContextThemeWrapper(context, R.style.Control_ListPopupWindow) private var retainCache = false private var lastSelections = emptyList() private var taskViewController: PanelTaskViewController? = null private val collator = Collator.getInstance(context.resources.configuration.locales[0]) private val localeComparator = compareBy(collator) { it.getTitle() } private var openAppIntent: Intent? = null private var overflowMenuAdapter: BaseAdapter? = null private var removeAppDialog: Dialog? = null private val onSeedingComplete = Consumer { accepted -> if (accepted) { selectedItem = controlsController.get().getFavorites().maxByOrNull { it.controls.size }?.let { SelectedItem.StructureItem(it) } ?: SelectedItem.EMPTY_SELECTION updatePreferences(selectedItem) } reload(parent) } private lateinit var activityContext: Context private lateinit var listingCallback: ControlsListingController.ControlsListingCallback override val isShowing: Boolean get() = !hidden init { dumpManager.registerDumpable(javaClass.name, this) } private fun createCallback( onResult: (List) -> Unit ): ControlsListingController.ControlsListingCallback { return object : ControlsListingController.ControlsListingCallback { override fun onServicesUpdated(serviceInfos: List) { val authorizedPanels = authorizedPanelsRepository.getAuthorizedPanels() val lastItems = serviceInfos.map { val uid = it.serviceInfo.applicationInfo.uid SelectionItem( it.loadLabel(), "", it.loadIcon(), it.componentName, uid, if (it.componentName.packageName in authorizedPanels) { it.panelActivity } else { null } ) } uiExecutor.execute { parent.removeAllViews() if (lastItems.size > 0) { onResult(lastItems) } } } } } override fun resolveActivity(): Class<*> { val allStructures = controlsController.get().getFavorites() val selected = getPreferredSelectedItem(allStructures) val anyPanels = controlsListingController.get().getCurrentServices() .any { it.panelActivity != null } return if (controlsController.get().addSeedingFavoritesCallback(onSeedingComplete)) { ControlsActivity::class.java } else if (!selected.hasControls && allStructures.size <= 1 && !anyPanels) { ControlsProviderSelectorActivity::class.java } else { ControlsActivity::class.java } } override fun show( parent: ViewGroup, onDismiss: Runnable, activityContext: Context ) { Log.d(ControlsUiController.TAG, "show()") Trace.instant(Trace.TRACE_TAG_APP, "ControlsUiControllerImpl#show") this.parent = parent this.onDismiss = onDismiss this.activityContext = activityContext this.openAppIntent = null this.overflowMenuAdapter = null hidden = false retainCache = false selectionItem = null controlActionCoordinator.activityContext = activityContext allStructures = controlsController.get().getFavorites() selectedItem = getPreferredSelectedItem(allStructures) if (controlsController.get().addSeedingFavoritesCallback(onSeedingComplete)) { listingCallback = createCallback(::showSeedingView) } else if ( selectedItem !is SelectedItem.PanelItem && !selectedItem.hasControls && allStructures.size <= 1 ) { // only show initial view if there are really no favorites across any structure listingCallback = createCallback(::initialView) } else { val selected = selectedItem if (selected is SelectedItem.StructureItem) { selected.structure.controls.map { ControlWithState(selected.structure.componentName, it, null) }.associateByTo(controlsById) { ControlKey(selected.structure.componentName, it.ci.controlId) } controlsController.get().subscribeToFavorites(selected.structure) } else { controlsController.get().bindComponentForPanel(selected.componentName) } listingCallback = createCallback(::showControlsView) } controlsListingController.get().addCallback(listingCallback) } private fun initialView(items: List) { if (items.any { it.isPanel }) { // We have at least a panel, so we'll end up showing that. showControlsView(items) } else { showInitialSetupView(items) } } private fun reload(parent: ViewGroup, dismissTaskView: Boolean = true) { if (hidden) return controlsListingController.get().removeCallback(listingCallback) controlsController.get().unsubscribe() taskViewController?.removeTask() taskViewController = null val fadeAnim = ObjectAnimator.ofFloat(parent, "alpha", 1.0f, 0.0f) fadeAnim.setInterpolator(AccelerateInterpolator(1.0f)) fadeAnim.setDuration(FADE_IN_MILLIS) fadeAnim.addListener(object : AnimatorListenerAdapter() { override fun onAnimationEnd(animation: Animator) { controlViewsById.clear() controlsById.clear() show(parent, onDismiss, activityContext) val showAnim = ObjectAnimator.ofFloat(parent, "alpha", 0.0f, 1.0f) showAnim.setInterpolator(DecelerateInterpolator(1.0f)) showAnim.setDuration(FADE_IN_MILLIS) showAnim.start() } }) fadeAnim.start() } private fun showSeedingView(items: List) { val inflater = LayoutInflater.from(context) inflater.inflate(R.layout.controls_no_favorites, parent, true) val subtitle = parent.requireViewById(R.id.controls_subtitle) subtitle.setText(context.resources.getString(R.string.controls_seeding_in_progress)) } private fun showInitialSetupView(items: List) { startProviderSelectorActivity() onDismiss.run() } private fun startFavoritingActivity(si: StructureInfo) { startTargetedActivity(si, ControlsFavoritingActivity::class.java) } private fun startEditingActivity(si: StructureInfo) { startTargetedActivity(si, ControlsEditingActivity::class.java) } private fun startDefaultActivity() { openAppIntent?.let { startActivity(it, animateExtra = false) } } @VisibleForTesting internal fun startRemovingApp(componentName: ComponentName, appName: CharSequence) { activityStarter.dismissKeyguardThenExecute({ showAppRemovalDialog(componentName, appName) true }, null, true) } private fun showAppRemovalDialog(componentName: ComponentName, appName: CharSequence) { removeAppDialog?.cancel() removeAppDialog = dialogsFactory.createRemoveAppDialog(context, appName) { shouldRemove -> if (!shouldRemove || !controlsController.get().removeFavorites(componentName)) { return@createRemoveAppDialog } if (selectedComponentRepository.getSelectedComponent()?.componentName == componentName) { selectedComponentRepository.removeSelectedComponent() } val selectedItem = getPreferredSelectedItem(controlsController.get().getFavorites()) if (selectedItem == SelectedItem.EMPTY_SELECTION) { // User removed the last panel. In this case we start app selection flow and don't // want to auto-add it again selectedComponentRepository.setShouldAddDefaultComponent(false) } reload(parent) }.apply { show() } } private fun startTargetedActivity(si: StructureInfo, klazz: Class<*>) { val i = Intent(activityContext, klazz) putIntentExtras(i, si) startActivity(i) retainCache = true } private fun putIntentExtras(intent: Intent, si: StructureInfo) { intent.apply { putExtra(ControlsFavoritingActivity.EXTRA_APP, controlsListingController.get().getAppLabel(si.componentName)) putExtra(ControlsFavoritingActivity.EXTRA_STRUCTURE, si.structure) putExtra(Intent.EXTRA_COMPONENT_NAME, si.componentName) } } private fun startProviderSelectorActivity() { val i = Intent(activityContext, ControlsProviderSelectorActivity::class.java) i.putExtra(ControlsProviderSelectorActivity.BACK_SHOULD_EXIT, true) startActivity(i) } private fun startActivity(intent: Intent, animateExtra: Boolean = true) { // Force animations when transitioning from a dialog to an activity if (animateExtra) { intent.putExtra(ControlsUiController.EXTRA_ANIMATE, true) } if (keyguardStateController.isShowing()) { activityStarter.postStartActivityDismissingKeyguard(intent, 0 /* delay */) } else { activityContext.startActivity( intent, ActivityOptions.makeSceneTransitionAnimation(activityContext as Activity).toBundle() ) } } private fun showControlsView(items: List) { controlViewsById.clear() val (panels, structures) = items.partition { it.isPanel } val panelComponents = panels.map { it.componentName }.toSet() val itemsByComponent = structures.associateBy { it.componentName } .filterNot { it.key in panelComponents } val panelsAndStructures = mutableListOf() allStructures.mapNotNullTo(panelsAndStructures) { itemsByComponent.get(it.componentName)?.copy(structure = it.structure) } panelsAndStructures.addAll(panels) panelsAndStructures.sortWith(localeComparator) lastSelections = panelsAndStructures val selectionItem = findSelectionItem(selectedItem, panelsAndStructures) ?: if (panels.isNotEmpty()) { // If we couldn't find a good selected item, but there's at least one panel, // show a panel. panels[0] } else { items[0] } maybeUpdateSelectedItem(selectionItem) createControlsSpaceFrame() if (taskViewFactory.isPresent && selectionItem.isPanel) { createPanelView(selectionItem.panelComponentName!!) } else if (!selectionItem.isPanel) { controlsMetricsLogger .refreshBegin(selectionItem.uid, !keyguardStateController.isUnlocked()) createListView(selectionItem) } else { Log.w(ControlsUiController.TAG, "Not TaskViewFactory to display panel $selectionItem") } this.selectionItem = selectionItem bgExecutor.execute { val intent = Intent(Intent.ACTION_MAIN) .addCategory(Intent.CATEGORY_LAUNCHER) .setPackage(selectionItem.componentName.packageName) .addFlags(Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_RESET_TASK_IF_NEEDED) val intents = packageManager .queryIntentActivities(intent, PackageManager.ResolveInfoFlags.of(0L)) intents.firstOrNull { it.activityInfo.exported }?.let { resolved -> intent.setPackage(null) intent.setComponent(resolved.activityInfo.componentName) openAppIntent = intent parent.post { // This will call show on the PopupWindow in the same thread, so make sure this // happens in the view thread. overflowMenuAdapter?.notifyDataSetChanged() } } } createDropDown(panelsAndStructures, selectionItem) val currentApps = panelsAndStructures.map { it.componentName }.toSet() val allApps = controlsListingController.get() .getCurrentServices().map { it.componentName }.toSet() createMenu( selectionItem = selectionItem, extraApps = (allApps - currentApps).isNotEmpty(), ) } private fun createPanelView(componentName: ComponentName) { val setting = controlsSettingsRepository .allowActionOnTrivialControlsInLockscreen.value val pendingIntent = PendingIntent.getActivityAsUser( context, 0, Intent() .setComponent(componentName) .putExtra( ControlsProviderService.EXTRA_LOCKSCREEN_ALLOW_TRIVIAL_CONTROLS, setting ), PendingIntent.FLAG_IMMUTABLE or PendingIntent.FLAG_UPDATE_CURRENT, null, userTracker.userHandle ) parent.requireViewById(R.id.controls_scroll_view).visibility = View.GONE val container = parent.requireViewById(R.id.controls_panel) container.visibility = View.VISIBLE container.post { taskViewFactory.get().create(activityContext, uiExecutor) { taskView -> taskViewController = PanelTaskViewController( activityContext, uiExecutor, pendingIntent, taskView, onDismiss::run ).also { container.addView(taskView) it.launchTaskView() } } } } private fun createMenu(selectionItem: SelectionItem, extraApps: Boolean) { val isPanel = selectedItem is SelectedItem.PanelItem val selectedStructure = (selectedItem as? SelectedItem.StructureItem)?.structure ?: EMPTY_STRUCTURE val items = buildList { add(OverflowMenuAdapter.MenuItem( context.getText(R.string.controls_open_app), OPEN_APP_ID )) if (extraApps) { add(OverflowMenuAdapter.MenuItem( context.getText(R.string.controls_menu_add_another_app), ADD_APP_ID )) } add(OverflowMenuAdapter.MenuItem( context.getText(R.string.controls_menu_remove), REMOVE_APP_ID, )) if (!isPanel) { add(OverflowMenuAdapter.MenuItem( context.getText(R.string.controls_menu_edit), EDIT_CONTROLS_ID )) } } val adapter = OverflowMenuAdapter(context, R.layout.controls_more_item, items) { position -> getItemId(position) != OPEN_APP_ID || openAppIntent != null } val anchor = parent.requireViewById(R.id.controls_more) anchor.setOnClickListener(object : View.OnClickListener { override fun onClick(v: View) { popup = ControlsPopupMenu(popupThemedContext).apply { width = ViewGroup.LayoutParams.WRAP_CONTENT anchorView = anchor setDropDownGravity(Gravity.END) setAdapter(adapter) setOnItemClickListener(object : AdapterView.OnItemClickListener { override fun onItemClick( parent: AdapterView<*>, view: View, pos: Int, id: Long ) { when (id) { OPEN_APP_ID -> startDefaultActivity() ADD_APP_ID -> startProviderSelectorActivity() ADD_CONTROLS_ID -> startFavoritingActivity(selectedStructure) EDIT_CONTROLS_ID -> startEditingActivity(selectedStructure) REMOVE_APP_ID -> startRemovingApp( selectionItem.componentName, selectionItem.appName ) } dismiss() } }) show() listView?.post { listView?.requestAccessibilityFocus() } } } }) overflowMenuAdapter = adapter } private fun createDropDown(items: List, selected: SelectionItem) { items.forEach { RenderInfo.registerComponentIcon(it.componentName, it.icon) } val adapter = ItemAdapter(context, R.layout.controls_spinner_item).apply { add(selected) addAll(items .filter { it !== selected } .sortedBy { it.appName.toString() } ) } val iconSize = context.resources .getDimensionPixelSize(R.dimen.controls_header_app_icon_size) /* * Default spinner widget does not work with the window type required * for this dialog. Use a textView with the ListPopupWindow to achieve * a similar effect */ val spinner = parent.requireViewById(R.id.app_or_structure_spinner).apply { setText(selected.getTitle()) // override the default color on the dropdown drawable (getBackground() as LayerDrawable).getDrawable(0) .setTint(context.resources.getColor(R.color.control_spinner_dropdown, null)) selected.icon.setBounds(0, 0, iconSize, iconSize) compoundDrawablePadding = (iconSize / 2.4f).toInt() setCompoundDrawablesRelative(selected.icon, null, null, null) } val anchor = parent.requireViewById(R.id.app_or_structure_spinner) if (items.size == 1) { spinner.setBackground(null) anchor.setOnClickListener(null) anchor.isClickable = false return } else { spinner.background = parent.context.resources .getDrawable(R.drawable.control_spinner_background) } anchor.setOnClickListener(object : View.OnClickListener { override fun onClick(v: View) { popup = ControlsPopupMenu(popupThemedContext).apply { anchorView = anchor width = ViewGroup.LayoutParams.MATCH_PARENT setAdapter(adapter) setOnItemClickListener(object : AdapterView.OnItemClickListener { override fun onItemClick( parent: AdapterView<*>, view: View, pos: Int, id: Long ) { val listItem = parent.getItemAtPosition(pos) as SelectionItem this@ControlsUiControllerImpl.switchAppOrStructure(listItem) dismiss() } }) show() listView?.post { listView?.requestAccessibilityFocus() } } } }) } private fun createControlsSpaceFrame() { val inflater = LayoutInflater.from(activityContext) inflater.inflate(R.layout.controls_with_favorites, parent, true) parent.requireViewById(R.id.controls_close).apply { setOnClickListener { _: View -> onDismiss.run() } visibility = View.VISIBLE } } private fun createListView(selected: SelectionItem) { if (selectedItem !is SelectedItem.StructureItem) return val selectedStructure = (selectedItem as SelectedItem.StructureItem).structure val inflater = LayoutInflater.from(activityContext) val maxColumns = ControlAdapter.findMaxColumns(activityContext.resources) val listView = parent.requireViewById(R.id.controls_list) as ViewGroup listView.removeAllViews() var lastRow: ViewGroup = createRow(inflater, listView) selectedStructure.controls.forEach { val key = ControlKey(selectedStructure.componentName, it.controlId) controlsById.get(key)?.let { if (lastRow.getChildCount() == maxColumns) { lastRow = createRow(inflater, listView) } val baseLayout = inflater.inflate( R.layout.controls_base_item, lastRow, false) as ViewGroup lastRow.addView(baseLayout) // Use ConstraintLayout in the future... for now, manually adjust margins if (lastRow.getChildCount() == 1) { val lp = baseLayout.getLayoutParams() as ViewGroup.MarginLayoutParams lp.setMarginStart(0) } val cvh = ControlViewHolder( baseLayout, controlsController.get(), uiExecutor, bgExecutor, controlActionCoordinator, controlsMetricsLogger, selected.uid, controlsController.get().currentUserId, ) cvh.bindData(it, false /* isLocked, will be ignored on initial load */) controlViewsById.put(key, cvh) } } // add spacers if necessary to keep control size consistent val mod = selectedStructure.controls.size % maxColumns var spacersToAdd = if (mod == 0) 0 else maxColumns - mod val margin = context.resources.getDimensionPixelSize(R.dimen.control_spacing) while (spacersToAdd > 0) { val lp = LinearLayout.LayoutParams(0, 0, 1f).apply { setMarginStart(margin) } lastRow.addView(Space(context), lp) spacersToAdd-- } } override fun getPreferredSelectedItem(structures: List): SelectedItem { val preferredPanel = selectedComponentRepository.getSelectedComponent() val component = preferredPanel?.componentName ?: EMPTY_COMPONENT return if (preferredPanel?.isPanel == true) { SelectedItem.PanelItem(preferredPanel.name, component) } else { if (structures.isEmpty()) return SelectedItem.EMPTY_SELECTION SelectedItem.StructureItem(structures.firstOrNull { component == it.componentName && preferredPanel?.name == it.structure } ?: structures[0]) } } private fun updatePreferences(selectedItem: SelectedItem) { selectedComponentRepository.setSelectedComponent( SelectedComponentRepository.SelectedComponent(selectedItem) ) } private fun maybeUpdateSelectedItem(item: SelectionItem): Boolean { val newSelection = if (item.isPanel) { SelectedItem.PanelItem(item.appName, item.componentName) } else { SelectedItem.StructureItem(allStructures.firstOrNull { it.structure == item.structure && it.componentName == item.componentName } ?: EMPTY_STRUCTURE) } return if (newSelection != selectedItem ) { selectedItem = newSelection updatePreferences(selectedItem) true } else { false } } private fun switchAppOrStructure(item: SelectionItem) { if (maybeUpdateSelectedItem(item)) { reload(parent) } } override fun closeDialogs(immediately: Boolean) { if (immediately) { popup?.dismissImmediate() } else { popup?.dismiss() } popup = null controlViewsById.forEach { it.value.dismiss() } controlActionCoordinator.closeDialogs() removeAppDialog?.cancel() } override fun hide(parent: ViewGroup) { // We need to check for the parent because it's possible that we have started showing in a // different activity. In that case, make sure to only clear things associated with the // passed parent if (parent == this.parent) { Log.d(ControlsUiController.TAG, "hide()") hidden = true closeDialogs(true) controlsController.get().unsubscribe() taskViewController?.removeTask() taskViewController = null controlsById.clear() controlViewsById.clear() controlsListingController.get().removeCallback(listingCallback) if (!retainCache) RenderInfo.clearCache() } parent.removeAllViews() } override fun onRefreshState(componentName: ComponentName, controls: List) { val isLocked = !keyguardStateController.isUnlocked() controls.forEach { c -> controlsById.get(ControlKey(componentName, c.getControlId()))?.let { Log.d(ControlsUiController.TAG, "onRefreshState() for id: " + c.getControlId()) iconCache.store(componentName, c.controlId, c.customIcon) val cws = ControlWithState(componentName, it.ci, c) val key = ControlKey(componentName, c.getControlId()) controlsById.put(key, cws) controlViewsById.get(key)?.let { uiExecutor.execute { it.bindData(cws, isLocked) } } } } } override fun onActionResponse(componentName: ComponentName, controlId: String, response: Int) { val key = ControlKey(componentName, controlId) uiExecutor.execute { controlViewsById.get(key)?.actionResponse(response) } } override fun onSizeChange() { selectionItem?.let { when (selectedItem) { is SelectedItem.StructureItem -> createListView(it) is SelectedItem.PanelItem -> taskViewController?.refreshBounds() ?: reload(parent) } } ?: reload(parent) } private fun createRow(inflater: LayoutInflater, listView: ViewGroup): ViewGroup { val row = inflater.inflate(R.layout.controls_row, listView, false) as ViewGroup listView.addView(row) return row } private fun findSelectionItem(si: SelectedItem, items: List): SelectionItem? = items.firstOrNull { it.matches(si) } override fun dump(pw: PrintWriter, args: Array) { pw.println("ControlsUiControllerImpl:") pw.asIndenting().indentIfPossible { println("hidden: $hidden") println("selectedItem: $selectedItem") println("lastSelections: $lastSelections") println("setting: ${controlsSettingsRepository .allowActionOnTrivialControlsInLockscreen.value}") } } } @VisibleForTesting internal data class SelectionItem( val appName: CharSequence, val structure: CharSequence, val icon: Drawable, val componentName: ComponentName, val uid: Int, val panelComponentName: ComponentName? ) { fun getTitle() = if (structure.isEmpty()) { appName } else { structure } val isPanel: Boolean = panelComponentName != null fun matches(selectedItem: SelectedItem): Boolean { if (componentName != selectedItem.componentName) { // Not the same component so they are not the same. return false } if (isPanel || selectedItem is SelectedItem.PanelItem) { // As they have the same component, if [this.isPanel] then we may be migrating from // device controls API into panel. Want this to match, even if the selectedItem is not // a panel. We don't want to match on app name because that can change with locale. return true } // Return true if we find a structure with the correct name return structure == (selectedItem as SelectedItem.StructureItem).structure.structure } } private class ItemAdapter(parentContext: Context, val resource: Int) : ArrayAdapter(parentContext, resource) { private val layoutInflater = LayoutInflater.from(context)!! override fun getView(position: Int, convertView: View?, parent: ViewGroup): View { val item: SelectionItem = getItem(position)!! val view = convertView ?: layoutInflater.inflate(resource, parent, false) with(view.tag as? ViewHolder ?: ViewHolder(view).also { view.tag = it }) { titleView.text = item.getTitle() iconView.setImageDrawable(item.icon) } return view } private class ViewHolder(itemView: View) { val titleView: TextView = itemView.requireViewById(R.id.controls_spinner_item) val iconView: ImageView = itemView.requireViewById(R.id.app_icon) } }