Xamarin Android 与 Mvvmcross。在 axml 中绑定 OnEditorAction
Xamarin Android with Mvvmcross. Binding an OnEditorAction in axml
情况:我有一个包含 EditText 的视图列表。我希望用户能够修改文本,并且只在他按下键盘上的完成按钮时才将他的新文本发送到视图模型。我的编辑文本绑定了 follow
<EditText
android:id="@+id/textNumero"
android:layout_width="0dp"
android:layout_height="fill_parent"
android:layout_weight="0.23"
android:textColor="#ffffffff"
local:MvxBind="Text BarilId" />
我在互联网上发现我可以使用这个事件来做我想做的事:
idEditText.OnEditorAction (ImeAction.Done) += //insert delegate here
不幸的是,我无法访问我的 activity,因为我正在绑定列表中的 class。
所以,我想像这样在 class 中绑定一个命令:
<EditText
android:id="@+id/textNumero"
android:layout_width="0dp"
android:layout_height="fill_parent"
android:layout_weight="0.23"
android:textColor="#ffffffff"
local:MvxBind="OnEditorAction EditCommand"
local:MvxBind="Text BarilId" />
命令:
private IMvxCommand _editCommand;
public IMvxCommand EditCommand
{
get {
_editCommand = _editCommand ?? new MvxCommand(() => {
//do validation here
});
return _editCommand;
}
}
但我不知道如何将 ImeAction.Done
传递给我的命令,或者我是否收到类似的东西。
剩下的部分我能帮忙吗?
您可以创建具有命令的自定义 EditText,然后可以将其绑定到 axml 中。
另一个选项是使用:
idEditText.OnEditorAction (ImeAction.Done) += (ViewModel as MyViewModel).EditCommand.Execute(whateveryouwanttouse);
要达到 Activity,您可以使用:
Mvx.Resolve<IMvxAndroidCurrentTopActivity>().Activity
最后,我在这里使用 Stuart 的答案:
我只是稍微更改了代码,这样我就可以在 SubscibeToEvents() 中使用我的 OnEditorAction 并更改句柄函数。整个 class 看起来像这样:
public class MvxEditTextEditForBinding: MvxAndroidTargetBinding
{
protected EditText EditText
{
get { return (EditText)Target; }
}
private bool _subscribed;
public MvxEditTextEditForBinding(EditText view)
: base(view)
{
}
protected override void SetValueImpl(object target, object value)
{
var editText = EditText;
if (editText == null)
return;
value = value ?? string.Empty;
editText.Text = value.ToString();
}
public override MvxBindingMode DefaultMode
{
get { return MvxBindingMode.TwoWay; }
}
public override void SubscribeToEvents()
{
var editText = EditText;
if (editText == null)
return;
editText.EditorAction += HandleEditForChange;
_subscribed = true;
}
private void HandleEditForChange(object sender, TextView.EditorActionEventArgs e)
{
var editText = EditText;
InputMethodManager imm = (InputMethodManager)editText.Context.GetSystemService(Context.InputMethodService);
imm.HideSoftInputFromWindow(editText.WindowToken, 0);
e.Handled = false;
if (e.ActionId == ImeAction.Done)
{
FireValueChanged(editText.Text);
e.Handled = true;
}
}
public override Type TargetType
{
get { return typeof(string); }
}
protected override void Dispose(bool isDisposing)
{
if (isDisposing)
{
var editText = EditText;
if (editText != null && _subscribed)
{
editText.EditorAction -= HandleEditForChange;
_subscribed = false;
}
}
base.Dispose(isDisposing);
}
}
}
情况:我有一个包含 EditText 的视图列表。我希望用户能够修改文本,并且只在他按下键盘上的完成按钮时才将他的新文本发送到视图模型。我的编辑文本绑定了 follow
<EditText
android:id="@+id/textNumero"
android:layout_width="0dp"
android:layout_height="fill_parent"
android:layout_weight="0.23"
android:textColor="#ffffffff"
local:MvxBind="Text BarilId" />
我在互联网上发现我可以使用这个事件来做我想做的事:
idEditText.OnEditorAction (ImeAction.Done) += //insert delegate here
不幸的是,我无法访问我的 activity,因为我正在绑定列表中的 class。
所以,我想像这样在 class 中绑定一个命令:
<EditText
android:id="@+id/textNumero"
android:layout_width="0dp"
android:layout_height="fill_parent"
android:layout_weight="0.23"
android:textColor="#ffffffff"
local:MvxBind="OnEditorAction EditCommand"
local:MvxBind="Text BarilId" />
命令:
private IMvxCommand _editCommand;
public IMvxCommand EditCommand
{
get {
_editCommand = _editCommand ?? new MvxCommand(() => {
//do validation here
});
return _editCommand;
}
}
但我不知道如何将 ImeAction.Done
传递给我的命令,或者我是否收到类似的东西。
剩下的部分我能帮忙吗?
您可以创建具有命令的自定义 EditText,然后可以将其绑定到 axml 中。
另一个选项是使用:
idEditText.OnEditorAction (ImeAction.Done) += (ViewModel as MyViewModel).EditCommand.Execute(whateveryouwanttouse);
要达到 Activity,您可以使用:
Mvx.Resolve<IMvxAndroidCurrentTopActivity>().Activity
最后,我在这里使用 Stuart 的答案:
我只是稍微更改了代码,这样我就可以在 SubscibeToEvents() 中使用我的 OnEditorAction 并更改句柄函数。整个 class 看起来像这样:
public class MvxEditTextEditForBinding: MvxAndroidTargetBinding
{
protected EditText EditText
{
get { return (EditText)Target; }
}
private bool _subscribed;
public MvxEditTextEditForBinding(EditText view)
: base(view)
{
}
protected override void SetValueImpl(object target, object value)
{
var editText = EditText;
if (editText == null)
return;
value = value ?? string.Empty;
editText.Text = value.ToString();
}
public override MvxBindingMode DefaultMode
{
get { return MvxBindingMode.TwoWay; }
}
public override void SubscribeToEvents()
{
var editText = EditText;
if (editText == null)
return;
editText.EditorAction += HandleEditForChange;
_subscribed = true;
}
private void HandleEditForChange(object sender, TextView.EditorActionEventArgs e)
{
var editText = EditText;
InputMethodManager imm = (InputMethodManager)editText.Context.GetSystemService(Context.InputMethodService);
imm.HideSoftInputFromWindow(editText.WindowToken, 0);
e.Handled = false;
if (e.ActionId == ImeAction.Done)
{
FireValueChanged(editText.Text);
e.Handled = true;
}
}
public override Type TargetType
{
get { return typeof(string); }
}
protected override void Dispose(bool isDisposing)
{
if (isDisposing)
{
var editText = EditText;
if (editText != null && _subscribed)
{
editText.EditorAction -= HandleEditForChange;
_subscribed = false;
}
}
base.Dispose(isDisposing);
}
}
}