/* * Nextcloud Talk application * * @author Marcel Hibbe * Copyright (C) 2021 Marcel Hibbe * * This program is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program. If not, see . */ package com.nextcloud.talk.location import android.app.SearchManager import android.content.Context import android.content.Intent import android.graphics.drawable.ColorDrawable import android.os.Build import android.os.Bundle import android.text.InputType import android.util.Log import android.view.Menu import android.view.MenuItem import android.view.inputmethod.EditorInfo import android.widget.AdapterView import android.widget.Toast import androidx.appcompat.widget.SearchView import androidx.core.view.MenuItemCompat import androidx.preference.PreferenceManager import autodagger.AutoInjector import com.nextcloud.talk.R import com.nextcloud.talk.activities.BaseActivity import com.nextcloud.talk.adapters.GeocodingAdapter import com.nextcloud.talk.api.NcApi import com.nextcloud.talk.application.NextcloudTalkApplication import com.nextcloud.talk.databinding.ActivityGeocodingBinding import com.nextcloud.talk.utils.bundle.BundleKeys import fr.dudie.nominatim.client.TalkJsonNominatimClient import fr.dudie.nominatim.model.Address import kotlinx.coroutines.CoroutineScope import kotlinx.coroutines.Dispatchers.IO import kotlinx.coroutines.Dispatchers.Main import kotlinx.coroutines.launch import kotlinx.coroutines.withContext import okhttp3.OkHttpClient import org.osmdroid.config.Configuration import javax.inject.Inject @AutoInjector(NextcloudTalkApplication::class) class GeocodingActivity : BaseActivity(), SearchView.OnQueryTextListener { private lateinit var binding: ActivityGeocodingBinding @Inject lateinit var ncApi: NcApi @Inject lateinit var okHttpClient: OkHttpClient lateinit var roomToken: String var nominatimClient: TalkJsonNominatimClient? = null var searchItem: MenuItem? = null var searchView: SearchView? = null var query: String? = null lateinit var adapter: GeocodingAdapter private var geocodingResults: List
= ArrayList() override fun onCreate(savedInstanceState: Bundle?) { super.onCreate(savedInstanceState) NextcloudTalkApplication.sharedApplication!!.componentApplication.inject(this) binding = ActivityGeocodingBinding.inflate(layoutInflater) setupActionBar() setContentView(binding.root) setupSystemColors() Configuration.getInstance().load(context, PreferenceManager.getDefaultSharedPreferences(context)) roomToken = intent.getStringExtra(BundleKeys.KEY_ROOM_TOKEN)!! query = intent.getStringExtra(BundleKeys.KEY_GEOCODING_QUERY) } override fun onStart() { super.onStart() initAdapter(geocodingResults) initGeocoder() } override fun onResume() { super.onResume() if (!query.isNullOrEmpty()) { searchLocation() } else { Log.e(TAG, "search string that was passed to GeocodingController was null or empty") } binding.geocodingResults.onItemClickListener = AdapterView.OnItemClickListener { parent, view, position, id -> val address: Address = adapter.getItem(position) as Address val geocodingResult = GeocodingResult(address.latitude, address.longitude, address.displayName) val intent = Intent(this, LocationPickerActivity::class.java) intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) intent.putExtra(BundleKeys.KEY_ROOM_TOKEN, roomToken) intent.putExtra(BundleKeys.KEY_GEOCODING_RESULT, geocodingResult) startActivity(intent) } } private fun setupActionBar() { setSupportActionBar(binding.geocodingToolbar) binding.geocodingToolbar.setNavigationOnClickListener { onBackPressedDispatcher.onBackPressed() } supportActionBar?.setDisplayHomeAsUpEnabled(true) supportActionBar?.setDisplayShowHomeEnabled(true) supportActionBar?.setIcon(ColorDrawable(resources!!.getColor(R.color.transparent, null))) supportActionBar?.title = "" viewThemeUtils.material.themeToolbar(binding.geocodingToolbar) } private fun initAdapter(addresses: List
) { adapter = GeocodingAdapter(binding.geocodingResults.context!!, addresses) binding.geocodingResults.adapter = adapter } override fun onCreateOptionsMenu(menu: Menu): Boolean { super.onCreateOptionsMenu(menu) menuInflater.inflate(R.menu.menu_geocoding, menu) searchItem = menu.findItem(R.id.geocoding_action_search) initSearchView() searchItem?.expandActionView() searchView?.setQuery(query, false) searchView?.clearFocus() return true } override fun onQueryTextSubmit(query: String?): Boolean { this.query = query searchLocation() searchView?.clearFocus() return true } override fun onQueryTextChange(newText: String?): Boolean { return true } private fun initSearchView() { val searchManager = getSystemService(Context.SEARCH_SERVICE) as SearchManager if (searchItem != null) { searchView = MenuItemCompat.getActionView(searchItem) as SearchView searchView?.maxWidth = Int.MAX_VALUE searchView?.inputType = InputType.TYPE_TEXT_VARIATION_FILTER var imeOptions = EditorInfo.IME_ACTION_DONE or EditorInfo.IME_FLAG_NO_FULLSCREEN if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O && appPreferences!!.isKeyboardIncognito) { imeOptions = imeOptions or EditorInfo.IME_FLAG_NO_PERSONALIZED_LEARNING } searchView?.imeOptions = imeOptions searchView?.queryHint = resources!!.getString(R.string.nc_search) searchView?.setSearchableInfo(searchManager.getSearchableInfo(componentName)) searchView?.setOnQueryTextListener(this) searchItem?.setOnActionExpandListener(object : MenuItem.OnActionExpandListener { override fun onMenuItemActionExpand(menuItem: MenuItem): Boolean { return true } override fun onMenuItemActionCollapse(menuItem: MenuItem): Boolean { val intent = Intent(context, LocationPickerActivity::class.java) intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP) intent.putExtra(BundleKeys.KEY_ROOM_TOKEN, roomToken) startActivity(intent) return true } }) } } private fun initGeocoder() { val baseUrl = getString(R.string.osm_geocoder_url) val email = context.getString(R.string.osm_geocoder_contact) nominatimClient = TalkJsonNominatimClient(baseUrl, okHttpClient, email) } private fun searchLocation(): Boolean { CoroutineScope(IO).launch { executeGeocodingRequest() } return true } @Suppress("Detekt.TooGenericExceptionCaught") private suspend fun executeGeocodingRequest() { var results: ArrayList
= ArrayList() try { results = nominatimClient!!.search(query) as ArrayList
for (address in results) { Log.d(TAG, address.displayName) Log.d(TAG, address.latitude.toString()) Log.d(TAG, address.longitude.toString()) } } catch (e: Exception) { Log.e(TAG, "Failed to get geocoded addresses", e) Toast.makeText(context, R.string.nc_common_error_sorry, Toast.LENGTH_LONG).show() } updateResultsOnMainThread(results) } private suspend fun updateResultsOnMainThread(results: ArrayList
) { withContext(Main) { initAdapter(results) } } companion object { private val TAG = GeocodingActivity::class.java.simpleName } }