update todo list
[gitmo/Mouse.git] / t / 000-recipes / 001_point.t
CommitLineData
d56e8388 1#!/usr/bin/perl
2
3use strict;
4use warnings;
5
78b163fb 6use Test::More;
7BEGIN {
8 if (eval "require Class::Method::Modifiers; 1") {
9 plan tests => 58;
10 }
11 else {
12 plan skip_all => "Class::Method::Modifiers required for this test";
13 }
14}
15
ba2da7f2 16use Mouse::Util;
17use t::Exception;
d56e8388 18
19BEGIN {
20 use_ok('Mouse');
21}
22
23{
24 package Point;
25 use Mouse;
26
27 has 'x' => (isa => 'Int', is => 'ro');
28 has 'y' => (isa => 'Int', is => 'rw');
29
30 sub clear {
31 my $self = shift;
32 $self->{x} = 0;
33 $self->y(0);
34 }
35
36}{
37 package Point3D;
38 use Mouse;
39
40 extends 'Point';
41
42 has 'z' => (isa => 'Int');
43
44 after 'clear' => sub {
45 my $self = shift;
46 $self->{z} = 0;
47 };
48
49}
50
51my $point = Point->new(x => 1, y => 2);
52isa_ok($point, 'Point');
53isa_ok($point, 'Mouse::Object');
54
55is($point->x, 1, '... got the right value for x');
56is($point->y, 2, '... got the right value for y');
57
58$point->y(10);
59is($point->y, 10, '... got the right (changed) value for y');
60
61dies_ok {
62 $point->y('Foo');
63} '... cannot assign a non-Int to y';
64
65dies_ok {
66 $point->x(1000);
67} '... cannot assign to a read-only method';
68is($point->x, 1, '... got the right (un-changed) value for x');
69
70$point->clear();
71
72is($point->x, 0, '... got the right (cleared) value for x');
73is($point->y, 0, '... got the right (cleared) value for y');
74
75# check the type constraints on the constructor
76
77lives_ok {
78 Point->new(x => 0, y => 0);
79} '... can assign a 0 to x and y';
80
81dies_ok {
82 Point->new(x => 10, y => 'Foo');
83} '... cannot assign a non-Int to y';
84
85dies_ok {
86 Point->new(x => 'Foo', y => 10);
87} '... cannot assign a non-Int to x';
88
89# Point3D
90
91my $point3d = Point3D->new({ x => 10, y => 15, z => 3 });
92isa_ok($point3d, 'Point3D');
93isa_ok($point3d, 'Point');
94isa_ok($point3d, 'Mouse::Object');
95
96is($point3d->x, 10, '... got the right value for x');
97is($point3d->y, 15, '... got the right value for y');
98is($point3d->{'z'}, 3, '... got the right value for z');
99
100dies_ok {
101 $point3d->z;
102} '... there is no method for z';
103
104$point3d->clear();
105
106is($point3d->x, 0, '... got the right (cleared) value for x');
107is($point3d->y, 0, '... got the right (cleared) value for y');
108is($point3d->{'z'}, 0, '... got the right (cleared) value for z');
109
110dies_ok {
111 Point3D->new(x => 10, y => 'Foo', z => 3);
112} '... cannot assign a non-Int to y';
113
114dies_ok {
115 Point3D->new(x => 'Foo', y => 10, z => 3);
116} '... cannot assign a non-Int to x';
117
118dies_ok {
119 Point3D->new(x => 0, y => 10, z => 'Bar');
120} '... cannot assign a non-Int to z';
121
122# test some class introspection
123
124can_ok('Point', 'meta');
125isa_ok(Point->meta, 'Mouse::Meta::Class');
126
127can_ok('Point3D', 'meta');
128isa_ok(Point3D->meta, 'Mouse::Meta::Class');
129
130isnt(Point->meta, Point3D->meta, '... they are different metaclasses as well');
131
132# poke at Point
133
134is_deeply(
135 [ Point->meta->superclasses ],
136 [ 'Mouse::Object' ],
137 '... Point got the automagic base class');
138
139my @Point_methods = qw(meta new x y clear);
140my @Point_attrs = ('x', 'y');
141
142SKIP: {
143 skip "Mouse has no method introspection", 2 + @Point_methods;
144
145 is_deeply(
146 [ sort @Point_methods ],
147 [ sort Point->meta->get_method_list() ],
148 '... we match the method list for Point');
149
150 is_deeply(
151 [ sort @Point_attrs ],
152 [ sort Point->meta->get_attribute_list() ],
153 '... we match the attribute list for Point');
154
155 foreach my $method (@Point_methods) {
156 ok(Point->meta->has_method($method), '... Point has the method "' . $method . '"');
157 }
158}
159
160foreach my $attr_name (@Point_attrs ) {
161 ok(Point->meta->has_attribute($attr_name), '... Point has the attribute "' . $attr_name . '"');
162 my $attr = Point->meta->get_attribute($attr_name);
163 ok($attr->has_type_constraint, '... Attribute ' . $attr_name . ' has a type constraint');
164
165 SKIP: {
166 skip "Mouse type constraints are not objects", 2;
167 isa_ok($attr->type_constraint, 'Mouse::Meta::TypeConstraint');
168 is($attr->type_constraint->name, 'Int', '... Attribute ' . $attr_name . ' has an Int type constraint');
169 }
170}
171
172# poke at Point3D
173
174is_deeply(
175 [ Point3D->meta->superclasses ],
176 [ 'Point' ],
177 '... Point3D gets the parent given to it');
178
179my @Point3D_methods = qw(new meta clear);
180my @Point3D_attrs = ('z');
181
182SKIP: {
183 skip "Mouse has no method introspection", 2 + @Point3D_methods;
184
185 is_deeply(
186 [ sort @Point3D_methods ],
187 [ sort Point3D->meta->get_method_list() ],
188 '... we match the method list for Point3D');
189
190 is_deeply(
191 [ sort @Point3D_attrs ],
192 [ sort Point3D->meta->get_attribute_list() ],
193 '... we match the attribute list for Point3D');
194
195 foreach my $method (@Point3D_methods) {
196 ok(Point3D->meta->has_method($method), '... Point3D has the method "' . $method . '"');
197 }
198};
199
200foreach my $attr_name (@Point3D_attrs ) {
201 ok(Point3D->meta->has_attribute($attr_name), '... Point3D has the attribute "' . $attr_name . '"');
202 my $attr = Point3D->meta->get_attribute($attr_name);
203 ok($attr->has_type_constraint, '... Attribute ' . $attr_name . ' has a type constraint');
204 SKIP: {
205 skip "Mouse type constraints are not objects", 2;
206 isa_ok($attr->type_constraint, 'Mouse::Meta::TypeConstraint');
207 is($attr->type_constraint->name, 'Int', '... Attribute ' . $attr_name . ' has an Int type constraint');
208 };
209}
210