/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ package org.mozilla.fenix.components.toolbar import android.view.Gravity import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import android.widget.LinearLayout import android.widget.PopupWindow import androidx.annotation.LayoutRes import androidx.appcompat.app.AppCompatActivity import androidx.core.content.ContextCompat import androidx.core.view.isVisible import com.google.android.material.snackbar.Snackbar import kotlinx.android.extensions.LayoutContainer import kotlinx.android.synthetic.main.browser_toolbar_popup_window.view.* import kotlinx.android.synthetic.main.component_browser_top_toolbar.view.* import mozilla.components.browser.domains.autocomplete.ShippedDomainsProvider import mozilla.components.browser.session.Session import mozilla.components.browser.toolbar.BrowserToolbar import mozilla.components.browser.toolbar.display.DisplayToolbar import mozilla.components.support.ktx.android.util.dpToFloat import org.mozilla.fenix.R import org.mozilla.fenix.components.FenixSnackbar import org.mozilla.fenix.customtabs.CustomTabToolbarIntegration import org.mozilla.fenix.customtabs.CustomTabToolbarMenu import org.mozilla.fenix.ext.bookmarkStorage import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.settings import org.mozilla.fenix.search.toolbar.setScrollFlagsForTopToolbar import org.mozilla.fenix.theme.ThemeManager interface BrowserToolbarViewInteractor { fun onBrowserToolbarPaste(text: String) fun onBrowserToolbarPasteAndGo(text: String) fun onBrowserToolbarClicked() fun onBrowserToolbarMenuItemTapped(item: ToolbarMenu.Item) fun onTabCounterClicked() } class BrowserToolbarView( private val container: ViewGroup, private val shouldUseBottomToolbar: Boolean, private val interactor: BrowserToolbarViewInteractor, private val customTabSession: Session? ) : LayoutContainer { override val containerView: View? get() = container private val settings = container.context.settings() @LayoutRes private val toolbarLayout = when { settings.shouldUseBottomToolbar -> R.layout.component_bottom_browser_toolbar else -> R.layout.component_browser_top_toolbar } private val layout = LayoutInflater.from(container.context) .inflate(toolbarLayout, container, true) val view: BrowserToolbar = layout .findViewById(R.id.toolbar) val toolbarIntegration: ToolbarIntegration init { val isCustomTabSession = customTabSession != null view.display.setOnUrlLongClickListener { val clipboard = view.context.components.clipboardHandler val customView = LayoutInflater.from(view.context) .inflate(R.layout.browser_toolbar_popup_window, null) val popupWindow = PopupWindow( customView, LinearLayout.LayoutParams.WRAP_CONTENT, view.context.resources.getDimensionPixelSize(R.dimen.context_menu_height), true ) val selectedSession = container.context.components.core.sessionManager.selectedSession popupWindow.elevation = view.context.resources.getDimension(R.dimen.mozac_browser_menu_elevation) customView.paste.isVisible = !clipboard.text.isNullOrEmpty() && !isCustomTabSession customView.paste_and_go.isVisible = !clipboard.text.isNullOrEmpty() && !isCustomTabSession customView.copy.setOnClickListener { popupWindow.dismiss() if (isCustomTabSession) { clipboard.text = customTabSession?.url } else { clipboard.text = selectedSession?.url } FenixSnackbar.makeWithToolbarPadding(view, Snackbar.LENGTH_SHORT) .setText(view.context.getString(R.string.browser_toolbar_url_copied_to_clipboard_snackbar)) .show() } customView.paste.setOnClickListener { popupWindow.dismiss() interactor.onBrowserToolbarPaste(clipboard.text!!) } customView.paste_and_go.setOnClickListener { popupWindow.dismiss() interactor.onBrowserToolbarPasteAndGo(clipboard.text!!) } popupWindow.showAsDropDown( view, view.context.resources.getDimensionPixelSize(R.dimen.context_menu_x_offset), 0, Gravity.START ) true } with(container.context) { val sessionManager = components.core.sessionManager view.apply { setScrollFlagsForTopToolbar() elevation = TOOLBAR_ELEVATION.dpToFloat(resources.displayMetrics) if (!isCustomTabSession) { display.setUrlBackground(getDrawable(R.drawable.search_url_background)) } display.onUrlClicked = { interactor.onBrowserToolbarClicked() false } display.progressGravity = if (shouldUseBottomToolbar) { DisplayToolbar.Gravity.TOP } else { DisplayToolbar.Gravity.BOTTOM } val primaryTextColor = ContextCompat.getColor( container.context, ThemeManager.resolveAttribute(R.attr.primaryText, container.context) ) val secondaryTextColor = ContextCompat.getColor( container.context, ThemeManager.resolveAttribute(R.attr.secondaryText, container.context) ) val separatorColor = ContextCompat.getColor( container.context, ThemeManager.resolveAttribute(R.attr.toolbarDivider, container.context) ) display.colors = display.colors.copy( text = primaryTextColor, securityIconSecure = primaryTextColor, securityIconInsecure = primaryTextColor, menu = primaryTextColor, hint = secondaryTextColor, separator = separatorColor, trackingProtection = primaryTextColor ) display.hint = context.getString(R.string.search_hint) } val menuToolbar = if (isCustomTabSession) { CustomTabToolbarMenu( this, sessionManager, customTabSession?.id, shouldReverseItems = !shouldUseBottomToolbar, onItemTapped = { interactor.onBrowserToolbarMenuItemTapped(it) } ) } else { DefaultToolbarMenu( context = this, hasAccountProblem = components.backgroundServices.accountManager.accountNeedsReauth(), shouldReverseItems = !shouldUseBottomToolbar, onItemTapped = { interactor.onBrowserToolbarMenuItemTapped(it) }, lifecycleOwner = container.context as AppCompatActivity, sessionManager = sessionManager, bookmarksStorage = bookmarkStorage ) } toolbarIntegration = if (customTabSession != null) { CustomTabToolbarIntegration( this, view, menuToolbar, customTabSession.id, isPrivate = customTabSession.private ) } else { DefaultToolbarIntegration( this, view, menuToolbar, ShippedDomainsProvider().also { it.initialize(this) }, components.core.historyStorage, components.core.sessionManager, sessionId = null, isPrivate = sessionManager.selectedSession?.private ?: false, interactor = interactor ) } } } @Suppress("UNUSED_PARAMETER") fun update(state: BrowserFragmentState) { // Intentionally leaving this as a stub for now since we don't actually want to update currently } fun expand() { if (!settings.shouldUseBottomToolbar) { layout.app_bar?.setExpanded(true) } } companion object { private const val TOOLBAR_ELEVATION = 16 } }