Search News Fragment
Learn how to implement search functionality in our application.
We'll cover the following...
Searching our data
We will implement an Android search widget in our fragment, allowing the user to search for articles by typing in a query. When the search is executed, the query will be sent to a function for processing, and the search will be performed.
Implementing search functionality
We’ll implement a search feature that allows users to search for specific articles by typing in the source, publishing company, or author’s name. This is done by subscribing to an observer in the ViewModel
and passing the latest data as a search query
to the searchNews()
method within a coroutine. The searchNews()
method takes in a search query
as one of its arguments and uses it to request the API to perform the desired search.
class SearchNewsFragment : Fragment(R.layout.fragment_search_news) {lateinit var viewModel: NewsViewModellateinit var newsAdapter: NewsAdapterlateinit var binding: FragmentSearchNewsBindingoverride fun onViewCreated(view: View, savedInstanceState: Bundle?) {super.onViewCreated(view, savedInstanceState)binding = FragmentSearchNewsBinding.bind(view)val newsRepository = NewsRepository(ArticleDatabase(requireContext() as NewsActivity))val viewModelProviderFactory =NewsViewModelProvider(activity?.application!!, newsRepository)viewModel = ViewModelProvider(this, viewModelProviderFactory).get(NewsViewModel::class.java)setUpRecyclerView(binding)newsAdapter.setOnItemClickListener {val bundle = Bundle().apply {putSerializable("article", it)}findNavController().navigate(R.id.action_searchNewsFragment2_to_articleFragment,bundle)}var job: Job? = nullbinding.editTextSearch.addTextChangedListener { editable ->job?.cancel()job = MainScope().launch {delay(Constants.SEARCH_NEWS_TIME_DELAY)editable?.let {if (editable.toString().isNotEmpty()) {viewModel.searchNews(editable.toString())}}}}viewModel.searchNews.observe(viewLifecycleOwner, Observer { response ->response.data?.let { newsResponse ->newsAdapter.differ.submitList(newsResponse.articles.toList())val totalPages = newsResponse.totalResultsisLastPage = viewModel.searchNewsPage == totalPages}})}private fun setUpRecyclerView(binding: FragmentSearchNewsBinding) {newsAdapter = NewsAdapter()binding.searchNewsRecyclerView.apply {adapter = newsAdapterlayoutManager = LinearLayoutManager(activity)addOnScrollListener(this@SearchNewsFragment.scrollListener)}}private fun hideProgressBar(binding: FragmentSearchNewsBinding) {binding.paginationProgressBar.visibility = View.INVISIBLEisLoading = false}private fun showProgressBar(binding: FragmentSearchNewsBinding) {binding.paginationProgressBar.visibility = View.VISIBLEisLoading = false}var isLastPage = falsevar isLoading = falsevar isScrolling = falseval scrollListener = object : RecyclerView.OnScrollListener() {override fun onScrolled(recyclerView: RecyclerView, dx: Int, dy: Int) {super.onScrolled(recyclerView, dx, dy)val layoutManager = recyclerView.layoutManager as LinearLayoutManagerval firstVisibleItemPosition = layoutManager.findFirstVisibleItemPosition()val visibleItemCount = layoutManager.childCountval totalItemCount = layoutManager.itemCountval isNotLoadingAndNotLastPage = !isLoading && !isLastPageval isAtLastItem = firstVisibleItemPosition + visibleItemCount >= totalItemCountval isNotAtBeginning = firstVisibleItemPosition >= 0val isTotalMoreThanVisible = totalItemCount >= Constants.QUERY_PAGE_SIZEval shouldPaginate = isNotLoadingAndNotLastPage && isAtLastItem && isNotAtBeginning &&isTotalMoreThanVisible && isScrollingif (shouldPaginate) {viewModel.searchNews(binding.editTextSearch.text.toString())isScrolling = false} else {binding.searchNewsRecyclerView.setPadding(0, 0, 0, 0)}}override fun onScrollStateChanged(recyclerView: RecyclerView, newState: Int) {super.onScrollStateChanged(recyclerView, newState)if (newState == AbsListView.OnScrollListener.SCROLL_STATE_TOUCH_SCROLL) {isScrolling = true}}}}
As you may have noticed, some of the ...