parent
2e4ab8b387
commit
e57aa67d60
|
@ -15,9 +15,12 @@ import androidx.appcompat.widget.Toolbar
|
||||||
import androidx.navigation.fragment.NavHostFragment
|
import androidx.navigation.fragment.NavHostFragment
|
||||||
import androidx.navigation.ui.AppBarConfiguration
|
import androidx.navigation.ui.AppBarConfiguration
|
||||||
import androidx.navigation.ui.NavigationUI
|
import androidx.navigation.ui.NavigationUI
|
||||||
|
import mozilla.components.browser.session.Session
|
||||||
import mozilla.components.concept.engine.EngineView
|
import mozilla.components.concept.engine.EngineView
|
||||||
import mozilla.components.feature.intent.IntentProcessor
|
import mozilla.components.feature.intent.IntentProcessor
|
||||||
import mozilla.components.support.base.feature.BackHandler
|
import mozilla.components.support.base.feature.BackHandler
|
||||||
|
import mozilla.components.support.ktx.kotlin.isUrl
|
||||||
|
import mozilla.components.support.ktx.kotlin.toNormalizedUrl
|
||||||
import mozilla.components.support.utils.SafeIntent
|
import mozilla.components.support.utils.SafeIntent
|
||||||
import org.mozilla.fenix.ext.components
|
import org.mozilla.fenix.ext.components
|
||||||
|
|
||||||
|
@ -41,17 +44,16 @@ open class HomeActivity : AppCompatActivity() {
|
||||||
|
|
||||||
setContentView(R.layout.activity_home)
|
setContentView(R.layout.activity_home)
|
||||||
|
|
||||||
if (intent?.extras?.getBoolean(OPEN_TO_BROWSER) == true) {
|
|
||||||
intent?.putExtra(OPEN_TO_BROWSER, false)
|
|
||||||
openToBrowser()
|
|
||||||
}
|
|
||||||
|
|
||||||
val host = supportFragmentManager.findFragmentById(R.id.container) as NavHostFragment
|
val host = supportFragmentManager.findFragmentById(R.id.container) as NavHostFragment
|
||||||
val hostNavController = host.navController
|
val hostNavController = host.navController
|
||||||
val appBarConfiguration = AppBarConfiguration.Builder(setOf(R.id.libraryFragment)).build()
|
val appBarConfiguration = AppBarConfiguration.Builder(setOf(R.id.libraryFragment)).build()
|
||||||
val navigationToolbar = findViewById<Toolbar>(R.id.navigationToolbar)
|
val navigationToolbar = findViewById<Toolbar>(R.id.navigationToolbar)
|
||||||
setSupportActionBar(navigationToolbar)
|
setSupportActionBar(navigationToolbar)
|
||||||
NavigationUI.setupWithNavController(navigationToolbar, hostNavController, appBarConfiguration)
|
NavigationUI.setupWithNavController(navigationToolbar, hostNavController, appBarConfiguration)
|
||||||
|
|
||||||
|
if (intent?.extras?.getBoolean(OPEN_TO_BROWSER) == true) {
|
||||||
|
handleOpenedFromExternalSource()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun onCreateView(
|
override fun onCreateView(
|
||||||
|
@ -88,13 +90,48 @@ open class HomeActivity : AppCompatActivity() {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private fun openToBrowser() {
|
private fun handleOpenedFromExternalSource() {
|
||||||
val sessionId = SafeIntent(intent).getStringExtra(IntentProcessor.ACTIVE_SESSION_ID)
|
intent?.putExtra(OPEN_TO_BROWSER, false)
|
||||||
|
openToBrowser(SafeIntent(intent).getStringExtra(IntentProcessor.ACTIVE_SESSION_ID))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Since we must always call load after navigation, we only directly expose the load when coupled with open.
|
||||||
|
fun openToBrowserAndLoad(text: String, sessionId: String? = null) {
|
||||||
|
openToBrowser(sessionId)
|
||||||
|
load(text, sessionId)
|
||||||
|
}
|
||||||
|
|
||||||
|
fun openToBrowser(sessionId: String?) {
|
||||||
val host = supportFragmentManager.findFragmentById(R.id.container) as NavHostFragment
|
val host = supportFragmentManager.findFragmentById(R.id.container) as NavHostFragment
|
||||||
val directions = NavGraphDirections.actionGlobalBrowser(sessionId)
|
val directions = NavGraphDirections.actionGlobalBrowser(sessionId)
|
||||||
host.navController.navigate(directions)
|
host.navController.navigate(directions)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun load(text: String, sessionId: String?) {
|
||||||
|
val isPrivate = this.browsingModeManager.isPrivate
|
||||||
|
|
||||||
|
val loadUrlUseCase = if (sessionId == null) {
|
||||||
|
if (isPrivate) {
|
||||||
|
components.useCases.tabsUseCases.addPrivateTab
|
||||||
|
} else {
|
||||||
|
components.useCases.tabsUseCases.addTab
|
||||||
|
}
|
||||||
|
} else components.useCases.sessionUseCases.loadUrl
|
||||||
|
|
||||||
|
val searchUseCase: (String) -> Unit = { searchTerms ->
|
||||||
|
if (sessionId == null) {
|
||||||
|
components.useCases.searchUseCases.newTabSearch
|
||||||
|
.invoke(searchTerms, Session.Source.USER_ENTERED, true, isPrivate)
|
||||||
|
} else components.useCases.searchUseCases.defaultSearch.invoke(searchTerms)
|
||||||
|
}
|
||||||
|
|
||||||
|
if (text.isUrl()) {
|
||||||
|
loadUrlUseCase.invoke(text.toNormalizedUrl())
|
||||||
|
} else {
|
||||||
|
searchUseCase.invoke(text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
companion object {
|
companion object {
|
||||||
const val OPEN_TO_BROWSER = "open_to_browser"
|
const val OPEN_TO_BROWSER = "open_to_browser"
|
||||||
}
|
}
|
||||||
|
|
|
@ -7,6 +7,7 @@ package org.mozilla.fenix
|
||||||
import android.app.Activity
|
import android.app.Activity
|
||||||
import android.content.Intent
|
import android.content.Intent
|
||||||
import android.os.Bundle
|
import android.os.Bundle
|
||||||
|
import android.preference.PreferenceManager
|
||||||
import mozilla.components.browser.session.tab.CustomTabConfig
|
import mozilla.components.browser.session.tab.CustomTabConfig
|
||||||
import mozilla.components.support.utils.SafeIntent
|
import mozilla.components.support.utils.SafeIntent
|
||||||
import org.mozilla.fenix.customtabs.CustomTabActivity
|
import org.mozilla.fenix.customtabs.CustomTabActivity
|
||||||
|
@ -17,8 +18,16 @@ class IntentReceiverActivity : Activity() {
|
||||||
override fun onCreate(savedInstanceState: Bundle?) {
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
super.onCreate(savedInstanceState)
|
super.onCreate(savedInstanceState)
|
||||||
|
|
||||||
components.utils.intentProcessor.process(intent)
|
val isPrivate = PreferenceManager
|
||||||
var openToBrowser = false
|
.getDefaultSharedPreferences(this).getBoolean(getString(R.string.pref_key_private_mode), false)
|
||||||
|
|
||||||
|
if (isPrivate) {
|
||||||
|
components.utils.privateIntentProcessor.process(intent)
|
||||||
|
} else {
|
||||||
|
components.utils.intentProcessor.process(intent)
|
||||||
|
}
|
||||||
|
|
||||||
|
val openToBrowser: Boolean
|
||||||
|
|
||||||
val intent = Intent(intent)
|
val intent = Intent(intent)
|
||||||
openToBrowser = when {
|
openToBrowser = when {
|
||||||
|
|
|
@ -26,4 +26,8 @@ class Utilities(
|
||||||
val intentProcessor by lazy {
|
val intentProcessor by lazy {
|
||||||
IntentProcessor(sessionUseCases, sessionManager, searchUseCases, context)
|
IntentProcessor(sessionUseCases, sessionManager, searchUseCases, context)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
val privateIntentProcessor by lazy {
|
||||||
|
IntentProcessor(sessionUseCases, sessionManager, searchUseCases, context, isPrivate = true)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -54,7 +54,7 @@ class ToolbarComponent(
|
||||||
data class SearchState(val query: String, val isEditing: Boolean) : ViewState
|
data class SearchState(val query: String, val isEditing: Boolean) : ViewState
|
||||||
|
|
||||||
sealed class SearchAction : Action {
|
sealed class SearchAction : Action {
|
||||||
data class UrlCommitted(val url: String) : SearchAction()
|
data class UrlCommitted(val url: String, val session: String?) : SearchAction()
|
||||||
data class TextChanged(val query: String) : SearchAction()
|
data class TextChanged(val query: String) : SearchAction()
|
||||||
object ToolbarTapped : SearchAction()
|
object ToolbarTapped : SearchAction()
|
||||||
data class ToolbarMenuItemTapped(val item: ToolbarMenu.Item) : SearchAction()
|
data class ToolbarMenuItemTapped(val item: ToolbarMenu.Item) : SearchAction()
|
||||||
|
|
|
@ -38,7 +38,7 @@ class ToolbarUIView(
|
||||||
init {
|
init {
|
||||||
view.apply {
|
view.apply {
|
||||||
setOnUrlCommitListener {
|
setOnUrlCommitListener {
|
||||||
actionEmitter.onNext(SearchAction.UrlCommitted(it))
|
actionEmitter.onNext(SearchAction.UrlCommitted(it, sessionId))
|
||||||
false
|
false
|
||||||
}
|
}
|
||||||
onUrlClicked = {
|
onUrlClicked = {
|
||||||
|
|
|
@ -10,11 +10,7 @@ import android.view.View
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import androidx.appcompat.app.AppCompatActivity
|
import androidx.appcompat.app.AppCompatActivity
|
||||||
import androidx.fragment.app.Fragment
|
import androidx.fragment.app.Fragment
|
||||||
import androidx.navigation.Navigation
|
|
||||||
import kotlinx.android.synthetic.main.fragment_search.view.*
|
import kotlinx.android.synthetic.main.fragment_search.view.*
|
||||||
import mozilla.components.browser.session.Session
|
|
||||||
import mozilla.components.support.ktx.kotlin.isUrl
|
|
||||||
import mozilla.components.support.ktx.kotlin.toNormalizedUrl
|
|
||||||
import org.mozilla.fenix.HomeActivity
|
import org.mozilla.fenix.HomeActivity
|
||||||
import org.mozilla.fenix.R
|
import org.mozilla.fenix.R
|
||||||
import org.mozilla.fenix.components.toolbar.SearchAction
|
import org.mozilla.fenix.components.toolbar.SearchAction
|
||||||
|
@ -57,7 +53,7 @@ class SearchFragment : Fragment() {
|
||||||
)
|
)
|
||||||
awesomeBarComponent = AwesomeBarComponent(
|
awesomeBarComponent = AwesomeBarComponent(
|
||||||
view.search_layout, ActionBusFactory.get(this),
|
view.search_layout, ActionBusFactory.get(this),
|
||||||
AwesomeBarState("", sessionId == null)
|
AwesomeBarState("", sessionId == null, isPrivate)
|
||||||
)
|
)
|
||||||
ActionBusFactory.get(this).logMergedObservables()
|
ActionBusFactory.get(this).logMergedObservables()
|
||||||
return view
|
return view
|
||||||
|
@ -85,8 +81,7 @@ class SearchFragment : Fragment() {
|
||||||
when (it) {
|
when (it) {
|
||||||
is SearchAction.UrlCommitted -> {
|
is SearchAction.UrlCommitted -> {
|
||||||
if (it.url.isNotBlank()) {
|
if (it.url.isNotBlank()) {
|
||||||
transitionToBrowser()
|
(activity as HomeActivity).openToBrowserAndLoad(it.url, it.session)
|
||||||
load(it.url)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
is SearchAction.TextChanged -> {
|
is SearchAction.TextChanged -> {
|
||||||
|
@ -98,46 +93,12 @@ class SearchFragment : Fragment() {
|
||||||
getAutoDisposeObservable<AwesomeBarAction>()
|
getAutoDisposeObservable<AwesomeBarAction>()
|
||||||
.subscribe {
|
.subscribe {
|
||||||
when (it) {
|
when (it) {
|
||||||
is AwesomeBarAction.ItemSelected -> transitionToBrowser()
|
is AwesomeBarAction.ItemSelected -> {
|
||||||
|
requireComponents.core.sessionManager.selectedSession?.let { session ->
|
||||||
|
(activity as HomeActivity).openToBrowser(session.id)
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Issue: https://github.com/mozilla-mobile/fenix/issues/626
|
|
||||||
// Currently we were kind of forcing all this logic through the Toolbar Feature.
|
|
||||||
// But since we cannot actually load a page without an available GeckoSession
|
|
||||||
// we have to wait until after we navigate to call the use case.
|
|
||||||
// We should move this logic into a place that makes more sense.
|
|
||||||
private fun load(text: String) {
|
|
||||||
val sessionId = SearchFragmentArgs.fromBundle(arguments!!).sessionId
|
|
||||||
val isPrivate = (activity as HomeActivity).browsingModeManager.isPrivate
|
|
||||||
|
|
||||||
val loadUrlUseCase = if (sessionId == null) {
|
|
||||||
if (isPrivate) {
|
|
||||||
requireComponents.useCases.tabsUseCases.addPrivateTab
|
|
||||||
} else {
|
|
||||||
requireComponents.useCases.tabsUseCases.addTab
|
|
||||||
}
|
|
||||||
} else requireComponents.useCases.sessionUseCases.loadUrl
|
|
||||||
|
|
||||||
val searchUseCase: (String) -> Unit = { searchTerms ->
|
|
||||||
if (sessionId == null) {
|
|
||||||
requireComponents.useCases.searchUseCases.newTabSearch
|
|
||||||
.invoke(searchTerms, Session.Source.USER_ENTERED, true, isPrivate)
|
|
||||||
} else requireComponents.useCases.searchUseCases.defaultSearch.invoke(searchTerms)
|
|
||||||
}
|
|
||||||
|
|
||||||
if (text.isUrl()) {
|
|
||||||
loadUrlUseCase.invoke(text.toNormalizedUrl())
|
|
||||||
} else {
|
|
||||||
searchUseCase.invoke(text)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private fun transitionToBrowser() {
|
|
||||||
val sessionId = SearchFragmentArgs.fromBundle(arguments!!).sessionId
|
|
||||||
val directions = SearchFragmentDirections.actionSearchFragmentToBrowserFragment(sessionId)
|
|
||||||
|
|
||||||
Navigation.findNavController(view!!.search_layout).navigate(directions)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,7 +11,7 @@ import org.mozilla.fenix.mvi.Reducer
|
||||||
import org.mozilla.fenix.mvi.UIComponent
|
import org.mozilla.fenix.mvi.UIComponent
|
||||||
import org.mozilla.fenix.mvi.ViewState
|
import org.mozilla.fenix.mvi.ViewState
|
||||||
|
|
||||||
data class AwesomeBarState(val query: String, val useNewTab: Boolean = false) : ViewState
|
data class AwesomeBarState(val query: String, val useNewTab: Boolean = false, val isPrivate: Boolean) : ViewState
|
||||||
|
|
||||||
sealed class AwesomeBarAction : Action {
|
sealed class AwesomeBarAction : Action {
|
||||||
object ItemSelected : AwesomeBarAction()
|
object ItemSelected : AwesomeBarAction()
|
||||||
|
@ -24,7 +24,7 @@ sealed class AwesomeBarChange : Change {
|
||||||
class AwesomeBarComponent(
|
class AwesomeBarComponent(
|
||||||
private val container: ViewGroup,
|
private val container: ViewGroup,
|
||||||
bus: ActionBusFactory,
|
bus: ActionBusFactory,
|
||||||
override var initialState: AwesomeBarState = AwesomeBarState("")
|
override var initialState: AwesomeBarState = AwesomeBarState("", isPrivate = false)
|
||||||
) : UIComponent<AwesomeBarState, AwesomeBarAction, AwesomeBarChange>(
|
) : UIComponent<AwesomeBarState, AwesomeBarAction, AwesomeBarChange>(
|
||||||
bus.getManagedEmitter(AwesomeBarAction::class.java),
|
bus.getManagedEmitter(AwesomeBarAction::class.java),
|
||||||
bus.getSafeManagedObservable(AwesomeBarChange::class.java)
|
bus.getSafeManagedObservable(AwesomeBarChange::class.java)
|
||||||
|
@ -35,7 +35,12 @@ class AwesomeBarComponent(
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
override fun initView() = AwesomeBarUIView(initialState.useNewTab, container, actionEmitter, changesObservable)
|
override fun initView() = AwesomeBarUIView(
|
||||||
|
initialState.useNewTab,
|
||||||
|
initialState.isPrivate,
|
||||||
|
container,
|
||||||
|
actionEmitter,
|
||||||
|
changesObservable)
|
||||||
|
|
||||||
init {
|
init {
|
||||||
render(reducer)
|
render(reducer)
|
||||||
|
|
|
@ -3,6 +3,7 @@ package org.mozilla.fenix.search.awesomebar
|
||||||
License, v. 2.0. If a copy of the MPL was not distributed with this
|
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/. */
|
file, You can obtain one at http://mozilla.org/MPL/2.0/. */
|
||||||
|
|
||||||
|
import android.content.Context
|
||||||
import android.view.LayoutInflater
|
import android.view.LayoutInflater
|
||||||
import android.view.ViewGroup
|
import android.view.ViewGroup
|
||||||
import io.reactivex.Observable
|
import io.reactivex.Observable
|
||||||
|
@ -13,6 +14,9 @@ import mozilla.components.feature.awesomebar.provider.ClipboardSuggestionProvide
|
||||||
import mozilla.components.feature.awesomebar.provider.HistoryStorageSuggestionProvider
|
import mozilla.components.feature.awesomebar.provider.HistoryStorageSuggestionProvider
|
||||||
import mozilla.components.feature.awesomebar.provider.SearchSuggestionProvider
|
import mozilla.components.feature.awesomebar.provider.SearchSuggestionProvider
|
||||||
import mozilla.components.feature.awesomebar.provider.SessionSuggestionProvider
|
import mozilla.components.feature.awesomebar.provider.SessionSuggestionProvider
|
||||||
|
import mozilla.components.feature.search.SearchUseCases
|
||||||
|
import mozilla.components.feature.session.SessionUseCases
|
||||||
|
import mozilla.components.feature.toolbar.SearchUseCase
|
||||||
import mozilla.components.support.ktx.android.graphics.drawable.toBitmap
|
import mozilla.components.support.ktx.android.graphics.drawable.toBitmap
|
||||||
import org.mozilla.fenix.R
|
import org.mozilla.fenix.R
|
||||||
import org.mozilla.fenix.ext.components
|
import org.mozilla.fenix.ext.components
|
||||||
|
@ -20,6 +24,7 @@ import org.mozilla.fenix.mvi.UIView
|
||||||
|
|
||||||
class AwesomeBarUIView(
|
class AwesomeBarUIView(
|
||||||
useNewTab: Boolean,
|
useNewTab: Boolean,
|
||||||
|
isPrivate: Boolean,
|
||||||
container: ViewGroup,
|
container: ViewGroup,
|
||||||
actionEmitter: Observer<AwesomeBarAction>,
|
actionEmitter: Observer<AwesomeBarAction>,
|
||||||
changesObservable: Observable<AwesomeBarChange>
|
changesObservable: Observable<AwesomeBarChange>
|
||||||
|
@ -33,11 +38,7 @@ class AwesomeBarUIView(
|
||||||
with(container.context) {
|
with(container.context) {
|
||||||
view.addProviders(ClipboardSuggestionProvider(
|
view.addProviders(ClipboardSuggestionProvider(
|
||||||
this,
|
this,
|
||||||
if (useNewTab) {
|
getSessionUseCase(this, isPrivate, useNewTab),
|
||||||
components.useCases.tabsUseCases.addTab
|
|
||||||
} else {
|
|
||||||
components.useCases.sessionUseCases.loadUrl
|
|
||||||
},
|
|
||||||
getDrawable(R.drawable.ic_link).toBitmap(),
|
getDrawable(R.drawable.ic_link).toBitmap(),
|
||||||
getString(R.string.awesomebar_clipboard_title)
|
getString(R.string.awesomebar_clipboard_title)
|
||||||
)
|
)
|
||||||
|
@ -50,15 +51,10 @@ class AwesomeBarUIView(
|
||||||
),
|
),
|
||||||
HistoryStorageSuggestionProvider(
|
HistoryStorageSuggestionProvider(
|
||||||
components.core.historyStorage,
|
components.core.historyStorage,
|
||||||
if (useNewTab) {
|
getSessionUseCase(this, isPrivate, useNewTab)),
|
||||||
components.useCases.tabsUseCases.addTab
|
|
||||||
} else components.useCases.sessionUseCases.loadUrl
|
|
||||||
),
|
|
||||||
SearchSuggestionProvider(
|
SearchSuggestionProvider(
|
||||||
components.search.searchEngineManager.getDefaultSearchEngine(this),
|
components.search.searchEngineManager.getDefaultSearchEngine(this),
|
||||||
if (useNewTab) {
|
getSearchUseCase(this, isPrivate, useNewTab),
|
||||||
components.useCases.searchUseCases.newTabSearch
|
|
||||||
} else components.useCases.searchUseCases.defaultSearch,
|
|
||||||
components.core.client,
|
components.core.client,
|
||||||
SearchSuggestionProvider.Mode.MULTIPLE_SUGGESTIONS
|
SearchSuggestionProvider.Mode.MULTIPLE_SUGGESTIONS
|
||||||
)
|
)
|
||||||
|
@ -68,6 +64,32 @@ class AwesomeBarUIView(
|
||||||
view.setOnStopListener { actionEmitter.onNext(AwesomeBarAction.ItemSelected) }
|
view.setOnStopListener { actionEmitter.onNext(AwesomeBarAction.ItemSelected) }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private fun getSearchUseCase(
|
||||||
|
context: Context,
|
||||||
|
isPrivate: Boolean,
|
||||||
|
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, isPrivate: Boolean, 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 updateView() = Consumer<AwesomeBarState> {
|
override fun updateView() = Consumer<AwesomeBarState> {
|
||||||
view.onInputChanged(it.query)
|
view.onInputChanged(it.query)
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue