在此屏幕中,您可以添加和删除人员。并非所有字段都是必需的,因此单元格高度是动态的。
如果我首先添加一个完成了所有字段的人,那么第二个人并非所有字段都已完成,然后我删除了第一个人,第二个人占据了第一个人但是第一个人的布局
删除第一个人后,如果我向上滚动直到剩余的单元格在屏幕外,那么它就会自行修复
这是表格来源
namespace xXxx.xXxx.iOS
{
public class SiniestroParticipantesSource : MvxTableViewSource
{
private readonly SiniestroParticipantesViewModel viewModel;
public SiniestroParticipantesSource(UITableView tableView, SiniestroParticipantesViewModel viewModel)
: base(tableView)
{
this.UseAnimations = true;
this.AddAnimation = UITableViewRowAnimation.Top;
this.RemoveAnimation = UITableViewRowAnimation.Middle;
this.viewModel = viewModel;
tableView.RegisterNibForCellReuse(UINib.FromName(PersonaDenunciaCellView.Key, NSBundle.MainBundle), PersonaDenunciaCellView.Key);
}
public override void RowSelected(UITableView tableView, NSIndexPath indexPath)
{
tableView.DeselectRow(indexPath, true);
var itemPersona = this.viewModel.Personas[indexPath.Section];
this.viewModel.AgregarCommand.Execute(itemPersona);
}
protected override UITableViewCell GetOrCreateCellFor(UITableView tableView, NSIndexPath indexPath, object item)
{
var cell = (PersonaDenunciaCellView)tableView.DequeueReusableCell(PersonaDenunciaCellView.Key, indexPath);
return cell;
}
public override nint RowsInSection(UITableView tableview, nint section)
{
return 1;
}
public override nint NumberOfSections(UITableView tableView)
{
return this.viewModel.Personas.Count;
}
protected override object GetItemAt(NSIndexPath indexPath)
{
return this.viewModel.Personas[indexPath.Section];
}
}
}
这是单元格视图
namespace xXxx.xXxx.iOS
{
public partial class PersonaDenunciaCellView : MvxTableViewCell
{
public static readonly NSString Key = new NSString("PersonaDenunciaCellView");
public static readonly UINib Nib;
static PersonaDenunciaCellView()
{
Nib = UINib.FromName("PersonaDenunciaCellView", NSBundle.MainBundle);
}
protected PersonaDenunciaCellView(IntPtr handle) : base(handle)
{
this.DelayBind(() =>
{
var set = this.CreateBindingSet<PersonaDenunciaCellView, PersonaDenunciaItemViewModel>();
set.Bind(btnRemove).To(vm => vm.RemoveCommand);
set.Bind(lblNombre).To(vm => vm.Persona.Persona.Nombre);
set.Bind(lblTipoDoc).To(vm => vm.Persona.Persona.TipoDoc.Descripcion);
set.Bind(tipoDocVisibilityConst).For("Priority").To(vm => vm.Persona.Persona.Nrodoc).WithConversion("iOSVisibility", true);
set.Bind(lblNrodoc).To(vm => vm.Persona.Persona.Nrodoc);
set.Bind(nroDocVisibilityConst).For("Priority").To(vm => vm.Persona.Persona.Nrodoc).WithConversion("iOSVisibility", true);
set.Bind(lblMailContacto).To(vm => vm.Persona.MailContacto);
set.Bind(mailContactoVisibilityConst).For("Priority").To(vm => vm.Persona.MailContacto).WithConversion("iOSVisibility", true);
set.Bind(lblTelContacto).To(vm => vm.Persona.TelContacto);
set.Bind(telContactoVisibilityConst).For("Priority").To(vm => vm.Persona.TelContacto).WithConversion("iOSVisibility", true);
set.Bind(lblLesionado).To(vm => vm.Persona.Lesionado).WithConversion("Lesionado");
set.Bind(vehiculoVisibilityConst).For("Priority").To(vm => vm.Persona.Patente).WithConversion("iOSVisibility", true);
set.Bind(viewVehiculo).For("Visibility").To(vm => vm.Persona.Patente).WithConversion("Visibility");
set.Bind(lblPatente).To(vm => vm.Persona.Patente);
set.Bind(lblCiaSeguroDesc).To(vm => vm.Persona.CiaSeguroDesc);
set.Apply();
});
}
}
}
Cell ViewModel
namespace xXxx.xXxx.Core.ViewModels.Items
{
public class PersonaDenunciaItemViewModel:MvxViewModel
{
private readonly IMvxMessenger messenger;
private readonly IUserInteraction userInteraction;
public string Index { get; set; }
public PersonaDenunciaItemViewModel (SiniestroCarga.PersonaDenuncia persona, IMvxMessenger messenger, IUserInteraction userInteraction, string index)
{
this.messenger = messenger;
this.userInteraction = userInteraction;
this.Persona = persona;
this.Index = index;
}
private SiniestroCarga.PersonaDenuncia persona;
public SiniestroCarga.PersonaDenuncia Persona
{
get
{
return this.persona;
}
set
{
this.persona = value;
this.RaisePropertyChanged(() => this.Persona);
}
}
private ICommand removeCommand;
public ICommand RemoveCommand
{
get
{
return this.removeCommand = this.removeCommand ?? new MvxCommand(this.RemovePersona);
}
}
private void RemovePersona()
{
this.userInteraction.Confirm("Are you sure?", () =>
{
this.messenger.Publish(new RemovePersonaDenunciaMessage(this, this.Persona, this.Index));
}, null, "OK", "Cancel");
}
}
}
最后一个messenger.Publish在另一个ViewModel
上订阅了这个OnRemovePersonaDenuncia = message =>
{
var listPersonas = new ObservableCollection<PersonaDenunciaItemViewModel>(this.Personas);
listPersonas.Remove(this.Personas.First(p => p.Index == message.Index));
this.Personas = listPersonas;
}
将RemoveCommand的实现更改为此
this.Personas.Remove(this.Personas.First(p => p.Index == message.Index));
这样做,当我按下删除按钮时,模拟器关闭没有任何错误。 Xamarin Studio上的应用程序跟踪显示了这个
***断言失败 - [UITableView _endCellAnimationsWithContext:],/ BuildRoot / Library / Cache / com.apple.xbs / Source / UIKit_Sim / UIKit-3512.60.7 / UITableView.m:1700
答案 0 :(得分:1)
当内容高度为动态时,iOS无法正确计算单元格的高度。这非常烦人,但您可以覆盖GetHeightForRow
中的EstimatedHeight
和SiniestroParticipantesSource
来计算每个单元格的确切高度,具体取决于数据:
public override nfloat EstimatedHeight(UITableView tableView, NSIndexPath indexPath) =>
GetHeightForRow(tableView, indexPath);
public override nfloat GetHeightForRow(UITableView tableView, NSIndexPath indexPath)
{
var data = (PersonaDenunciaItemViewModel)ItemsSource.ElementAt(indexPath.Row);
var cell = (PersonaDenunciaCellView)tableView.DequeueReusableCell(PersonaDenunciaCellView.Key, indexPath);
// TODO set the cell data manually (ignoring bindings)
// i.e: **cell.lblNombre = data.Persona.Persona.Nombre**; // and every other field required
cell.SetNeedsLayout();
cell.LayoutIfNeeded();
var size = cell.ContentView.SystemLayoutSizeFittingSize(UIView.UILayoutFittingCompressedSize);
return NMath.Ceiling(size.Height) + 1;
}
这是使用相同代码的example。