annotate d2/qtd/Attribute.d @ 348:29ea6511681f

more attribute prototyping
author maxter <spambox@d-coding.com>
date Sat, 15 May 2010 00:29:47 +0300
parents 970332a88b72
children 31520b2c0b3c
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
345
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
1 /*********************************************************
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
2 Copyright: QtD Team, 2010
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
3 Authors: Max Samukha
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
4 License: Boost Software License 1.0
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
5 *********************************************************/
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
6 module qtd.Attribute;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
7
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
8 import
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
9 lds.meta.compiletime,
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
10 std.traits,
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
11 std.conv,
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
12 std.variant,
345
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
13 std.typetuple;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
14
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
15 enum standardNamespace = "qtd";
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
16
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
17 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
18 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
19 enum AttributeOptions
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
20 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
21 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
22 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
23 none,
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
24
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
25 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
26 Allows multiple attributes of the same category to be associated with the symbol.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
27 */
346
2691dd58d7e1 fixed QSlitter.getRange
Max Samukha <maxter@spambox.com>
parents: 345
diff changeset
28 allowMultiple = 0x0000_0001,
2691dd58d7e1 fixed QSlitter.getRange
Max Samukha <maxter@spambox.com>
parents: 345
diff changeset
29
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
30 /* internal */ inner = 0x0000_0002,
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
31
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
32 /**
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
33 Specifies that the attribute data are in name-value form.
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
34 */
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
35 nameValue = 0x0000_0004
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
36 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
37
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
38 /**
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
39 When mixed in an aggregate, converts a compile-time tuple to
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
40 members of that aggregate.
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
41 */
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
42 mixin template tupleToMembers!(string nameSpace, size_t index, A...)
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
43 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
44 static if (index < A.length)
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
45 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
46 enum indexStr = to!string(index);
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
47
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
48 static if (is(__traits(compiles, { struct { typeof(A[index]) x; } }() })))
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
49 mixin("typeof(A[" ~ indexStr ~ "]) " ~ nameSpace ~ ~ " = A[" ~ indexStr ~"];\n" ~ next;
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
50 else
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
51 mixin("alias A[" ~ indexStr ~ "] " ~ nameSpace ~ indexStr ~ ";\n" ~ next;
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
52
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
53 mixin tupleToFields!(nameSpace, index + 1, A);
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
54 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
55 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
56
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
57 /**
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
58 When mixed in an aggregate, converts a compile-time tuple of name-value pairs to
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
59 members of that aggregate.
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
60 */
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
61 struct NameValueTupleToFields(A...)
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
62 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
63
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
64 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
65
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
66 version (QtdUnittest)
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
67 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
68 unittest
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
69 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
70 static int foo()
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
71 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
72 return 42;
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
73 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
74
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
75 static struct S
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
76 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
77 mixin TupleToFields!("field", 0,
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
78 int,
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
79 "a",
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
80 22,
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
81 foo);
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
82 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
83
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
84 static assert(is(S.field0 == int));
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
85 S s;
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
86 assert(s.field1 == "a");
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
87 assert(s.field2 == "22");
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
88 assert(S.foo() == 42);
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
89 }
345
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
90 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
91
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
92 private template attributeId(alias symbol, uint index = 0)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
93 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
94 enum attributeId = standardNamespace ~ "_attr_" ~ uniqueId!symbol ~ "_" ~ to!string(index);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
95 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
96
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
97 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
98 Attributes allow to associate arbitrary compile-time data
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
99 with a declaration and optionaly make that data available at run-time.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
100
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
101 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
102 class A
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
103 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
104 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
105 mixin Attribute!(C, "someAttribute", "B");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
106 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
107
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
108 The example above associates the string "B" with class A under the attribute category "someAttribute".
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
109 Multiple data can be associated with an attribute category for a single declaration:
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
110
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
111 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
112 class A
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
113 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
114 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
115 mixin Attribute!(C, "someAttribute", "B");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
116 mixin Attribute!(C, "someAttribute", AttributeOptions.allowMultiple, "C");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
117 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
118
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
119 Attributes of an attribute category can be accessed at compile-time:
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
120
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
121 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
122 alias GetAttributes!(C, "someAttribute") attrs;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
123 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
124
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
125 GetAttribute returns a compile time tuple, which contains "someAttribute" category attributes in the form
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
126 of TypeTupleWrapper instances. The first element of the tuple wrapped in the TypeTupleWrapper
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
127 contains the attribute category name, other elements contain the attrubute data.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
128
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
129 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
130 alias attrs[0] attr0;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
131 alias attrs[1] attr1;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
132
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
133 static assert(attrs0.tuple[0] == "someAttribute" && attrs0.tuple[1] == "B");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
134 static assert(attrs1.tuple[0] == "someAttribute" && attrs1.tuple[1] == "C");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
135 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
136
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
137 Attributes can be inserted inside the body of a declaration.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
138 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
139 class A
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
140 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
141 mixin Attribute!("someAttribute", "B");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
142 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
143 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
144
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
145 Attributes can be made available at run time by means of the declaration's meta-object
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
146 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
147 // prints category names of all attributes associated with class A
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
148 auto a = new A;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
149 foreach (attr; a.metaObject.attributes)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
150 writeln(attr.name);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
151 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
152
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
153 Attributes can be specialized
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
154
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
155 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
156 mixin template DbFieldAttribute(alias prop, string columnName)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
157 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
158 mixin Attribute!(prop, "DbFieldAttribute", columnName);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
159 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
160
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
161 class A
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
162 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
163 int value;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
164 mixin DbFieldAttribute!(value, "Value");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
165
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
166 int anotherValue;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
167 mixin DbFieldAttribute!(anotherValue, "Value2");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
168 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
169
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
170 assert(GetAttributes!(A.value, "DbFieldAttribute")[0].tuple[1] == "Value");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
171 assert(GetAttributes!(A.anotherValue, "DbFieldAttribute")[0].tuple[1] == "Value2");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
172 ----
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
173
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
174 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
175 mixin template Attribute(alias symbol, string attrClass, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
176 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
177 mixin Attribute!(symbol, attrClass, AttributeOptions.none, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
178 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
179
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
180 /// ditto
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
181 mixin template Attribute(alias symbol, string attrClass, AttributeOptions opts, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
182 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
183 mixin AttributeImpl!(symbol, attrClass, opts, 0, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
184 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
185
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
186
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
187 /// ditto
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
188 // TODO: probably make this an overload of Attribute
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
189 mixin template InnerAttribute(string attrClass, AttributeOptions opts, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
190 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
191 // BUG: needs to be generalized to accept any parent
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
192 mixin Attribute!(typeof(this), attrClass, opts | AttributeOptions.inner, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
193 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
194
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
195 // ditto
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
196 mixin template InnerAttribute(string attrClass, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
197 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
198 // BUG: needs to be generalized to accept any parent
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
199 mixin InnerAttribute!(attrClass, AttributeOptions.none, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
200 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
201
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
202 private mixin template AttributeImpl(alias symbol, string attrClass, AttributeOptions opts, size_t index, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
203 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
204 private enum attrId = attributeId!(symbol, index) ~ (opts & AttributeOptions.inner ? "_inner" : "");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
205
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
206 static if (is(typeof(mixin(attrId))))
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
207 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
208 mixin ("alias " ~ attrId ~ " attr;");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
209 static if (!(opts & AttributeOptions.allowMultiple))
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
210 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
211 static assert (attr[0] != attrClass, "Multiple " ~ attrClass ~ " attributes are not allowed for "
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
212 ~ __traits(parent, symbol).stringof ~ "." ~ stringOf!symbol);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
213 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
214
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
215 mixin AttributeImpl!(symbol, attrClass, opts, index + 1, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
216 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
217 else
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
218 mixin ("alias TypeTuple!(attrClass, A) " ~ attrId ~ ";");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
219 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
220
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
221 /**
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
222 Base class for run time attribute implementations
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
223 */
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
224 abstract class MetaAttribute
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
225 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
226 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
227
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
228 /**
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
229 A run-time attribute implementation that stores the attribute data in an
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
230 array of variants.
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
231 */
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
232 final class MetaVariantAttribute : MetaAttribute
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
233 {
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
234 Variant[] values;
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
235
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
236 private this()
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
237 {
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
238 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
239
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
240 static MetaVariantAttribute create(string category, AttributeOptions opts, A...)()
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
241 {
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
242 auto ret = new MetaVariantAttribute;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
243 super.construct!(category, opts, A)();
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
244 foreach(i, _; A)
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
245 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
246 static if (__traits(compiles, { ret.values ~= Variant(A[i]); }() ))
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
247 ret.values ~= Variant(A[i]);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
248 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
249 return ret;
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
250 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
251 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
252
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
253 /**
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
254 A run-time attribute implementation that stores the attribute data in an
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
255 assiciative array of variants.
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
256 */
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
257 final class MetaVariantDictAttribute : MetaAttribute
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
258 {
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
259 Variant[string] values;
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
260
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
261 private this()
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
262 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
263 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
264
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
265 static MetaVariantAttribute create(string category, AttributeOptions opts, A...)()
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
266 {
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
267 auto ret = new MetaVariantAttribute;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
268 super.construct!(category, opts, A)();
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
269 foreach(i, _; A)
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
270 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
271 static if (i % 2 == 0 && __traits(compiles, { ret.values[A[i]] = Variant(A[i + 1]); }() ))
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
272 ret.values[A[i]] ~= Variant(A[i + 1]);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
273 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
274 return ret;
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
275 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
276 }
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
277
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
278
348
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
279 /**
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
280 A run-time attribute implementation that stores the attribute data in
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
281 typed fields named fieldN, where N is the index of the original attribute data element.
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
282 */
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
283 abstract class MetaTypedAttribute : MetaAttribute
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
284 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
285 private this() {}
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
286
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
287 static class Impl(A) : typeof(this)
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
288 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
289 private this() {}
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
290
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
291 mixin tupleToMembers!("field", 0, A);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
292 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
293
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
294 static MetaAttribute create(string category, AttributeOptions opts, A...)()
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
295 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
296 auto ret = new Impl!A;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
297 super.construct(category, opts, A);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
298 return ret;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
299 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
300 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
301
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
302 /**
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
303 A run-time attribute implementation that stores the attribute data in
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
304 typed fields by interpreting the original attribute data as name-value pairs.
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
305 */
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
306 abstract class MetaTypedDictAttribute : MetaAttribute
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
307 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
308 private this() {}
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
309
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
310 static class Impl(A) : typeof(this)
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
311 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
312 private this() {}
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
313
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
314 mixin nameValueTupleToMembers!("", A);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
315 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
316
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
317 static MetaAttribute create(string category, AttributeOptions opts, A...)()
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
318 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
319 auto ret = new Impl!A;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
320 super.construct(category, opts, A);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
321 return ret;
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
322 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
323 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
324
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
325 version (QtdUnittest)
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
326 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
327 unittest
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
328 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
329 static void foo() {}
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
330
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
331 static class C
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
332 {
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
333 mixin InnerAttribute!("someAttribute", MetaVariantAttribute, "22", foo, 33);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
334 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
335
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
336 auto attr = cast(MetaVariantAttribute) meta!(C).attributes[0];
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
337 assert(attr.name == "someAttribute");
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
338 assert(qttr.length == 2);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
339 assert(attr.values[0] == "22");
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
340 assert(attr.values[1] == 33);
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
341 }
29ea6511681f more attribute prototyping
maxter <spambox@d-coding.com>
parents: 347
diff changeset
342 }
347
970332a88b72 lowered the marshall in rank
maxter <spambox@d-coding.com>
parents: 346
diff changeset
343
345
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
344 private string stringOfFunction(alias symbol)()
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
345 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
346 auto ptrType = typeof(&symbol).stringof;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
347 auto paramList = ParameterTypeTuple!(symbol).stringof;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
348
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
349 string result = ReturnType!(symbol).stringof ~ " " ~ __traits(identifier, symbol) ~ paramList;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
350
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
351 if (ptrType[$ - 1] != ')')
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
352 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
353 for (size_t i = ptrType.length - 2;; --i)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
354 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
355 if (ptrType[i] == ')')
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
356 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
357 result ~= ptrType[i + 1..$];
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
358 break;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
359 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
360 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
361 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
362
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
363 return result;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
364 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
365
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
366 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
367 String of any symbol, including functions
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
368 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
369 template stringOf(alias symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
370 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
371 static if (isFunction!symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
372 enum stringOf = stringOfFunction!symbol;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
373 else
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
374 enum stringOf = symbol.stringof;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
375 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
376
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
377 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
378 Returns the string uniquely identifying the
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
379 symbol in its container.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
380 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
381 template uniqueName(alias symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
382 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
383 enum uniqueName = stringOf!symbol;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
384 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
385
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
386 // TODO: works only for simple types. implement
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
387 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
388 *
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
389 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
390 string uniqueIdImpl(string symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
391 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
392 char[] r = symbol.dup;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
393 foreach (i, c; symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
394 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
395 if (c == '(' || c == ')' || c == ' ')
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
396 r[i] = '_';
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
397 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
398
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
399 return cast(immutable)r;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
400 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
401
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
402 template uniqueId(alias symbol)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
403 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
404 enum uniqueId = uniqueIdImpl(stringOf!symbol);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
405 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
406
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
407 version (QtdUnittest)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
408 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
409 unittest
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
410 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
411 static class C
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
412 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
413 void foo() const {};
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
414 bool bar(int) { return true; };
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
415 int x;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
416 static assert (stringOf!foo == "void foo() const");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
417 static assert (uniqueName!foo == "void foo() const");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
418 static assert (stringOf!bar == "bool bar(int)");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
419 static assert (uniqueName!bar == "bool bar(int)");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
420 static assert (stringOf!x == "x");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
421 static assert (uniqueName!x == "x");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
422 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
423
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
424 static assert (stringOf!(C.foo) == "void foo() const");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
425 static assert (uniqueName!(C.foo) == "void foo() const");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
426 static assert (stringOf!(C.bar) == "bool bar(int)");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
427 static assert (uniqueName!(C.bar) == "bool bar(int)");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
428
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
429 static assert (stringOf!(C.x) == "x");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
430 static assert (uniqueName!(C.x) == "x");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
431 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
432 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
433
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
434 template truePred(A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
435 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
436 enum truePred = true;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
437 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
438
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
439 template attrCategoryPred(string name)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
440 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
441 template attrCategoryPred(A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
442 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
443 enum attrCategoryPred = A[0] == name;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
444 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
445 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
446
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
447 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
448 Returns a compile-time tuple of attributes that
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
449 match pred.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
450 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
451 template GetAttributes(alias symbol, alias pred = truePred)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
452 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
453 alias GetAttributesImpl!(symbol, 0, pred).result GetAttributes;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
454 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
455
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
456 /**
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
457 Returns a compile-time tuple of attributes
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
458 matching the specified attribute category.
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
459 */
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
460 template GetAttributes(alias symbol, string attrCategory)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
461 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
462 alias GetAttributes!(symbol, attrCategoryPred!attrCategory) GetAttributes;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
463 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
464
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
465
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
466 template GetAttributesImpl(alias symbol, size_t index, alias pred)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
467 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
468 //pragma(msg, mixin("symbol." ~ attributeId!(symbol, index) ~ "_inner").stringof);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
469 static if (is(typeof(mixin("__traits(parent, symbol)." ~ attributeId!(symbol, index)))))
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
470 mixin ("alias lds.meta.compiletime.Alias!(__traits(parent, symbol))." ~ attributeId!(symbol, index) ~ " attr;");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
471 else static if (is(typeof(mixin("symbol." ~ attributeId!(symbol, index) ~ "_inner"))))
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
472 mixin ("alias symbol." ~ attributeId!(symbol, index) ~ "_inner attr;");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
473
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
474 static if (is(typeof(attr)))
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
475 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
476 alias GetAttributesImpl!(symbol, index + 1, pred).result next;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
477
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
478 static if (pred!attr)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
479 alias TypeTuple!(TypeTupleWrapper!attr, next) result;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
480 else
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
481 alias next result;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
482 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
483 else
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
484 alias TypeTuple!() result;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
485 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
486
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
487 version (QtdUnittest)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
488 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
489 mixin template MyAttribute(alias symbol, A...)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
490 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
491 mixin Attribute!(symbol, "MyAttribute", AttributeOptions.allowMultiple, A);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
492 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
493
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
494 mixin template ClassInfo(string name, alias value)
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
495 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
496 mixin InnerAttribute!("ClassInfo", AttributeOptions.allowMultiple, name, value);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
497 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
498
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
499 unittest
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
500 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
501 static class C
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
502 {
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
503 // inner C attributes
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
504 mixin InnerAttribute!("Inner", 33); // generic
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
505 mixin ClassInfo!("version", 123);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
506 mixin ClassInfo!("author", "James Bond");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
507
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
508
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
509 void foo() {};
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
510 // foo attributes
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
511 mixin Attribute!(foo, "SomeAttribute", 42);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
512 mixin MyAttribute!(foo, 1, 2);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
513 mixin MyAttribute!(foo, 3, 4);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
514
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
515 alias GetAttributes!(typeof(this), "Inner") innerAttrs;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
516 static assert(innerAttrs[0].tuple[0] == "Inner");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
517 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
518 // outer C attribute
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
519 mixin MyAttribute!(C, 24);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
520
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
521 alias GetAttributes!(C, "Inner") innerAttrs;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
522 static assert(innerAttrs[0].tuple[0] == "Inner" && innerAttrs[0].tuple[1] == 33);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
523
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
524 alias GetAttributes!(C, "ClassInfo") ciAttrs;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
525 static assert(ciAttrs[0].tuple[1] == "version" && ciAttrs[0].tuple[2] == 123);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
526
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
527 alias GetAttributes!(C.foo, "SomeAttribute") someAttr;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
528 static assert(someAttr.length == 1);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
529 static assert(someAttr[0].tuple[0] == "SomeAttribute");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
530
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
531 alias GetAttributes!(C.foo, "MyAttribute") myAttrs;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
532
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
533 //COMPILER BUG: cannot 'alias myAttrs[0].tuple myAttrs_0';
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
534 static assert(myAttrs[0].tuple[0] == "MyAttribute");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
535 static assert(myAttrs[0].tuple[1] == 1 && myAttrs[0].tuple[2] == 2);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
536
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
537 static assert(myAttrs[1].tuple[0] == "MyAttribute");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
538 static assert(myAttrs[1].tuple[1] == 3 && myAttrs[1].tuple[2] == 4);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
539
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
540 /+ BUG: Fails: local declarations cannot be accessed as parent.localDecl
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
541 alias GetAttributes!(C, "MyAttribute") myAttrs2;
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
542 static assert(myAttrs2[0].tuple[0] == "MyAttribute");
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
543 static assert(myAttrs2[0].tuple[1] == 24);
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
544 +/
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
545 }
719604a71da0 added attribute exercise
Max Samukha <maxter@spambox.com>
parents:
diff changeset
546 }