Android ViewHolder不能强制转换为其他ViewHolder
我的Recyclerview的自定义适配器有问题。 我试图在一个回收器视图中显示两个不同的视图和不同的数据源。 在我做一些改变之前它工作得很好,现在它得到了一个例外 它工作正常,直到notifyDataSetChanged()函数! 当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃 我找了很多,但找不到答案。谢谢你的帮助 我的自定义适配器:Android ViewHolder不能强制转换为其他ViewHolder,android,kotlin,android-recyclerview,custom-adapter,Android,Kotlin,Android Recyclerview,Custom Adapter,我的Recyclerview的自定义适配器有问题。 我试图在一个回收器视图中显示两个不同的视图和不同的数据源。 在我做一些改变之前它工作得很好,现在它得到了一个例外 它工作正常,直到notifyDataSetChanged()函数! 当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃 我找了很多,但找不到答案。谢谢你的帮助 我的自定义适配器: import android.app.Activity import android.
import android.app.Activity
import android.content.Context
import android.content.DialogInterface
import android.content.Intent
import android.os.Vibrator
import android.view.LayoutInflater
import android.view.View
import android.view.ViewGroup
import android.widget.*
import androidx.cardview.widget.CardView
import androidx.recyclerview.widget.RecyclerView
import artemis.team.picandtext.*
import com.squareup.picasso.Picasso
import org.jetbrains.anko.alert
import org.jetbrains.anko.sdk27.coroutines.onClick
import java.util.*
class RecAdapter(
context: Context,
activity: Activity,
grplist: MutableList<GrpsValues> = mutableListOf(),
itemslist: MutableList<ItemsValues> = mutableListOf()
) : RecyclerView.Adapter<RecyclerView.ViewHolder>() {
private var context: Context? = null
private var activity: Activity? = null
private var grpList = mutableListOf<GrpsValues>()
private var itemsList = mutableListOf<ItemsValues>()
private lateinit var tts: TextToSpeech
private var voiceText = "Text not found!"
private var voiceLocale = "en-US"
private var mode = 0
private var itemsSize = 0
private var groupsSize = 0
private val ITEM = 100
private val GROUP = 200
private var HOLDER = GROUP
init {
this.activity = activity
this.context = context
this.grpList = grplist
this.itemsList = itemslist
itemsSize = itemsList.size
groupsSize = grpList.size
}
override fun getItemCount(): Int {
return (itemsList.size + grpList.size)
} // getItemCount
override fun onBindViewHolder(recHolder: RecyclerView.ViewHolder, pos: Int) {
Mode()
if (mode == ITEM && HOLDER == ITEM) {
val holder = recHolder as ItemsViewHolder
if (itemsList.size > 0) {
val position = pos - grpList.size
Picasso.get()
.load(itemsList[position].url)
.error(R.drawable.empty)
.into(holder.image)
holder.title.text = itemsList[position].title
holder.txtID.text = itemsList[position].id
holder.grpID.text = itemsList[position].grpID
val grpID = holder.grpID.text.toString()
lateinit var dialog: DialogInterface
holder.layout.onClick {
(context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
dialog = context!!.alert {
val view = activity!!.layoutInflater.inflate(R.layout.dialog_item_details, null)
val prgDatabase = PrgDatabase(context)
val value = prgDatabase.ItemsFind(holder.txtID.text.toString())
view.findViewById<TextView>(R.id.TxtText).text = value.title
Picasso.get()
.load(value.url)
.error(R.drawable.empty)
.into(view.findViewById<ImageView>(R.id.imageView))
customView = view
}.show()
}
holder.txtDelete.onClick {
(context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
context!!.alert {
title = "ItemsDelete"
message = "ItemsDelete item from the list?"
positiveButton("Yup") {
val prgDatabase = PrgDatabase(context)
prgDatabase.ItemsDelete(holder.txtID.text.toString())
itemsList = PrgDatabase(context).ItemsGetAll(grpID)
this@RecAdapter.notifyDataSetChanged()
it.dismiss()
}
negativeButton("NO") {
it.dismiss()
}
}.show()
}
holder.txtEdit.onClick {
(context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
dialog = context!!.alert {
val view = activity!!.layoutInflater.inflate(R.layout.dialog_items, null)
view.findViewById<Button>(R.id.BTNsaveID).text = "Save changes"
val prgDatabase = PrgDatabase(context)
val adapter = ArrayAdapter<String>(
context!!,
android.R.layout.simple_spinner_item,
context!!.resources.getStringArray(
R.array.langueges
)
)
adapter.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item)
view.findViewById<Spinner>(R.id.SpLangsID).adapter = adapter
val value = prgDatabase.ItemsFind(holder.txtID.text.toString())
view.findViewById<EditText>(R.id.ETtitleID).setText(value.title)
view.findViewById<EditText>(R.id.ETurlID).setText(value.url)
view.findViewById<Spinner>(R.id.SpLangsID).setSelection(
when (value.local) {
"English" -> 0
"Russian" -> 1
"Turkish" -> 2
"French" -> 3
"German" -> 4
"Italian" -> 5
"Korean" -> 6
else -> 0
}
)
view.findViewById<Button>(R.id.BTNsaveID).onClick {
if (view.findViewById<EditText>(R.id.ETtitleID).text.isNotEmpty()
&& view.findViewById<EditText>(R.id.ETurlID).text.isNotEmpty()
) {
val value = ItemsValues(
title = view.findViewById<EditText>(R.id.ETtitleID).text.toString(),
url = view.findViewById<EditText>(R.id.ETurlID).text.toString(),
id = holder.txtID.text.toString(),
grpID = grpID,
local = view.findViewById<Spinner>(R.id.SpLangsID).selectedItem.toString()
)
prgDatabase.ItemsUpdate(
value
)
itemsList[position] = value
dialog.dismiss()
this@RecAdapter.notifyDataSetChanged()
this@RecAdapter.notifyItemChanged(position)
}// if
} // btn on click
customView = view
}.show()
} // on click
}
} // if Items
else if (mode == GROUP && HOLDER == GROUP) {
val holder = recHolder as GrpsViewHolder
if (grpList.size > 0) {
val position = pos
holder.titleGrp.text = grpList[position].title
holder.txtItemIDGrp.text = grpList[position].id
holder.txtParentGrpIDGrp.text = grpList[position].parent
holder.layoutGrp.onClick {
val intent = Intent(context, MainActivity::class.java)
intent.putExtra("GrpID", holder.txtItemIDGrp.text.toString())
intent.putExtra("GrpTitle", grpList[position].title)
activity!!.startActivity(intent)
}
holder.txtDeleteGrp.onClick {
(context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
context!!.alert {
title = "Delete"
message = """
Delete Group from the list?
* With delete this group, all it's words and pictures will be removed!
""".trimIndent()
positiveButton("Yup") {
val prgDatabase = PrgDatabase(context)
prgDatabase.GrpsDelete(holder.txtItemIDGrp.text.toString())
grpList = PrgDatabase(context).GrpsGetAll(grpList[position].parent)
this@RecAdapter.notifyItemRemoved(position)
this@RecAdapter.notifyDataSetChanged()
it.dismiss()
}
negativeButton("NO") {
it.dismiss()
}
}.show()
}
lateinit var dialog: DialogInterface
holder.txtEditGrp.onClick {
(context!!.getSystemService(Context.VIBRATOR_SERVICE) as Vibrator?)!!.vibrate(30)
dialog = context!!.alert {
val view = activity!!.layoutInflater.inflate(R.layout.dialog_grps, null)
view.findViewById<Button>(R.id.BTNsaveID).text = "Save changes"
val prgDatabase = PrgDatabase(context)
val value = prgDatabase.GrpsFind(holder.txtItemIDGrp.text.toString())
view.findViewById<EditText>(R.id.ETtitleID).setText(value.title)
view.findViewById<Button>(R.id.BTNsaveID).onClick {
if (view.findViewById<EditText>(R.id.ETtitleID).text.isNotEmpty()) {
val value = GrpsValues(
title = view.findViewById<EditText>(R.id.ETtitleID).text.toString(),
id = holder.txtItemIDGrp.text.toString(),
parent = holder.txtParentGrpIDGrp.text.toString()
)
prgDatabase.GrpsUpdate(
value
)
grpList[position] = value
dialog.dismiss()
this@RecAdapter.notifyDataSetChanged()
this@RecAdapter.notifyItemChanged(position)
}// if
} // btn on click
customView = view
}.show()
} // on click
} // if
} // GROUP
} // onBindViewHolder
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): RecyclerView.ViewHolder {
if (groupsSize > 0) {
HOLDER = GROUP
return GrpsViewHolder(
LayoutInflater.from(parent.context).inflate(
R.layout.custom_rec_grps_layout,
parent,
false
)
)
} else {
HOLDER = ITEM
return ItemsViewHolder(
LayoutInflater.from(parent.context).inflate(
R.layout.custom_rec_items_layout,
parent,
false
)
)
}
}
inner class ItemsViewHolder(view: View) : RecyclerView.ViewHolder(view) {
val image: ImageView = view.findViewById(R.id.imageView)
val title: TextView = view.findViewById(R.id.TxtText)
val txtID: TextView = view.findViewById(R.id.TxtItemID)
val grpID: TextView = view.findViewById(R.id.TxtGrpID)
val txtDelete: TextView = view.findViewById(R.id.TxtDeleteID)
val txtEdit: TextView = view.findViewById(R.id.TxtEditID)
val layout: CardView = view.findViewById(R.id.ConstItemsID)
}
inner class GrpsViewHolder(view: View) : RecyclerView.ViewHolder(view) {
val titleGrp: TextView = view.findViewById(R.id.GrpTxtText)
val txtItemIDGrp: TextView = view.findViewById(R.id.GrpTxtItemID)
val txtParentGrpIDGrp: TextView = view.findViewById(R.id.GrpTxtGrpID)
val txtDeleteGrp: TextView = view.findViewById(R.id.GrpTxtDeleteID)
val txtEditGrp: TextView = view.findViewById(R.id.GrpTxtEditID)
val layoutGrp: CardView = view.findViewById(R.id.CnstGrpsID)
}
override fun getItemViewType(position: Int): Int {
return position
}
fun Mode() {
if (groupsSize > 0) {
groupsSize--
mode = GROUP
} else if (itemsSize > 0) {
itemsSize--
mode = ITEM
}
}
}
不能将ViewHolder类型的一个子类强制转换为ViewHolder类的另一个类型,因为它们不是同一类型
ViewHolder
/\
RecHolder ItemsViewHolder
RecHolder
是一个视图持有者,ItemsViewHolder
是一个视图持有者,但仅仅因为它们共享同一个父对象,并不能使RecHolder
成为ItemsViewHolder
,所以演员阵容永远不会成功
检查您的逻辑以确定您的onCreateViewHolder
返回的ViewHolder,因为这是进入onBindViewHolder的ViewHolder
类型
如果你想要一种方法,在你投下每一个之前,在你的逻辑内部进行双重检查,那么做If(viewHolder是RecHolder)
或If(viewHolder是ItemViewHolder
)来验证你得到的是你认为得到的那一个。你能为GrpsViewHolder
@BlackHatSamurai发布你的代码吗。在getItemViewType函数之前。在notifyDataSetChanged()函数之前,它工作正常!当holder.txtDelete.onClick或holder.txtEdit.onClick函数启动时,将发生崩溃!同一类演员例外?是的!现在,我改变了逻辑,正如你所说,现在它的工作没有崩溃!但是当删除一个组时,它会完全从db中删除,但在recyclerview中,它会删除最后一项。关闭并重新打开应用程序后显示正常。这是因为在更新数据库后,您可能没有填充recyclerview的数据。您正在将itemsList声明为private`private var itemsList=mutableListOf()`并且您没有在任何地方使用该列表中的项的方法。这意味着您只在每次重新创建适配器时获得新数据。`init{this.activity=activity this.context=context this.grpList=grpList this.itemsList=itemsList}`您还可以不必分配这些作为参数传入的值,通过将var
关键字添加到构造函数类RecAdapter(var-context:context,var-activity:activity
ViewHolder
/\
RecHolder ItemsViewHolder