200 lines
8.6 KiB
Kotlin
200 lines
8.6 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.gecko.search
|
|
|
|
import android.app.PendingIntent
|
|
import android.appwidget.AppWidgetManager
|
|
import android.appwidget.AppWidgetManager.OPTION_APPWIDGET_MIN_WIDTH
|
|
import android.appwidget.AppWidgetProvider
|
|
import android.content.Context
|
|
import android.content.Intent
|
|
import android.os.Build
|
|
import android.os.Bundle
|
|
import android.speech.RecognizerIntent
|
|
import android.view.View
|
|
import android.widget.RemoteViews
|
|
import androidx.annotation.Dimension
|
|
import androidx.annotation.Dimension.DP
|
|
import androidx.appcompat.content.res.AppCompatResources
|
|
import androidx.core.graphics.drawable.toBitmap
|
|
import org.mozilla.fenix.HomeActivity
|
|
import org.mozilla.fenix.IntentReceiverActivity
|
|
import org.mozilla.fenix.R
|
|
import org.mozilla.fenix.ext.settings
|
|
import org.mozilla.fenix.home.intent.StartSearchIntentProcessor
|
|
import org.mozilla.fenix.widget.VoiceSearchActivity
|
|
import org.mozilla.fenix.widget.VoiceSearchActivity.Companion.SPEECH_PROCESSING
|
|
|
|
@Suppress("TooManyFunctions")
|
|
class SearchWidgetProvider : AppWidgetProvider() {
|
|
// Implementation note:
|
|
// This class name (SearchWidgetProvider) and package name (org.mozilla.gecko.search) should
|
|
// not be changed because otherwise this widget will disappear from the home screen of the user.
|
|
// The existing name replicates the name and package we used in Fennec.
|
|
|
|
override fun onEnabled(context: Context) {
|
|
context.settings().addSearchWidgetInstalled(1)
|
|
}
|
|
|
|
override fun onDeleted(context: Context, appWidgetIds: IntArray) {
|
|
context.settings().addSearchWidgetInstalled(-appWidgetIds.size)
|
|
}
|
|
|
|
override fun onUpdate(context: Context, appWidgetManager: AppWidgetManager, appWidgetIds: IntArray) {
|
|
val textSearchIntent = createTextSearchIntent(context)
|
|
val voiceSearchIntent = createVoiceSearchIntent(context)
|
|
|
|
appWidgetIds.forEach { appWidgetId ->
|
|
val currentWidth = appWidgetManager.getAppWidgetOptions(appWidgetId).getInt(OPTION_APPWIDGET_MIN_WIDTH)
|
|
val layoutSize = getLayoutSize(currentWidth)
|
|
// It's not enough to just hide the microphone on the "small" sized widget due to its design.
|
|
// The "small" widget needs a complete redesign, meaning it needs a new layout file.
|
|
val showMic = (voiceSearchIntent != null)
|
|
val layout = getLayout(layoutSize, showMic)
|
|
val text = getText(layoutSize, context)
|
|
|
|
val views = createRemoteViews(context, layout, textSearchIntent, voiceSearchIntent, text)
|
|
appWidgetManager.updateAppWidget(appWidgetId, views)
|
|
}
|
|
}
|
|
|
|
override fun onAppWidgetOptionsChanged(
|
|
context: Context,
|
|
appWidgetManager: AppWidgetManager,
|
|
appWidgetId: Int,
|
|
newOptions: Bundle?
|
|
) {
|
|
val textSearchIntent = createTextSearchIntent(context)
|
|
val voiceSearchIntent = createVoiceSearchIntent(context)
|
|
|
|
val currentWidth = appWidgetManager.getAppWidgetOptions(appWidgetId).getInt(OPTION_APPWIDGET_MIN_WIDTH)
|
|
val layoutSize = getLayoutSize(currentWidth)
|
|
val showMic = (voiceSearchIntent != null)
|
|
val layout = getLayout(layoutSize, showMic)
|
|
val text = getText(layoutSize, context)
|
|
|
|
val views = createRemoteViews(context, layout, textSearchIntent, voiceSearchIntent, text)
|
|
appWidgetManager.updateAppWidget(appWidgetId, views)
|
|
}
|
|
|
|
private fun getLayoutSize(@Dimension(unit = DP) dp: Int) = when {
|
|
dp >= DP_LARGE -> SearchWidgetProviderSize.LARGE
|
|
dp >= DP_MEDIUM -> SearchWidgetProviderSize.MEDIUM
|
|
dp >= DP_SMALL -> SearchWidgetProviderSize.SMALL
|
|
dp >= DP_EXTRA_SMALL -> SearchWidgetProviderSize.EXTRA_SMALL_V2
|
|
else -> SearchWidgetProviderSize.EXTRA_SMALL_V1
|
|
}
|
|
|
|
private fun getLayout(size: SearchWidgetProviderSize, showMic: Boolean) = when (size) {
|
|
SearchWidgetProviderSize.LARGE -> R.layout.search_widget_large
|
|
SearchWidgetProviderSize.MEDIUM -> R.layout.search_widget_medium
|
|
SearchWidgetProviderSize.SMALL -> {
|
|
if (showMic) R.layout.search_widget_small
|
|
else R.layout.search_widget_small_no_mic
|
|
}
|
|
SearchWidgetProviderSize.EXTRA_SMALL_V2 -> R.layout.search_widget_extra_small_v2
|
|
SearchWidgetProviderSize.EXTRA_SMALL_V1 -> R.layout.search_widget_extra_small_v1
|
|
}
|
|
|
|
private fun getText(layout: SearchWidgetProviderSize, context: Context) = when (layout) {
|
|
SearchWidgetProviderSize.MEDIUM -> context.getString(R.string.search_widget_text_short)
|
|
SearchWidgetProviderSize.LARGE -> context.getString(R.string.search_widget_text_long)
|
|
else -> null
|
|
}
|
|
|
|
private fun createTextSearchIntent(context: Context): PendingIntent {
|
|
return Intent(context, IntentReceiverActivity::class.java)
|
|
.let { intent ->
|
|
intent.flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK
|
|
intent.putExtra(HomeActivity.OPEN_TO_SEARCH, StartSearchIntentProcessor.SEARCH_WIDGET)
|
|
PendingIntent.getActivity(context,
|
|
REQUEST_CODE_NEW_TAB, intent, PendingIntent.FLAG_UPDATE_CURRENT)
|
|
}
|
|
}
|
|
|
|
private fun createVoiceSearchIntent(context: Context): PendingIntent? {
|
|
val voiceIntent = Intent(context, VoiceSearchActivity::class.java).apply {
|
|
flags = Intent.FLAG_ACTIVITY_NEW_TASK or Intent.FLAG_ACTIVITY_CLEAR_TASK
|
|
putExtra(SPEECH_PROCESSING, true)
|
|
}
|
|
|
|
val intentSpeech = Intent(RecognizerIntent.ACTION_RECOGNIZE_SPEECH)
|
|
|
|
return intentSpeech.resolveActivity(context.packageManager)?.let {
|
|
PendingIntent.getActivity(context,
|
|
REQUEST_CODE_VOICE, voiceIntent, 0)
|
|
}
|
|
}
|
|
|
|
private fun createRemoteViews(
|
|
context: Context,
|
|
layout: Int,
|
|
textSearchIntent: PendingIntent,
|
|
voiceSearchIntent: PendingIntent?,
|
|
text: String?
|
|
): RemoteViews {
|
|
return RemoteViews(context.packageName, layout).apply {
|
|
setIcon(context)
|
|
when (layout) {
|
|
R.layout.search_widget_extra_small_v1,
|
|
R.layout.search_widget_extra_small_v2,
|
|
R.layout.search_widget_small_no_mic -> {
|
|
setOnClickPendingIntent(R.id.button_search_widget_new_tab, textSearchIntent)
|
|
}
|
|
R.layout.search_widget_small -> {
|
|
setOnClickPendingIntent(R.id.button_search_widget_new_tab, textSearchIntent)
|
|
setOnClickPendingIntent(R.id.button_search_widget_voice, voiceSearchIntent)
|
|
}
|
|
R.layout.search_widget_medium,
|
|
R.layout.search_widget_large -> {
|
|
setOnClickPendingIntent(R.id.button_search_widget_new_tab, textSearchIntent)
|
|
setOnClickPendingIntent(R.id.button_search_widget_voice, voiceSearchIntent)
|
|
setOnClickPendingIntent(R.id.button_search_widget_new_tab_icon, textSearchIntent)
|
|
setTextViewText(R.id.button_search_widget_new_tab, text)
|
|
// Unlike "small" widget, "medium" and "large" sizes do not have separate layouts
|
|
// that exclude the microphone icon, which is why we must hide it accordingly here.
|
|
if (voiceSearchIntent == null) {
|
|
this.setViewVisibility(R.id.button_search_widget_voice, View.GONE)
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
private fun RemoteViews.setIcon(context: Context) {
|
|
// gradient color available for android:fillColor only on SDK 24+
|
|
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.N) {
|
|
setImageViewResource(
|
|
R.id.button_search_widget_new_tab_icon,
|
|
R.drawable.ic_launcher_foreground)
|
|
} else {
|
|
setImageViewBitmap(
|
|
R.id.button_search_widget_new_tab_icon,
|
|
AppCompatResources.getDrawable(
|
|
context,
|
|
R.drawable.ic_launcher_foreground
|
|
)?.toBitmap())
|
|
}
|
|
}
|
|
|
|
// Cell sizes obtained from the actual dimensions listed in search widget specs
|
|
companion object {
|
|
private const val DP_EXTRA_SMALL = 64
|
|
private const val DP_SMALL = 100
|
|
private const val DP_MEDIUM = 192
|
|
private const val DP_LARGE = 256
|
|
private const val REQUEST_CODE_NEW_TAB = 0
|
|
private const val REQUEST_CODE_VOICE = 1
|
|
}
|
|
}
|
|
|
|
enum class SearchWidgetProviderSize {
|
|
EXTRA_SMALL_V1,
|
|
EXTRA_SMALL_V2,
|
|
SMALL,
|
|
MEDIUM,
|
|
LARGE,
|
|
}
|