/* 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.search import android.content.Context import android.os.Bundle import android.view.LayoutInflater import android.view.View import android.view.ViewGroup import androidx.appcompat.app.AppCompatActivity import androidx.fragment.app.Fragment import kotlinx.android.synthetic.main.component_search.* import kotlinx.android.synthetic.main.fragment_search.* import kotlinx.android.synthetic.main.fragment_search.view.* import mozilla.components.browser.search.SearchEngine import mozilla.components.feature.qr.QrFeature import mozilla.components.feature.search.SearchUseCases import mozilla.components.feature.session.SessionUseCases import mozilla.components.support.base.feature.BackHandler import mozilla.components.support.base.feature.ViewBoundFeatureWrapper import mozilla.components.support.ktx.kotlin.isUrl import org.mozilla.fenix.BrowserDirection import org.mozilla.fenix.HomeActivity import org.mozilla.fenix.R import org.mozilla.fenix.components.metrics.Event import org.mozilla.fenix.components.toolbar.SearchAction import org.mozilla.fenix.components.toolbar.SearchChange import org.mozilla.fenix.components.toolbar.SearchState import org.mozilla.fenix.components.toolbar.ToolbarComponent import org.mozilla.fenix.components.toolbar.ToolbarUIView import org.mozilla.fenix.ext.components import org.mozilla.fenix.ext.requireComponents import org.mozilla.fenix.mvi.ActionBusFactory import org.mozilla.fenix.mvi.getAutoDisposeObservable import org.mozilla.fenix.mvi.getManagedEmitter import org.mozilla.fenix.search.awesomebar.AwesomeBarAction import org.mozilla.fenix.search.awesomebar.AwesomeBarChange import org.mozilla.fenix.search.awesomebar.AwesomeBarComponent import org.mozilla.fenix.search.awesomebar.AwesomeBarUIView @Suppress("TooManyFunctions") class SearchFragment : Fragment(), BackHandler { private lateinit var toolbarComponent: ToolbarComponent private lateinit var awesomeBarComponent: AwesomeBarComponent private var sessionId: String? = null private var isPrivate = false private val qrFeature = ViewBoundFeatureWrapper() override fun onCreateView( inflater: LayoutInflater, container: ViewGroup?, savedInstanceState: Bundle? ): View? { sessionId = SearchFragmentArgs.fromBundle(arguments!!).sessionId isPrivate = (activity as HomeActivity).browsingModeManager.isPrivate val session = sessionId?.let { requireComponents.core.sessionManager.findSessionById(it) } val view = inflater.inflate(R.layout.fragment_search, container, false) val url = session?.url ?: "" toolbarComponent = ToolbarComponent( view.toolbar_component_wrapper, ActionBusFactory.get(this), sessionId, isPrivate, SearchState(url, session?.searchTerms ?: "", isEditing = true), view.search_engine_icon ) awesomeBarComponent = AwesomeBarComponent(view.search_layout, ActionBusFactory.get(this)) ActionBusFactory.get(this).logMergedObservables() return view } override fun onViewCreated(view: View, savedInstanceState: Bundle?) { super.onViewCreated(view, savedInstanceState) layoutComponents(view.search_layout) qrFeature.set( QrFeature( requireContext(), fragmentManager = requireFragmentManager(), onNeedToRequestPermissions = { permissions -> requestPermissions(permissions, REQUEST_CODE_CAMERA_PERMISSIONS) }, onScanResult = { result -> (activity as HomeActivity) .openToBrowserAndLoad(result, from = BrowserDirection.FromSearch) // TODO add metrics, also should we have confirmation before going to a URL? }), owner = this, view = view ) view.search_scan_button.setOnClickListener { getManagedEmitter().onNext(SearchChange.ToolbarClearedFocus) qrFeature.get()?.scan(R.id.container) } lifecycle.addObserver((toolbarComponent.uiView as ToolbarUIView).toolbarIntegration) view.toolbar_wrapper.clipToOutline = false search_shortcuts_button.setOnClickListener { val isOpen = (awesomeBarComponent.uiView as AwesomeBarUIView).state?.showShortcutEnginePicker ?: false getManagedEmitter().onNext(AwesomeBarChange.SearchShortcutEnginePicker(!isOpen)) if (isOpen) { requireComponents.analytics.metrics.track(Event.SearchShortcutMenuClosed) } else { requireComponents.analytics.metrics.track(Event.SearchShortcutMenuOpened) } } } override fun onResume() { super.onResume() getManagedEmitter().onNext(SearchChange.ToolbarRequestedFocus) (activity as AppCompatActivity).supportActionBar?.hide() } override fun onPause() { super.onPause() getManagedEmitter().onNext(SearchChange.ToolbarClearedFocus) } override fun onStart() { super.onStart() subscribeToSearchActions() subscribeToAwesomeBarActions() } override fun onBackPressed(): Boolean { return when { qrFeature.onBackPressed() -> { view?.search_scan_button?.isChecked = false getManagedEmitter().onNext(SearchChange.ToolbarRequestedFocus) true } else -> false } } private fun subscribeToSearchActions() { getAutoDisposeObservable() .subscribe { when (it) { is SearchAction.UrlCommitted -> { if (it.url.isNotBlank()) { (activity as HomeActivity).openToBrowserAndLoad( it.url, engine = it.engine, from = BrowserDirection.FromSearch ) val event = if (it.url.isUrl()) { Event.EnteredUrl(false) } else { val engine = it.engine ?: requireComponents .search.searchEngineManager.getDefaultSearchEngine(requireContext()) createSearchEvent(engine, false) } requireComponents.analytics.metrics.track(event) } } is SearchAction.TextChanged -> { getManagedEmitter().onNext(AwesomeBarChange.UpdateQuery(it.query)) } is SearchAction.EditingCanceled -> { activity?.onBackPressed() } } } } private fun subscribeToAwesomeBarActions() { getAutoDisposeObservable() .subscribe { when (it) { is AwesomeBarAction.URLTapped -> { getSessionUseCase(requireContext(), sessionId == null).invoke(it.url) (activity as HomeActivity).openToBrowser(BrowserDirection.FromSearch) requireComponents.analytics.metrics.track(Event.EnteredUrl(false)) } is AwesomeBarAction.SearchTermsTapped -> { getSearchUseCase(requireContext(), sessionId == null) .invoke(it.searchTerms, it.engine) (activity as HomeActivity).openToBrowser(BrowserDirection.FromSearch) val engine = it.engine ?: requireComponents .search.searchEngineManager.getDefaultSearchEngine(requireContext()) val event = createSearchEvent(engine, true) requireComponents.analytics.metrics.track(event) } is AwesomeBarAction.SearchShortcutEngineSelected -> { getManagedEmitter() .onNext(AwesomeBarChange.SearchShortcutEngineSelected(it.engine)) getManagedEmitter() .onNext(SearchChange.SearchShortcutEngineSelected(it.engine)) requireComponents.analytics.metrics.track(Event.SearchShortcutSelected(it.engine.name)) } } } } private fun createSearchEvent(engine: SearchEngine, isSuggestion: Boolean): Event.PerformedSearch { val isShortcut = engine != requireComponents.search.searchEngineManager.defaultSearchEngine val engineSource = if (isShortcut) Event.PerformedSearch.EngineSource.Shortcut(engine) else Event.PerformedSearch.EngineSource.Default(engine) val source = if (isSuggestion) Event.PerformedSearch.EventSource.Suggestion(engineSource) else Event.PerformedSearch.EventSource.Action(engineSource) return Event.PerformedSearch(source) } private fun getSearchUseCase(context: Context, useNewTab: Boolean): SearchUseCases.SearchUseCase { if (!useNewTab) { return context.components.useCases.searchUseCases.defaultSearch } return when (isPrivate) { true -> context.components.useCases.searchUseCases.newPrivateTabSearch false -> context.components.useCases.searchUseCases.newTabSearch } } private fun getSessionUseCase(context: Context, useNewTab: Boolean): SessionUseCases.LoadUrlUseCase { if (!useNewTab) { return context.components.useCases.sessionUseCases.loadUrl } return when (isPrivate) { true -> context.components.useCases.tabsUseCases.addPrivateTab false -> context.components.useCases.tabsUseCases.addTab } } override fun onRequestPermissionsResult(requestCode: Int, permissions: Array, grantResults: IntArray) { when (requestCode) { REQUEST_CODE_CAMERA_PERMISSIONS -> qrFeature.withFeature { it.onPermissionsResult(permissions, grantResults) } else -> super.onRequestPermissionsResult(requestCode, permissions, grantResults) } } companion object { private const val REQUEST_CODE_CAMERA_PERMISSIONS = 1 } }