summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorRico Tzschichholz <ricotz@ubuntu.com>2021-03-10 12:12:06 +0100
committerRico Tzschichholz <ricotz@ubuntu.com>2021-03-10 12:12:06 +0100
commit724c3c5ecaeaddd54c0f8d5fb744bfeb680fc5e8 (patch)
tree4f912e37548397ba2b59e02a871e2af5601f9ecb
parentf969ed96471f224a7bb8774ee23801ebe39ef1f8 (diff)
downloadvala-724c3c5ecaeaddd54c0f8d5fb744bfeb680fc5e8.tar.gz
vala: Replace all type parameter occurances in parameters for signal delegate
Improves 36999b5ffd63cc56a8648791b02bf07e7da88077
-rw-r--r--tests/Makefile.am1
-rw-r--r--tests/objects/bug626038-2.vala32
-rw-r--r--vala/valadatatype.vala32
-rw-r--r--vala/valasignal.vala7
4 files changed, 68 insertions, 4 deletions
diff --git a/tests/Makefile.am b/tests/Makefile.am
index 6e7bd28b8..3373316ed 100644
--- a/tests/Makefile.am
+++ b/tests/Makefile.am
@@ -539,6 +539,7 @@ TESTS = \
objects/bug620706.vala \
objects/bug624594.vala \
objects/bug626038.vala \
+ objects/bug626038-2.vala \
objects/bug628639.vala \
objects/bug629593.vala \
objects/bug631267.vala \
diff --git a/tests/objects/bug626038-2.vala b/tests/objects/bug626038-2.vala
new file mode 100644
index 000000000..2a2422846
--- /dev/null
+++ b/tests/objects/bug626038-2.vala
@@ -0,0 +1,32 @@
+class Bar<K,V> {
+ public K k;
+ public V v;
+ public Bar (K k, V v) {
+ this.k = k;
+ this.v = v;
+ }
+}
+
+class Foo<G> {
+ public signal void bar (Bar<int,G> item);
+
+ public void fire (Bar<int,G> item) {
+ bar (item);
+ }
+}
+
+bool fired;
+
+void on_bar (Bar<int,string> item) {
+ assert (item.k == 42);
+ assert (item.v == "bar");
+ fired = true;
+}
+
+void main () {
+ Foo<string> foo = new Foo<string> ();
+ foo.bar.connect (on_bar);
+ var bar = new Bar<int,string> (42, "bar");
+ foo.fire (bar);
+ assert (fired);
+}
diff --git a/vala/valadatatype.vala b/vala/valadatatype.vala
index 59be21bd7..8ad4cc0cc 100644
--- a/vala/valadatatype.vala
+++ b/vala/valadatatype.vala
@@ -506,6 +506,38 @@ public abstract class Vala.DataType : CodeNode {
return result;
}
+ public bool is_generic () {
+ if (this is GenericType) {
+ return true;
+ }
+
+ if (!has_type_arguments ()) {
+ return false;
+ }
+ foreach (var type_arg in type_argument_list) {
+ if (type_arg.is_generic ()) {
+ return true;
+ }
+ }
+ return false;
+ }
+
+ public void replace_type_parameter (TypeParameter old_type_param, TypeParameter new_type_param) {
+ if (this is GenericType) {
+ unowned GenericType generic_type = (GenericType) this;
+ if (generic_type.type_parameter == old_type_param) {
+ generic_type.type_parameter = new_type_param;
+ }
+ return;
+ }
+ if (!has_type_arguments ()) {
+ return;
+ }
+ foreach (var type_arg in type_argument_list) {
+ type_arg.replace_type_parameter (old_type_param, new_type_param);
+ }
+ }
+
/**
* Search for the type parameter in this formal type and match it in
* value_type.
diff --git a/vala/valasignal.vala b/vala/valasignal.vala
index 3acbf3b03..5c790a7cd 100644
--- a/vala/valasignal.vala
+++ b/vala/valasignal.vala
@@ -122,7 +122,7 @@ public class Vala.Signal : Symbol, Callable {
actual_param.variable_type = actual_param.variable_type.get_actual_type (sender_type, null, node_reference);
generated_delegate.add_parameter (actual_param);
- if (actual_param.variable_type is GenericType) {
+ if (actual_param.variable_type.is_generic ()) {
is_generic = true;
}
}
@@ -136,9 +136,8 @@ public class Vala.Signal : Symbol, Callable {
// parameter types must refer to the delegate type parameters
// instead of to the class type parameters
foreach (var param in generated_delegate.get_parameters ()) {
- unowned GenericType? generic_type = param.variable_type as GenericType;
- if (generic_type != null) {
- generic_type.type_parameter = generated_delegate.get_type_parameters ().get (generated_delegate.get_type_parameter_index (generic_type.type_parameter.name));
+ foreach (var type_param in generated_delegate.get_type_parameters ()) {
+ param.variable_type.replace_type_parameter (cl.get_type_parameters ().get (cl.get_type_parameter_index (type_param.name)), type_param);
}
}
}