使用C#对Godot属性进行改写(其实是覆盖)
背景
我还真没想到C#这个鬼东西还能这么用……
起初是想通过某种办法改写一下Label
的text
属性的赋值机制来改写一个自己的Label
运作方式,之前想尽办法用GDScript改写set_text
(实际情况是GDScript下根本不提供这个函数)和_set
(这玩意改完了不对赋值操作起作用,感觉更像是给反射用的)然而并没有什么卵用……虽然说直接调用那些函数的方式还是可以使用的,但是写法上我还是喜欢用直接赋值的来得简洁。
但总之我希望修改对于text
属性赋值时的运作方式,我甚至去想写NativeScript直接把Label重写一下,不过NativeScript在官方文档里的一顿操作猛如虎我愣是没看懂,Native这块我先放着,哪天用得上哪天再研究。(但我估计重写还是能办到的)
好在我下的Godot版本是mono版(虽然官方到了3.3rc6的文档里都说C#绑定得不太完善,不过考虑到随时有可能出现需要C#的情形以及mono版也不耽误写GDScript的情况下我就下的mono版,Steam上没有mono版也就罢了),所以我又重新考虑C#是否可以解决这个问题。
改造方法
就以改写Label.text
的运作方式为例……
Godot源码是C++的,GDScript里面怎么做的封装我不太清楚,但总之只剩下了text属性变量,因为GDScript并不允许对属性重定义,所以不能用setget
来指定setter和getter函数(因为文档规定setget只能在var
声明句中使用,但GDScript的语法又不让重声明,至少不能再次声明同名的属性)。
不过在C++的源码中,Godot实际上是有set_text
和get_text
的,而且这两个函数被表述为public
,所以就很迷……
// setter
void Label::set_text(const String &p_string) {
if (text == p_string) {
return;
}
text = p_string;
xl_text = tr(p_string);
dirty = true;
if (percent_visible < 1) {
visible_chars = get_total_character_count() * percent_visible;
}
update();
}
// getter
String Label::get_text() const {
return text;
}
至于为什么在GDScript里没了,可能是跟某种绑定机制有关,不过这个改天再看看。
不过用C#脚本的话,情况就不一样了,因为在C#里对应的Label.Text
就是C#概念下的Property,而且由于C++源码里是public
(事实上也必须是,因为这东西可改),所以Label
里的绑定多半写的也肯定是:
public class Label : Control
{
//...
public string Text { get{/*...*/} set{/*...*/} }
//...
}
本来我想用override
把原来版本的Text
属性直接推翻重写,不过因为Label.Text
并非virtual
所以我不能推翻,只能强制隐藏……
public class MyLabel : Label
{
public new String Text
{
get {/*...*/}
set {/*...*/}
}
//...
}
不过另外一个问题来了,getter和setter怎么重写的问题……
本来傻fufu的想直接对新的Text
做手脚结果运行的时候Godot调试器直接卡崩溃了……
报错告诉我内存空间爆了,那估计是被当成无尽递归闹得,于是我就改成了给父类捆绑的方式……
public class MyLabel : Label
{
public new String Text
{
get => base.Text;
set
{
// ...
base.Text = value;
// ...
}
}
//...
}
成功了,当我操作Text的时候,我可以在此基础上改写这些性质。
当然了,这种做法有局限:
- 涉及底层API的操作很难复现,所以除非保留对base属性的赋值句
base.XXX=value;
,因为base属性也是相当于一个名字带两个函数,直接执行这个语句会连同原有的操作也做一遍,但是有些底层属性对继承类是private
的,继承类看不到。 - 如果彻底不想使用原有的运作机制的话,那可能还需要引入字段来解决这个问题。(但是实际上这跟独立重开一个属性没差,只是不用换名字了而已)
- 结合上述两点,这种方法可能无法保留局部的原有机制,也就是说要么全用要么全不用,你只能在原有的基础上叠加,如果真的想达到真正意义上的重写可能还得下到底层去……