243 lines
9.1 KiB
Kotlin
243 lines
9.1 KiB
Kotlin
/* 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.make(view, Snackbar.LENGTH_SHORT)
|
|
.setText(view.context.getString(R.string.browser_toolbar_url_copied_to_clipboard_snackbar))
|
|
.setAnchorView(view)
|
|
.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 (isCustomTabSession) {
|
|
DisplayToolbar.Gravity.BOTTOM
|
|
} else {
|
|
DisplayToolbar.Gravity.TOP
|
|
}
|
|
|
|
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(),
|
|
requestDesktopStateProvider = {
|
|
sessionManager.selectedSession?.desktopMode ?: false
|
|
},
|
|
readerModeStateProvider = {
|
|
sessionManager.selectedSession?.readerMode ?: false
|
|
},
|
|
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
|
|
}
|
|
}
|