ios - Swift:UICollectionViewCell 中的 UIButton 图像不会改变
问题描述
我是编码新手,所以请原谅我乱七八糟的代码。我正在尝试在点击该按钮时更改该按钮的图像。该按钮位于 UICollectionViewCell 这使得这有点棘手。
这是按钮被添加到单元格的地方:
func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "cell", for: indexPath)
let d = self.data[indexPath.row].image
let button = UIButton(frame: CGRect(x:0, y:0, width:68,height:80))
button.setImage(d, for: UIControl.State.normal)
button.addTarget(self, action: #selector(buttonTapped), for: UIControl.Event.touchUpInside)
cell.addSubview(button)
return cell }
这是按钮从中获取图像的数据数组:
var data: [CustomData] = [CustomData]()
override func viewDidLoad() {
super.viewDidLoad()
self.data.append(CustomData(title: "abc", image: #imageLiteral(resourceName: "bookOne"), url: "typeURLa"))
self.data.append(CustomData(title: "def", image: #imageLiteral(resourceName: "bookTwo"), url: "typeURLb"))
self.data.append(CustomData(title: "ghi", image: #imageLiteral(resourceName: "bookThree"), url: "typeURLc"))
}
这是点击按钮时发生的情况:
@objc func buttonTapped(sender: UIButton!) {
print("button tapped")
let r = data[0]
print(sender.currentImage!)
print(r.image)
sender.setImage(r.image, for: UIControl.State.normal)
collectionView.reloadData()
print(sender.currentImage!)
print(r.image)
}
在调试控制台中,我可以看到它确实更新了按钮图像,但在模拟器中图像保持不变。
button tapped
<UIImage:0x600000adad90 named(main: bookTwo) {600, 754}>
<UIImage:0x600000adafd0 named(main: bookOne) {798, 1210}>
<UIImage:0x600000adafd0 named(main: bookOne) {798, 1210}>
<UIImage:0x600000adafd0 named(main: bookOne) {798, 1210}>
当我点击第 2 行中包含图像“bookTwo”的按钮时,我希望它更改为图像“bookOne”。当我在点击后打印出第 2 行中按钮的图像时,我可以看到它在调试控制台中成功地将“bookTwo”更改为“bookOne”图像。“bookOne”位于数组 data[0] 中,“bookTwo”位于 data[1] 中。但是屏幕上仍然显示图像“bookTwo”。
解决方案
问题是您每次更新单元格后都在重新加载collectionView。每当您重新加载 collectionView 时,都会调用 collectionViewCellForItemAt 方法。但是在collectionViewCellForItemAt 方法中已经设置了将按钮中的图像写入数据数组的indexpath.row 中的图像。由于数据数组未更改,因此单元格被设置为数据数组给出的默认图像。
相反,您可以更新代码。更新后的代码如下所示
@objc func buttonTapped(sender: UIButton!) {
print("button tapped")
let row = sender.tag
let intialData = data[row]
print(sender.currentImage!)
print(data[row].image)
data[row] = CustomData(title: intialData.title, image: #imageLiteral(resourceName: "bookOne"), url: intialData.url)
sender.setImage(data[row].image, for: UIControl.State.normal)
print(sender.currentImage!)
print(data[row].image)
}
在 collectionViewCellForItemAt 添加这一行 button.tag = indexPath.row。更新后的代码如下所示
func collectionView(_ collectionView: UICollectionView, cellForItemAt indexPath: IndexPath) -> UICollectionViewCell {
let cell = collectionView.dequeueReusableCell(withReuseIdentifier: "cell", for: indexPath)
let d = self.data[indexPath.row].image
let button = UIButton(frame: CGRect(x:0, y:0, width:68,height:80))
button.setImage(d, for: UIControl.State.normal)
button.tag = indexPath.row
button.addTarget(self, action: #selector(buttonTapped), for: UIControl.Event.touchUpInside)
cell.addSubview(button)
return cell
}
推荐阅读
- c# - 事件 'control.mousemove' 只能出现在 += 或 -= 的左侧
- android - 在根设备android中安装apk后启动应用程序
- amazon-web-services - AWS Cognito + 自定义 OpenID + AppSync
- linq - Linq left join 和 groug by 并检查空值
- python - 针对布尔列绘制列(条形图)
- c++ - 如何将 Boost 安装到 CLion 中?
- vert.x - 带有 Vert.x 的 QuickFIX/J 服务器
- javascript - 创建一个使用 Vue 和 Typescript 的 Meteor 项目
- javascript - Discord.js Bot 仅在特定时间响应消息或命令
- asp.net-core - Blazor 组件构建