/[Frey]/trunk/lib/Frey/Introspect.pm
This is repository of my old source code which isn't updated any more. Go to git.rot13.org for current projects!
ViewVC logotype

Annotation of /trunk/lib/Frey/Introspect.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 380 - (hide annotations)
Mon Nov 17 18:42:51 2008 UTC (15 years, 5 months ago) by dpavlin
File size: 6748 byte(s)
view source below pod which looks like my vim configuration
1 dpavlin 49 package Frey::Introspect;
2    
3     use Moose;
4     use Carp;
5 dpavlin 100 #use Moose::Meta::Role;
6     #use Moose::Meta::Class;
7 dpavlin 49 use Data::Dump qw/dump/;
8     use File::Slurp;
9 dpavlin 51 use List::Util;
10 dpavlin 49
11 dpavlin 380 use PPI;
12     use PPI::HTML;
13    
14 dpavlin 55 use lib 'lib';
15 dpavlin 53
16 dpavlin 308 extends 'Frey::ClassLoader';
17 dpavlin 100 with 'Frey::Web';
18 dpavlin 49
19 dpavlin 270 has 'class' => (
20 dpavlin 49 is => 'rw',
21     isa => 'Str',
22     required => 1,
23     );
24    
25 dpavlin 51 has 'path' => (
26     is => 'rw',
27     );
28    
29 dpavlin 53 =head2 joose
30    
31 dpavlin 137 my $js = $o->joose;
32 dpavlin 53
33     =cut
34    
35     sub joose {
36     my ($self) = @_;
37    
38 dpavlin 308 my ( $meta, $is_role ) = $self->class_meta;
39 dpavlin 53
40     if ( ! $is_role ) {
41 dpavlin 50 my @superclasses = map{ $_->meta->name }
42     grep { $_ ne 'Moose::Object' } $meta->superclasses;
43 dpavlin 100 warn "superclasses ",dump( @superclasses ) if $self->debug;
44 dpavlin 50 }
45 dpavlin 49
46     my $out;
47    
48 dpavlin 270 my ( $m, $c ) = split(/::/, $self->class, 2);
49 dpavlin 51 my $filename = $m . '.' . ( $c ? "$c." : '' ) . 'js';
50 dpavlin 49
51 dpavlin 137 $c ||= '';
52    
53 dpavlin 49 $out .= "Module(\"$m\", function (m) {\n\tClass(\"$c\", {\n\t\thas: {\n";
54    
55 dpavlin 137 foreach ( $meta->get_attribute_list ) {
56 dpavlin 49 $out .= "\t\t\t$_: {\n";
57    
58 dpavlin 137 my $attr = $meta->get_attribute($_);
59 dpavlin 50 my $is = $attr->_is_metadata;
60     $out .= "\t\t\t\tis: \"$is\",\n" if defined $is;
61     $out .= "\t\t\t\tlazy: true,\n" if $attr->is_lazy;
62     $out .= "\t\t\t\trequired: true,\n" if $attr->is_required;
63     $out .= "\t\t\t\tinit: \"" . $attr->init_arg . "\",\n" if $attr->init_arg; # FIXME
64    
65     if( defined(my $isa = $attr->_isa_metadata) ){
66     if( blessed $isa ){
67     while( blessed $isa ){
68     $isa = $isa->name;
69     }
70     }
71     $isa =~ s/\s+\|\s+undef//gi;
72     $out .= "\t\t\t\tisa: Moose.$isa,\n";
73     }
74    
75    
76 dpavlin 49 $out .= "\t\t\t},\n";
77    
78     }
79    
80 dpavlin 51 $out .= "\t\t},\n\t\tmeta: Frey.HTML,
81     classMethods: {
82     renderHTML: function () {
83 dpavlin 292 return new Joose.SimpleRequest().getText(\"/" . $self->class . "\")
84 dpavlin 51 },\n";
85    
86 dpavlin 49 $out .= "\t\t},\n";
87    
88     $out .= "\t}),\n";
89 dpavlin 50
90     $out =~ s/,\n$/\n/;
91 dpavlin 49 $out .= "});\n";
92    
93 dpavlin 270 $out .= "\nconsole.log( 'loaded " . $self->class . " from $filename' );\n";
94 dpavlin 49
95 dpavlin 137 warn "method_list = ",dump( $meta->get_method_list ) if $self->debug;
96 dpavlin 49
97 dpavlin 51 # print $out;
98 dpavlin 49 my $path = "static/blib/$filename";
99     write_file( $path, $out );
100     warn "# created $path\n";
101 dpavlin 51 $self->path( $path );
102    
103     return $out;
104 dpavlin 49 }
105    
106 dpavlin 53 =head2 methods
107    
108     my @methods = $o->methods;
109    
110     =cut
111    
112     sub methods {
113     my $self = shift;
114    
115 dpavlin 308 my ( $meta, $is_role ) = $self->class_meta;
116 dpavlin 53
117     my $attr;
118 dpavlin 137 $attr->{$_}++ foreach $meta->get_attribute_list;
119     my @methods = grep { ! defined($attr->{$_}) } $meta->get_method_list;
120 dpavlin 100 warn "# methods = ",dump( @methods ) if $self->debug;
121 dpavlin 53
122 dpavlin 120 return sort @methods;
123 dpavlin 53 }
124    
125     =head1 OUTPUT GENERATION
126    
127 dpavlin 269 =head2 markup
128 dpavlin 53
129 dpavlin 269 $o->markup;
130 dpavlin 53
131     =cut
132    
133 dpavlin 269 sub markup {
134     my ( $self ) = @_;
135 dpavlin 53
136 dpavlin 160 $self->add_head( 'static/introspect.css' );
137 dpavlin 139
138 dpavlin 308 my ( $meta, $is_role ) = $self->class_meta;
139 dpavlin 53
140 dpavlin 270 my $class = $self->class;
141 dpavlin 54
142 dpavlin 310 my ( $superclasses, $roles ) = ( '<b>Role</b>', '' );
143     if ( ! $is_role ) {
144     if ( $meta->superclasses ) {
145     $superclasses = 'Superclasses: ' .
146     join(', ',
147     map {
148     my $name = $_->meta->name;
149     qq|<a class="frey-popdown" href="/$name">$name<code>| . $_->meta->dump(2) . qq|</code></a>|;
150     }
151     #grep { $_ ne 'Moose::Object' }
152     $meta->superclasses
153     );
154     }
155     }
156    
157     my $role_method;
158 dpavlin 313 my $role_attribute;
159 dpavlin 310
160     if ( $meta->can('roles') ) {
161     my $role_nr = 1;
162     $roles = join(' ',
163     grep { ! m/\Q$class\E/ } # skip me
164     map {
165     my $name = $_->name;
166 dpavlin 313 $role_method->{ $_ }->{$name} = $role_nr foreach $_->get_method_list;
167     $role_attribute->{ $_ }->{$name} = $role_nr foreach $_->get_attribute_list;
168 dpavlin 310 qq|<a class="frey-popdown" href="/$name">$name<code>| . $name->meta->dump(2) . qq|</code></a><sup>| . $role_nr++ . qq|</sup>|;
169     }
170     $meta->calculate_all_roles
171     );
172     $roles = qq| with roles: $roles| if $roles;
173     }
174     warn "# role_method ",dump( $role_method );
175    
176 dpavlin 269 my @methods;
177 dpavlin 310 @methods = map {
178 dpavlin 313 my $name = $_;
179     if ( $role_method->{$name} ) {
180     my ( $role_name, $nr ) = each %{ $role_method->{$name} };
181     $name .= qq|<sup title="$role_name">$nr</sup>|;
182 dpavlin 310 }
183 dpavlin 313 qq|<td class="m">$name</td>|
184 dpavlin 310 } $self->methods;
185 dpavlin 53
186 dpavlin 269 my @attributes;
187     if ( $meta->get_attribute_list ) {
188     @attributes = map {
189 dpavlin 313 my $name = $_;
190     my $attr = $meta->get_attribute($name);
191 dpavlin 331 warn "## ref attr: ",ref( $attr );
192 dpavlin 269 my ( $before, $title, $after ) = ( '', '', '' );
193     ( $before, $title, $after ) = ( '<b>', ' title="required"', '</b>' ) if $attr->is_required;
194 dpavlin 198 warn $attr->dump(3);
195 dpavlin 288 foreach my $check ( qw/has_type_constraint has_handles is_weak_ref is_required is_lazy should_coerce should_auto_deref has_default has_trigger has_documentation has_applied_traits/ ) {
196 dpavlin 269 my $getter;
197    
198     $getter = $check;
199     $getter =~ s/^has_//;
200    
201     if ( $attr->$check ) {
202     if ( $getter eq $check ) {
203     $after .= "$check";
204     } else {
205     $after .= qq{<span class="frey-popdown">$check};
206 dpavlin 288 # we need dump here instead of $attr->$getter->dump because default can return scalar
207     $after .= '<code>' . dump( $attr->$getter ) . '</code>' if $getter ne $check;
208 dpavlin 269 $after .= '</span>';
209 dpavlin 197 }
210     }
211 dpavlin 269 $after .= ' ';
212     }
213 dpavlin 290 my $type = $attr->has_type_constraint ? $attr->type_constraint->name : '';
214 dpavlin 313
215     if ( $role_attribute->{$name} ) {
216     my ( $role_name, $nr ) = each %{ $role_attribute->{$name} };
217     $name .= qq|<sup title="$role_name">$nr</sup>|;
218     }
219    
220     qq|<td class="a">$before $name</td><td class="t">$type</td><td>$after</td>|
221 dpavlin 269 } sort $meta->get_attribute_list
222     }
223 dpavlin 54
224 dpavlin 312 my $table = qq|<table class="frey-introspect"><tr><th class="m">Methods</th><th class="a">Attributes</th><th>Type</th><th class="p">Properties</th></tr>|;
225 dpavlin 269 while ( @methods || @attributes ) {
226     my ($m,$a) = ( shift @methods, shift @attributes );
227     $m ||= '<td></td>';
228     $a ||= '<td></td>';
229     $table .= qq|<tr>$m$a</tr>|;
230     }
231     $table .= qq|</table>|;
232    
233 dpavlin 270 my $pod = Frey::Pod->new( class => $class )->markup;
234 dpavlin 380 $pod = $pod->{body} if ref($pod);
235 dpavlin 134
236 dpavlin 380 my $Document = PPI::Document->new( $self->class_path( $class ) );
237    
238     # Create a reusable syntax highlighter
239     my $Highlight = PPI::HTML->new(
240     # line_numbers => 1,
241     # page => 1,
242     # colors => {
243     # line_number => '#CCCCCC',
244     # number => '#990000',
245     # },
246     );
247    
248     # Spit out the HTML
249     my $source = $Highlight->html( $Document );
250    
251     # $source =~ s{^.*<body[^>]+>}{}s;
252     # $source =~ s{</body.*$}{}s;
253    
254 dpavlin 269 use Frey::Run;
255 dpavlin 314 my $runnable = join("\n", map { qq|<a href="/$class/$_">$_</a>| } grep { $class->can($_) } Frey::Run->runnable );
256     $runnable = " runnable: $runnable" if $runnable;
257 dpavlin 128
258 dpavlin 358 my $html = {
259 dpavlin 380 title => $class,
260     body => join("\n"
261     , qq|<h1>$class</h1>|,
262     , qq|<div class="frey-introspect">$superclasses\n$roles\n$runnable\n|,
263     , ( $pod ? qq|<a href="#___top" title="Skip to POD" class="frey-skip">pod</a>| : '' )
264     , qq|<a href="#source" title="Skip to source" class="frey-skip">source</a>|
265     , qq|$table\n$pod\n</div>\n|
266     , qq|<h1>Source</h1><a name="source"></a><div class="frey-source">$source</div>|
267     ),
268 dpavlin 358 };
269 dpavlin 128
270 dpavlin 269 return $html;
271 dpavlin 53 }
272    
273 dpavlin 49 =head1 SEE ALSO
274    
275     L<MooseX::AutoDoc> on which this code is based
276    
277     =cut
278    
279     1;

  ViewVC Help
Powered by ViewVC 1.1.26