使用两个相似的集合视图单元格,从而避免代码重复
Using two similar collection view cells and thereby avoiding code duplication
我想要两个几乎相似的单元格。区别仅在于多显示一个视图。
因此我想使用自定义构造函数。通常你有一个类似于这个的构造函数:
public class AnimalCell : UICollectionViewCell
{
[Export ("initWithFrame:")]
public AnimalCell (CGRect frame) : base (frame)
{
// do something
}
}
我想传递一个类型,根据这个类型,我想在单元格上显示不同的项目。最好的方法是使用这样的构造函数:
public AnimalCell(MyCustomType type)
{
if(type == XXX){
// add as subview, add constraints, ...
}else{
// normal setup
}
}
我当然想保留单元重用。这可以实现吗?怎么样?
我想到的另一件事是使用子类化。有没有人知道我如何定义 cell 这样我就不必重复相同的代码?例如
var cell = null;
if (MyCustomType == XXX) {
cell = collectionView.DequeueReusableCell (DefaultCell.Key, indexPath) as DefaultCell;
} else {
cell = collectionView.DequeueReusableCell (CustomizedCell.Key, indexPath) as CustomizedCell;
}
cell.DoSomething("someValue"); // this doesn't work because you have to define cell with a certain class
// do some more initialization
我目前的解决方案:
public abstract class DefaultCell : UICollectionViewCell
{
protected bool someVariable;
[Export ("initWithFrame:")]
public DefaultCell (CGRect frame) : base (frame)
{
}
public void DoSomething(string text)
{
label.Text = text;
}
}
public class Custom1Cell : DefaultCell
{
public static readonly NSString Key = new NSString ("Custom1Cell");
[Export ("initWithFrame:")]
public Custom1Cell (CGRect frame) : base (frame)
{
initialize ();
}
private void initialize()
{
// some initialization
}
}
public class Custom2Cell : DefaultCell
{
public static readonly NSString Key = new NSString ("Custom2Cell");
[Export ("initWithFrame:")]
public Custom2Cell (CGRect frame) : base (frame)
{
initialize ();
}
private void initialize()
{
// some initialization
}
public void SomeMethod(string text)
{
if(someVariable)
someOtherLabel.Text = text;
}
}
如 Mahmoud 所描述的出队工作:
DefaultCell cell;
if (type = XXX) {
cell = collectionView.DequeueReusableCell (Custom1Cell.Key, indexPath) as Custom1Cell;
} else {
cell = collectionView.DequeueReusableCell (Custom2Cell.Key, indexPath) as Custom2Cell;
}
cell.DoSomething("works on both cell types");
((Custom2Cell)cell).SomeMethod("works only on one cell type");
您可以添加 UICollectionViewCell 的子类,比方说 YourCell。现在将 YourCell 提供给故事板中的单元格。在 YourCell 中,您可以添加任意数量的视图,并且可以将它们从 Storyboard 单元格输出到 YourCell Class,然后您可以根据需要自定义视图。
更新:
好吧,假设你没有在情节提要中添加那个视图,但你把它作为一个 属性(我只是一个指针,所以不会占用内存 :) 很多)只是在 getter 上做惰性实例化并在此 getter 方法中设置它的约束。这是延迟实例化的示例,我们有一个不在故事板视图中的标签。
-(UILabel *)yourLabel
{
if(!_yourLabel){
_yourLabel=[[UILabel alloc]init];
NSLayoutConstraint *leadingConstraint= [NSLayoutConstraint constraintWithItem:self.contentView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:_yourLabel attribute:NSLayoutAttributeLeading multiplier:1.0 constant:0];
NSLayoutConstraint *topConstraint= [NSLayoutConstraint constraintWithItem:self.contentView attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:_yourLabel attribute:NSLayoutAttributeTop multiplier:1.0 constant:0];
[self.contentView addConstraints:@[leadingConstraint,topConstraint]];
}
return _yourLabel;
}
也许您考虑一个抽象 class 具有共享变量和方法(将其命名为 BaseClass)和另外两个 class 继承自抽象的 classes,并且在代码中您可以将 cell 定义为对象并在 if 语句中初始化它,如下所示:
Object cell;
if (MyCustomType == XXX) {
cell = new class1()
((Class1) cell).MethodInClass1();
} else{
cell = new class2();
((Class2) cell).MethodInClass2();
}
我想要两个几乎相似的单元格。区别仅在于多显示一个视图。
因此我想使用自定义构造函数。通常你有一个类似于这个的构造函数:
public class AnimalCell : UICollectionViewCell
{
[Export ("initWithFrame:")]
public AnimalCell (CGRect frame) : base (frame)
{
// do something
}
}
我想传递一个类型,根据这个类型,我想在单元格上显示不同的项目。最好的方法是使用这样的构造函数:
public AnimalCell(MyCustomType type)
{
if(type == XXX){
// add as subview, add constraints, ...
}else{
// normal setup
}
}
我当然想保留单元重用。这可以实现吗?怎么样?
我想到的另一件事是使用子类化。有没有人知道我如何定义 cell 这样我就不必重复相同的代码?例如
var cell = null;
if (MyCustomType == XXX) {
cell = collectionView.DequeueReusableCell (DefaultCell.Key, indexPath) as DefaultCell;
} else {
cell = collectionView.DequeueReusableCell (CustomizedCell.Key, indexPath) as CustomizedCell;
}
cell.DoSomething("someValue"); // this doesn't work because you have to define cell with a certain class
// do some more initialization
我目前的解决方案:
public abstract class DefaultCell : UICollectionViewCell
{
protected bool someVariable;
[Export ("initWithFrame:")]
public DefaultCell (CGRect frame) : base (frame)
{
}
public void DoSomething(string text)
{
label.Text = text;
}
}
public class Custom1Cell : DefaultCell
{
public static readonly NSString Key = new NSString ("Custom1Cell");
[Export ("initWithFrame:")]
public Custom1Cell (CGRect frame) : base (frame)
{
initialize ();
}
private void initialize()
{
// some initialization
}
}
public class Custom2Cell : DefaultCell
{
public static readonly NSString Key = new NSString ("Custom2Cell");
[Export ("initWithFrame:")]
public Custom2Cell (CGRect frame) : base (frame)
{
initialize ();
}
private void initialize()
{
// some initialization
}
public void SomeMethod(string text)
{
if(someVariable)
someOtherLabel.Text = text;
}
}
如 Mahmoud 所描述的出队工作:
DefaultCell cell;
if (type = XXX) {
cell = collectionView.DequeueReusableCell (Custom1Cell.Key, indexPath) as Custom1Cell;
} else {
cell = collectionView.DequeueReusableCell (Custom2Cell.Key, indexPath) as Custom2Cell;
}
cell.DoSomething("works on both cell types");
((Custom2Cell)cell).SomeMethod("works only on one cell type");
您可以添加 UICollectionViewCell 的子类,比方说 YourCell。现在将 YourCell 提供给故事板中的单元格。在 YourCell 中,您可以添加任意数量的视图,并且可以将它们从 Storyboard 单元格输出到 YourCell Class,然后您可以根据需要自定义视图。
更新: 好吧,假设你没有在情节提要中添加那个视图,但你把它作为一个 属性(我只是一个指针,所以不会占用内存 :) 很多)只是在 getter 上做惰性实例化并在此 getter 方法中设置它的约束。这是延迟实例化的示例,我们有一个不在故事板视图中的标签。
-(UILabel *)yourLabel
{
if(!_yourLabel){
_yourLabel=[[UILabel alloc]init];
NSLayoutConstraint *leadingConstraint= [NSLayoutConstraint constraintWithItem:self.contentView attribute:NSLayoutAttributeLeading relatedBy:NSLayoutRelationEqual toItem:_yourLabel attribute:NSLayoutAttributeLeading multiplier:1.0 constant:0];
NSLayoutConstraint *topConstraint= [NSLayoutConstraint constraintWithItem:self.contentView attribute:NSLayoutAttributeTop relatedBy:NSLayoutRelationEqual toItem:_yourLabel attribute:NSLayoutAttributeTop multiplier:1.0 constant:0];
[self.contentView addConstraints:@[leadingConstraint,topConstraint]];
}
return _yourLabel;
}
也许您考虑一个抽象 class 具有共享变量和方法(将其命名为 BaseClass)和另外两个 class 继承自抽象的 classes,并且在代码中您可以将 cell 定义为对象并在 if 语句中初始化它,如下所示:
Object cell;
if (MyCustomType == XXX) {
cell = new class1()
((Class1) cell).MethodInClass1();
} else{
cell = new class2();
((Class2) cell).MethodInClass2();
}