Android 如何从Kotlin中的ViewModel将URL中的图像动态加载到ImageView中

Android 如何从Kotlin中的ViewModel将URL中的图像动态加载到ImageView中,android,kotlin,imageview,picasso,android-viewmodel,Android,Kotlin,Imageview,Picasso,Android Viewmodel,我正在编写一个应用程序来显示NHL分数,并希望在RecyclerView中的每个团队都在其旁边有自己的徽标。我可以请求一个带有团队ID的URL,该URL将返回团队徽标的高分辨率图像。我正在尝试这样做,以便我可以在我的viewModel中加载图像并在视图中设置它们,就像我在处理团队名称、当前分数等事情一样 我曾尝试使用毕加索来实现这一点,但它需要一个viewModel没有的上下文,并且viewModel无法直接访问imageView来更改它。那么,如何加载图像并通过数据绑定或其他方式公开它们,以允

我正在编写一个应用程序来显示NHL分数,并希望在RecyclerView中的每个团队都在其旁边有自己的徽标。我可以请求一个带有团队ID的URL,该URL将返回团队徽标的高分辨率图像。我正在尝试这样做,以便我可以在我的viewModel中加载图像并在视图中设置它们,就像我在处理团队名称、当前分数等事情一样

我曾尝试使用毕加索来实现这一点,但它需要一个viewModel没有的上下文,并且viewModel无法直接访问imageView来更改它。那么,如何加载图像并通过数据绑定或其他方式公开它们,以允许视图显示它们呢

以下是我的主要活动:

class MainActivity : AppCompatActivity() {

    private lateinit var binding: ActivityMainBinding
    private lateinit var viewModel: GameListViewModel
    private var errorSnackbar: Snackbar? = null

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)

        binding = DataBindingUtil.setContentView(this, R.layout.activity_main)
        binding.lifecycleOwner = this
        binding.gameList.layoutManager = LinearLayoutManager(this, LinearLayoutManager.VERTICAL, false)

        viewModel = ViewModelProviders.of(this).get(GameListViewModel::class.java)
        viewModel.errorMessage.observe(this, Observer { errorMessage ->
            if (errorMessage != null)
                showError(errorMessage)
            else
                hideError()
        })
        binding.viewModel = viewModel
    }

    private fun showError(@StringRes errorMessage:Int) {
        errorSnackbar = Snackbar.make(binding.root, errorMessage, Snackbar.LENGTH_INDEFINITE)
        errorSnackbar?.setAction(R.string.retry, viewModel.errorClickListener)
        errorSnackbar?.show()
    }

    private fun hideError() {
        errorSnackbar?.dismiss()
    }
}
视图模型:

class GameViewModel:BaseViewModel() {
    private val awayTeamName = MutableLiveData<String>()
    private val homeTeamName = MutableLiveData<String>()
    private val awayTeamScore = MutableLiveData<String>()
    private val homeTeamScore = MutableLiveData<String>()
    private val timeRemaining = MutableLiveData<String>()

    fun bind(response: Game) {
        awayTeamName.value = response.gameData.teams.away.name
        homeTeamName.value = response.gameData.teams.home.name
        awayTeamScore.value = response.liveData.linescore.teams["away"]?.goals.toString()
        homeTeamScore.value = response.liveData.linescore.teams["home"]?.goals.toString()

        if (response.gameData.status.detailedState == "Scheduled") {
            val parser = SimpleDateFormat("yyyy-MM-dd'T'HH:mm:ss", Locale.getDefault())
            parser.timeZone = TimeZone.getTimeZone("UTC")
            val formatter = SimpleDateFormat("hh:mm a", Locale.getDefault())
            formatter.timeZone = TimeZone.getDefault()
            timeRemaining.value = formatter.format(parser.parse(response.gameData.datetime.dateTime))
        } else {
            timeRemaining.value = response.liveData.linescore.currentPeriodTimeRemaining + " " + response.liveData.linescore.currentPeriodOrdinal
        }
    }

    fun getAwayTeamName(): MutableLiveData<String> {
        return awayTeamName
    }

    fun getHomeTeamName(): MutableLiveData<String> {
        return homeTeamName
    }

    fun getAwayTeamScore(): MutableLiveData<String> {
        return awayTeamScore
    }

    fun getHomeTeamScore(): MutableLiveData<String> {
        return homeTeamScore
    }

    fun getTimeRemaining(): MutableLiveData<String> {
        return timeRemaining
    }
}
class GameViewModel:BaseViewModel(){
private val awayTeamName=MutableLiveData()
private val homeTeamName=MutableLiveData()
private val awayTeamScore=MutableLiveData()
私有val homeTeamScore=MutableLiveData()
private val timeRemaining=MutableLiveData()
趣味绑定(回应:游戏){
awayTeamName.value=response.gameData.teams.away.name
homeTeamName.value=response.gameData.teams.home.name
awayTeamScore.value=response.liveData.linescore.teams[“客场”]?.goals.toString()
homeTeamScore.value=response.liveData.linescore.teams[“home”]?.goals.toString()
如果(response.gameData.status.detailedState==“计划”){
val parser=SimpleDateFormat(“yyyy-MM-dd'T'HH:MM:ss”,Locale.getDefault())
parser.timeZone=timeZone.getTimeZone(“UTC”)
val formatter=SimpleDataFormat(“hh:mm a”,Locale.getDefault())
formatter.timeZone=timeZone.getDefault()
timeRemaining.value=formatter.format(parser.parse(response.gameData.datetime.datetime))
}否则{
timeRemaining.value=response.liveData.linescore.currentPeriodTimeRemaining+“”+response.liveData.linescore.currentPeriodOrdinal
}
}
乐趣getAwayTeamName():MutableLiveData{
返回awayTeamName
}
fun getHomeTeamName():MutableLiveData{
返回homeTeamName
}
乐趣getAwayTeamScore():可变LiveData{
返回awayTeamScore
}
fun getHomeTeamScore():MutableLiveData{
返回homeTeamScore
}
fun getTimeRemaining():MutableLiveData{
返回剩余时间
}
}
和recyclerView行的XML:

<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    xmlns:app="http://schemas.android.com/apk/res-auto">

    <data>
        <variable
            name="viewModel"
            type="com.example.nhlstats.ui.game.GameViewModel" />
    </data>
    <LinearLayout
        android:orientation="vertical"
        android:layout_width="match_parent"
        android:layout_height="wrap_content">

        <LinearLayout
            android:id="@+id/awayTeam"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:orientation="horizontal"
            android:layout_marginTop="12dp">

            <ImageView
                android:id="@+id/awayTeamLogo"
                android:layout_height="36dp"
                android:layout_width="0dp"
                android:layout_weight="1"
                tools:src="@drawable/ic_launcher_background"/>

            <TextView
                android:id="@+id/awayTeamName"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="3"
                android:layout_gravity="center_vertical"
                android:text="@{viewModel.awayTeamName}"
                tools:text="CHI Blackhawks"/>

            <TextView
                android:id="@+id/awayScore"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="1"
                android:layout_gravity="center_vertical"
                android:text="@{viewModel.awayTeamScore}"
                tools:text="0"/>
            <TextView

                android:id="@+id/gameTime"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="1"
                android:layout_gravity="center_vertical"
                android:text="@{viewModel.timeRemaining}"
                tools:text="14:26 3rd"/>

        </LinearLayout>

        <LinearLayout
            android:id="@+id/homeTeam"
            android:layout_width="match_parent"
            android:layout_height="wrap_content"
            android:layout_marginTop="12dp"
            android:layout_marginBottom="24dp">

            <ImageView
                android:id="@+id/homeTeamLogo"
                android:layout_height="36dp"
                android:layout_width="0dp"
                android:layout_weight="1"
                tools:src="@drawable/ic_launcher_background"/>

            <TextView
                android:id="@+id/homeTeamName"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_weight="3"
                android:layout_gravity="center_vertical"
                android:text="@{viewModel.homeTeamName}"
                tools:text="CAR Hurricanes"/>

            <TextView
                android:id="@+id/homeScore"
                android:layout_width="0dp"
                android:layout_height="wrap_content"
                android:layout_gravity="center_vertical"
                android:layout_weight="2"
                android:text="@{viewModel.homeTeamScore}"
                tools:text="4"/>
        </LinearLayout>
    </LinearLayout>
</layout>


提前感谢。

I guest您将为每个itemView创建
GameViewModel
,因此在绑定视图持有者时:

您的
GameViewModel
课程

val awayLogoUrl = MutableLiveData<String>()
val homeLogoUrl = MutableLiveData<String>()

fun bind(response: Game) {
    awayLogoUrl.value = response... //set away logo url here
    homeLogoUrl.value = response... //set home logo url here
}
viewModel.awayLogoUrl.observe(this, Observer { 
    it?.let { url ->
        //Show image into itemView using Picasso or Glide
        Glide.with(itemView.context).load(url).into(binding.awayTeamLogo)
    }
})

viewModel.homeLogoUrl.observe(this, Observer { 
    it?.let { url ->
        //Show image into itemView using Picasso or Glide
        Glide.with(itemView.context).load(url).into(binding.homeTeamLogo)
    }
})

我建议您为每个itemView创建
GameViewModel
,因此在绑定视图持有者时:

您的
GameViewModel
课程

val awayLogoUrl = MutableLiveData<String>()
val homeLogoUrl = MutableLiveData<String>()

fun bind(response: Game) {
    awayLogoUrl.value = response... //set away logo url here
    homeLogoUrl.value = response... //set home logo url here
}
viewModel.awayLogoUrl.observe(this, Observer { 
    it?.let { url ->
        //Show image into itemView using Picasso or Glide
        Glide.with(itemView.context).load(url).into(binding.awayTeamLogo)
    }
})

viewModel.homeLogoUrl.observe(this, Observer { 
    it?.let { url ->
        //Show image into itemView using Picasso or Glide
        Glide.with(itemView.context).load(url).into(binding.homeTeamLogo)
    }
})

对于Android架构组件视图模型

将活动上下文传递给活动的ViewModel不是一个好的做法,因为它是内存泄漏。我不支持那件事

您可以在viewmodel中创建图像url观察者,并在视图类(活动或片段)中观察它,如下所示(如Duy Khanh Nguyen所回答):-

但是如果您想使用其他方法,您可以简单地使用
AndroidViewModel
提供的应用程序上下文,您应该扩展
AndroidViewModel
,它只是一个包含
应用程序
引用的
ViewModel
。我会将您的案例写入您的
BaseViewModel
。例如:-

class BaseViewModel(application: Application) : AndroidViewModel(application) {

val context = getApplication<Application>().applicationContext

//... ViewModel methods 

}
class BaseViewModel(应用程序:应用程序):AndroidViewModel(应用程序){
val context=getApplication().applicationContext
//…视图模型方法
}

希望有帮助。如果你需要进一步的帮助,请告诉我。快乐编码

用于Android架构组件视图模型

将活动上下文传递给活动的ViewModel不是一个好的做法,因为它是内存泄漏。我不支持那件事

您可以在viewmodel中创建图像url观察者,并在视图类(活动或片段)中观察它,如下所示(如Duy Khanh Nguyen所回答):-

但是如果您想使用其他方法,您可以简单地使用
AndroidViewModel
提供的应用程序上下文,您应该扩展
AndroidViewModel
,它只是一个包含
应用程序
引用的
ViewModel
。我会将您的案例写入您的
BaseViewModel
。例如:-

class BaseViewModel(application: Application) : AndroidViewModel(application) {

val context = getApplication<Application>().applicationContext

//... ViewModel methods 

}
class BaseViewModel(应用程序:应用程序):AndroidViewModel(应用程序){
val context=getApplication().applicationContext
//…视图模型方法
}

希望有帮助。如果你需要进一步的帮助,请告诉我。快乐编码

使用数据绑定您应该创建一个自定义绑定适配器,如下所示:

  @BindingAdapter("app:imageUri")
    fun loadImageWithUri(imageView: ImageView, imageUri: String){
      Glide.with(imageView.context).load(Uri.parse(imageUri)).into(imageView)
  }
并按如下方式更改您的imageview:

 <androidx.appcompat.widget.AppCompatImageView
  android:layout_height="36dp"
  android:layout_width="0dp"
  android:layout_weight="1"
  app:imageUri="@{viewmodel.teamLogoUri}"/>

使用数据绑定,您应该创建一个自定义绑定适配器,如下所示:

  @BindingAdapter("app:imageUri")
    fun loadImageWithUri(imageView: ImageView, imageUri: String){
      Glide.with(imageView.context).load(Uri.parse(imageUri)).into(imageView)
  }
并按如下方式更改您的imageview:

 <androidx.appcompat.widget.AppCompatImageView
  android:layout_height="36dp"
  android:layout_width="0dp"
  android:layout_weight="1"
  app:imageUri="@{viewmodel.teamLogoUri}"/>


您可以在ViewModel中为url创建一个LiveData变量,并在itemViewhey@cpgreen2中观察它。在VewModel中发出网络请求不是一个好的做法,创建repository类以执行以下工作@Duy Khan Nguyen观察URL如何使图像显示在ImageView中?您可以在下面检查我的答案您可以在ViewModel中为URL创建LiveData变量并在itemViewhey@cpgreen2中观察它在VewModel中发出网络请求不是一个好的做法,创建repository类来完成以下工作@Duy Khan Nguyen观察URL如何使图像显示在ImageView中?您可以在下面查看我的答案谢谢您提供的示例,这很有帮助。如果我有一个单一的imageView,但是