I'm Currently facing an issue with API, the issue is that the data I got is 200 OK but the adapter cannot show the data even though there is no error, and I Receive the data
This is the home fragment
package com.example.papb_tubes
import android.os.Bundle
import android.util.Log
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import androidx.fragment.app.Fragment
import androidx.recyclerview.widget.LinearLayoutManager
import com.example.papb_tubes.databinding.FragmentHomeBinding
import retrofit2.Call
import retrofit2.Callback
import retrofit2.Response
class HomeFragment : Fragment() {
private var _binding: FragmentHomeBinding? = null
private val binding get() = _binding!!
private lateinit var apiService: ApiService
private lateinit var weatherAdapter: WeatherAdapter
override fun onCreateView(
inflater: LayoutInflater, container: ViewGroup?,
savedInstanceState: Bundle?
): View? {
_binding = FragmentHomeBinding.inflate(inflater, container, false)
return binding.root
}
override fun onViewCreated(view: View, savedInstanceState: Bundle?) {
super.onViewCreated(view, savedInstanceState)
apiService = ApiConfig.getApiService("London") // Set initial city
weatherAdapter = WeatherAdapter(object : WeatherAdapter.OnClickListener {
override fun onClickItem(data: WeatherResponse) {
// Handle item click if needed
}
})
binding.rvHomepage.apply {
layoutManager = LinearLayoutManager(requireContext())
adapter = weatherAdapter
}
binding.ivPerson.setOnClickListener {
// Navigate to profile fragment if needed
}
// Fetch initial weather data for London
fetchWeatherData("London")
}
private fun fetchWeatherData(city: String) {
apiService = ApiConfig.getApiService(city)
apiService.getAllProvinsi(city).enqueue(object : Callback<WeatherResponse> {
override fun onResponse(
call: Call<WeatherResponse>,
response: Response<WeatherResponse>
) {
if (response.isSuccessful) {
val weatherResponse = response.body()
if (weatherResponse != null) {
Log.d("WeatherResponse", weatherResponse.toString())
weatherAdapter.submitData(weatherResponse)
} else {
Log.e("WeatherResponse", "Response body is null")
}
} else {
Log.e("WeatherResponse", "Error: ${response.code()}")
}
}
override fun onFailure(call: Call<WeatherResponse>, t: Throwable) {
}
})
}
override fun onDestroyView() {
super.onDestroyView()
_binding = null
}
}
this is the adapter
package com.example.papb_tubes
import android.util.Log
import android.view.LayoutInflater
import android.view.ViewGroup
import androidx.recyclerview.widget.AsyncListDiffer
import androidx.recyclerview.widget.DiffUtil
import androidx.recyclerview.widget.RecyclerView
import com.example.papb_tubes.databinding.ItemRowBinding
class WeatherAdapter(
private val onClickItem: OnClickListener // Use the correct interface name here
) : RecyclerView.Adapter<WeatherAdapter.ViewHolder>() {
private val differCallback = object : DiffUtil.ItemCallback<WeatherResponse>() {
override fun areItemsTheSame(oldItem: WeatherResponse, newItem: WeatherResponse): Boolean {
return oldItem.id == newItem.id
}
override fun areContentsTheSame(oldItem: WeatherResponse, newItem: WeatherResponse): Boolean = oldItem.hashCode() == newItem.hashCode()
}
val differ = AsyncListDiffer(this, differCallback)
fun submitData(value: WeatherResponse) = differ.submitList(listOf(value))
inner class ViewHolder(private val binding: ItemRowBinding) :
RecyclerView.ViewHolder(binding.root) {
fun bind(data: WeatherResponse) {
val temp = (data.main.temp) - 273
binding.apply {
tvNamakota.text = data.name.toString()
tvTemperatur.text = temp.toString()
tvTipe.text = data.weather[0].description.toString()
root.setOnClickListener {
onClickItem.onClickItem(data)
}
}
}
}
override fun onBindViewHolder(holder: ViewHolder, position: Int) {
val data = differ.currentList[position]
Log.d("WeatherAdapter", "Data size: ${differ.currentList.size}")
data.let { holder.bind(data) }
}
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): ViewHolder {
val inflater = LayoutInflater.from(parent.context)
return ViewHolder(ItemRowBinding.inflate(inflater,parent,false))
}
override fun getItemCount(): Int = differ.currentList.size
interface OnClickListener {
fun onClickItem(data: WeatherResponse)
}
}
this is the apiconfig
package com.example.papb_tubes
import okhttp3.Interceptor
import okhttp3.OkHttpClient
import okhttp3.logging.HttpLoggingInterceptor
import retrofit2.Retrofit
import retrofit2.converter.gson.GsonConverterFactory
class ApiConfig {
companion object {
private const val BASE_URL = "https://api.openweathermap.org/data/2.5/"
private const val API_KEY = ""
fun getApiService(city: String): ApiService {
val loggingInterceptor =
HttpLoggingInterceptor().setLevel(HttpLoggingInterceptor.Level.BODY)
val authInterceptor = Interceptor { chain ->
val originalRequest = chain.request()
val newUrl = originalRequest.url.newBuilder()
.addQueryParameter("appid", API_KEY)
.build()
val newRequest = originalRequest.newBuilder()
.url(newUrl)
.build()
chain.proceed(newRequest)
}
val client = OkHttpClient.Builder()
.addInterceptor(loggingInterceptor)
.addInterceptor(authInterceptor)
.build()
val retrofit = Retrofit.Builder()
.baseUrl(BASE_URL)
.addConverterFactory(GsonConverterFactory.create())
.client(client)
.build()
return retrofit.create(ApiService::class.java)
}
}
}
and this is the service
package com.example.papb_tubes
import retrofit2.Call
import retrofit2.http.GET
import retrofit2.http.Query
interface ApiService {
@GET("weather")
fun getAllProvinsi(
@Query("q") city:String,
): Call<WeatherResponse>
}
and this is the homefragment.xml
<?xml version="1.0" encoding="utf-8"?>
<FrameLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".HomeFragment"
android:background="@color/primary">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_ellipse"
android:id="@+id/iv_elipse_1"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_ellipse"
android:id="@+id/iv_elipse_2"
android:layout_marginTop="300dp"
android:layout_marginLeft="160dp"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_ellipse"
android:id="@+id/iv_elipse_3"
android:layout_marginTop="500dp"
android:layout_marginLeft="0dp"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_rectangle"
android:layout_gravity="right"
android:id="@+id/iv_rectangle1"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_rectangle"
android:layout_gravity="right|bottom"
android:id="@+id/iv_rectangle2"/>
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:src="@drawable/ic_rectangle"
android:layout_gravity="left|center"
android:scaleX="-1"
android:id="@+id/iv_rectangle3"/>
<LinearLayout
android:id="@+id/ll_homepage"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:orientation="vertical"
android:layout_marginTop="20sp"
android:layout_marginBottom="20sp">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:orientation="horizontal"
android:paddingLeft="20dp"
android:paddingTop="30dp"
android:paddingRight="20dp"
>
<TextView
android:id="@+id/tv_username_homepage"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_weight="1"
android:fontFamily="@font/poppins_r"
android:text="Welcome, Username"
android:textSize="15sp"
android:textStyle="normal"
android:textColor="@color/white"/>
<ImageView
android:id="@+id/iv_person"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="right"
android:layout_weight="1"
android:paddingLeft="90dp"
android:src="@drawable/baseline_person_24" />
</LinearLayout>
<EditText
android:id="@+id/etSearch"
android:layout_width="match_parent"
android:layout_height="48dp"
android:hint="Search City"
android:imeOptions="actionSearch"
android:inputType="text"
android:layout_margin="16dp"/>
<TextView
android:layout_marginTop="30dp"
android:id="@+id/tv_home"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:fontFamily="@font/poppins_r"
android:text="Home"
android:textSize="20sp"
android:textStyle="bold"
android:textColor="@color/white"/>
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/rv_homepage"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_marginTop="10dp" />
<androidx.recyclerview.widget.RecyclerView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
android:id="@+id/tv_weather"
android:layout_gravity="center"
android:layout_marginTop="10dp"/>
</LinearLayout>
</FrameLayout>
I expect that I can receive the data that can be shown in recycler view
what I just tried is like
Edit:
Welp I Just Figure it out, that using logcat would be very useful, I found that I'm totally wrong when assign the dataclass type there is float type but I assign it using Extension is Int