- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我目前正在开发一个项目,用户将输入动态编译和执行的 C# 脚本。我现在将跳过提供更多细节,因为稍后我可能会对项目的其他方面有疑问。
由于一些我无法控制的高层决策,用户界面将使用 WPF 开发,Quantum Whale Editor.NET 控件将用作编辑器。可悲的是,QWhale Editor.NET 的 WPF 版本似乎还没有完全成熟,因此它缺乏文档,最糟糕的是,它似乎并不友好。
虽然我对 WPF 还是个新手,但我对 MVVM 比较熟悉,所以我很乐意应用它。然而,当我测试评估版并尝试将编辑器的文本绑定(bind)到模型的属性时,我遇到了第一个挑战,并收到一个不可能的异常:
A 'Binding' cannot be set on the 'Text' property of type 'TextEditor'. A 'Binding' can only be set on a DependencyProperty of a DependencyObject.
在检查备选方案时尝试了 AvalonEdit,我想起了另一个 Stack Overflow 问题:Making AvalonEdit MVVM Compatible .所以我遵循了相同的概念。
我已经定义了一个继承自编辑器的类,添加了一个依赖属性,并首先尝试使用 new
将其与隐藏编辑器的原始 Text
属性结合起来。但显然这是一个远景,我的属性没有被使用,直接调用了 base 属性。
失败后,我定义了一个名为 DocumentText
的全新属性。我让它包装了 base.Text
,使用它定义了绑定(bind),因此得到了绑定(bind)工作的一个方向。那就是从模型到控件。但根据我的发现,在另一个方向上进行绑定(bind)的最佳方法是 override
OnTextChanged
事件(或等效事件)以使其引发属性更改通知.问题是控件没有这样的事件,听起来很奇怪。
现在我可能会重写一堆其他事件(例如 OnKeyUp、OnMouseClick 等),这样我就可以处理所有可能修改文本的操作(键入、拖放、粘贴等),但那样做似乎不太实用,并且可能无法重复我以后可能有兴趣绑定(bind)的其他属性。在我有空的时候在网上搜索了几天之后,我仍然没有找到其他想法。那么,除了深入研究控件本身的代码之外,我的问题是否有任何适当的解决方案?(据推测,许可证将使我能够访问源代码,但我宁愿避免对其进行直接修改)。
我避免在问题标题和标签中指定 QWhale 编辑器,因为我觉得我正在寻找的内容不依赖于这个特定的控件。如果我错了,请纠正我。
由于在不同的计算机上,我目前无法提供我的测试代码,但如果您认为有必要,请给我留言,我会添加。
更新:这是代码,因为我不确定我是否能够清楚地描述我的问题。
class ExtenEdit : TextEditor
{
public static DependencyProperty TextProperty =
DependencyProperty.Register("Text", typeof(string), typeof(ExtenEdit),
new PropertyMetadata((obj, args) =>
{
TextEditor target = (TextEditor)obj;
target.DocumentText = (string)args.NewValue;
})
);
public new string Text
{
get { return base.Text; }
set
{
if (base.Text != value)
{
base.Text = value;
}
}
}
}
这在我修改 ViewModel
中的值时有效,但在编辑器中键入时,我的属性被绕过并直接调用基本属性。如果我添加这样的东西,它会起作用:
protected override void OnKeyUp(System.Windows.Input.KeyEventArgs e)
{
SetCurrentValue(TextProperty, base.Text);
base.OnKeyUp(e);
}
但正如我上面所说,我不能将其视为有效的“干净”解决方案。
下面是我的 XAML 中的相关部分(您会注意到我也在玩 AvalonDock):
命名空间:
<Window x:Class="AvalonDockQWhale.MainWindow"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:avalonDock="http://avalondock.codeplex.com"
xmlns:converter="clr-namespace:AvalonDockQWhale.Converter"
xmlns:pane="clr-namespace:AvalonDockQWhale.View.Pane"
xmlns:editor="clr-namespace:QWhale.Editor.Wpf;assembly=QWhale.Editor.Wpf"
xmlns:control="clr-namespace:AvalonDockQWhale.Control"
xmlns:controlHelper="clr-namespace:AvalonDockQWhale.ControlHelper"
x:Name="mainWindow"
Title="MainWindow" Height="600" Width="800">
以及我最初尝试的绑定(bind):
<control:ExtenEdit Text="{Binding Path=ScriptText, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
我为 DHN 的解决方案尝试过的绑定(bind):
<editor:TextEditor controlHelper:AttachedProperties.Text="{Binding ScriptText}" />
和
<editor:TextEditor controlHelper:AttachedProperties.Text="{Binding Path=ScriptText, Mode=TwoWay, UpdateSourceTrigger=PropertyChanged}" />
最终细节:我已经接受了 DHN 给出的答案,尽管它在我的具体情况下不起作用,因为它似乎是解决类似问题的一个不错的解决方案。
最佳答案
Here是我昨天为了帮助另一个 SOler 而写的东西。但它也可能适合您的需求。我通过 DependencyProperty
“扩展”了一个 Window
,因此它的 DialogResult
属性是可绑定(bind)的。
开始
public class AttachedProperties : DependencyObject //adds a bindable DialogResult to window
{
public static readonly DependencyProperty DialogResultProperty =
DependencyProperty.RegisterAttached("DialogResult", typeof(bool?), typeof(AttachedProperties),
new PropertyMetaData(default(bool?), OnDialogResultChanged));
public bool? DialogResult
{
get { return (bool?)GetValue(DialogResultProperty); }
set { SetValue(DialogResultProperty, value); }
}
private static void OnDialogResultChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var window = d as Window;
if (window == null)
return;
window.DialogResult = (bool?)e.NewValue;
}
}
现在您可以将 DialogResult 绑定(bind)到 VM 并设置其属性值。设置值后,窗口将关闭。
<!-- Assuming that the VM is bound to the DataContext and the bound VM has a property DialogResult -->
<Window someNs:AttachedProperties.DialogResult={Binding DialogResult} />
结束
恕我直言,如果您只需要一个特定属性,这是提供可绑定(bind)性的好方法。如果有更多属性必须以这种方式改进,那么我会扩展或包装该类。
编辑 - 这是我们生产环境中运行的内容的摘要
<Window x:Class="AC.Frontend.Controls.DialogControl.Dialog"
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
xmlns:DialogControl="clr-namespace:AC.Frontend.Controls.DialogControl"
xmlns:hlp="clr-namespace:AC.Frontend.Helper"
MinHeight="150" MinWidth="300" ResizeMode="NoResize" SizeToContent="WidthAndHeight"
WindowStartupLocation="CenterScreen" Title="{Binding Title}"
hlp:AttachedProperties.DialogResult="{Binding DialogResult}" WindowStyle="ToolWindow" ShowInTaskbar="True"
Language="{Binding UiCulture, Source={StaticResource Strings}}">
<!-- A lot more stuff here -->
</Window>
如您所见,我先声明命名空间 xmlns:hlp="clr-namespace:AC.Frontend.Helper"
,然后再声明绑定(bind) hlp:AttachedProperties.DialogResult= “{绑定(bind) DialogResult}”
。
AttachedProperty
看起来像这样。这与我之前发布的不一样,但恕我直言,这应该没有任何区别。
public class AttachedProperties
{
#region DialogResult
public static readonly DependencyProperty DialogResultProperty =
DependencyProperty.RegisterAttached("DialogResult", typeof (bool?), typeof (AttachedProperties), new PropertyMetadata(default(bool?), OnDialogResultChanged));
private static void OnDialogResultChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
{
var wnd = d as Window;
if (wnd == null)
return;
wnd.DialogResult = (bool?) e.NewValue;
}
public static bool? GetDialogResult(DependencyObject dp)
{
if (dp == null) throw new ArgumentNullException("dp");
return (bool?)dp.GetValue(DialogResultProperty);
}
public static void SetDialogResult(DependencyObject dp, object value)
{
if (dp == null) throw new ArgumentNullException("dp");
dp.SetValue(DialogResultProperty, value);
}
#endregion
}
关于c# - 扩展 WPF 第三方控件以启用对现有属性的双向绑定(bind),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/16180353/
我有一个 foo 类,它有一个 bar 方法,它接受可调用的东西(函数指针/仿函数)。这个可调用的东西应该作为绑定(bind)元素传递给另一个方法 doit 和第三个方法 bar_cb 方法。 #in
我正在尝试在我的 WPF 4.0 应用程序(使用 VS 2010 Pro RTM)中创建自定义 TabItem 模板/样式,但尽管一切似乎都正常工作,但我注意到跟踪窗口中存在绑定(bind)错误。 我
作为一名刚接触 Android 的开发人员,我想我可能误解了绑定(bind)服务。 我创建了一项服务来结束对服务器的访问。作为此服务的一部分,该服务正在监听多播地址,以识别本地网络上的设备何时出现和消
这个问题在这里已经有了答案: What is the use of the JavaScript 'bind' method? (23 个回答) 关闭 7 年前。 所以我一直在尝试了解一些 JS 上
我不明白这三种语法之间的区别: where a = f (b) do a <- f (b) do let a = f (b) 我确实明白了a <- f(b)与其他两个不同,在大多数情况下,我尝试了所有
我在将 Cocoa 项目从手动同步接口(interface)模型转换为绑定(bind)模型时遇到问题,这样我就不必担心接口(interface)粘合代码。 我关注了 CocoaDevCentral C
我正在尝试找出一种好的方法来对处理大数据集的代码进行并行化,然后将结果数据导入 RavenDb。 数据处理受 CPU 限制和数据库导入 IO 限制。 我正在寻找一种解决方案,以对 Environmen
我正在 foreach 循环中生成单选按钮。我试图将选中的属性绑定(bind)到父级中的基本可观察值。不幸的是,当单击单选按钮时,父级的属性似乎没有在单击处理程序中更新。 基于一些previous w
在我的 Windows Phone 应用程序中,我有两个 LongListSelectors并排在页面上。我想做到这一点,以便当用户滚动其中一个时,另一个滚动相同的量。 两个 LongListSele
我在网上看到这个问题准备面试: Given a non-preemptive kernel which type of process will get affected morein terms o
我有一个 foreach 绑定(bind),如下所示: Summary Permitting 原因是有两个选项卡始终存在,并且我根据是否添加了其他选项卡来添加其他选项
任何人都有绑定(bind)相同的情况DataContext到 TextBlock 中的 Text 属性(例如)。 我必须分配 DataContext以我的风格反射(reflect)基于 Datacon
给定以下代码: Login 和下面的javascript $(function () { $('#btnLogin').click(function () { co
我使用 boost::asio 创建了一个服务器。我在绑定(bind)到端点时遇到问题。所以,如果我在构造函数中初始化一个接受器: Server::Server(QWidget *parent) :
我正在将现有项目从 MySQL 转换为 Postgres。代码中有相当多的原始 SQL 文字使用 ? 作为占位符,例如 SELECT id FROM users WHERE
似乎在绑定(bind)某些数据时出错了,有人可以帮我解决我哪里出错了,尽管我无法弄清楚。 真的不需要在这里显示太多,这是 Binding,我已经通过移除背景并在其中放置颜色来测试背景,效果很好。 编辑
我正在尝试使用 wcf 构建一个 http 监听器(web 服务)。这个监听器是一个更大的桌面应用程序的一部分。此桌面应用程序还会调用 http 监听器。 当监听器接收到数据时,它应该被传递到桌面应用
嘿嘿。 我正在使用 Node.JS 和 child_process 来生成 bash 进程。我试图了解我是否正在执行 I/O 绑定(bind)、CPU 绑定(bind)或两者兼而有之。 我正在使用 p
尝试执行以下操作并出现“Got interpolation ({{}}) where expression was expected”错误。 {{item.name}} 谢谢!
我有一个导入的 Java 库,它是我解决方案中的“绑定(bind)库”项目。 我正在尝试从解决方案中的另一个项目绑定(bind)到第 3 方库中的服务。 第 3 方库文档 [在 java 中] 非常简
我是一名优秀的程序员,十分优秀!