/[Frey]/trunk/lib/Frey/Action.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

Contents of /trunk/lib/Frey/Action.pm

Parent Directory Parent Directory | Revision Log Revision Log


Revision 1113 - (show annotations)
Mon Jun 29 18:25:33 2009 UTC (14 years, 10 months ago) by dpavlin
File size: 9738 byte(s)
fix critique
1 package Frey::Action;
2 use Moose;
3 extends 'Frey::PPI';
4 with 'Frey::Web';
5 with 'Frey::Config';
6
7 use Clone qw/clone/;
8 use Data::Dump qw/dump/;
9
10 =head1 DESCRIPTION
11
12 Invoke any L<Frey> object creating html for with various default parameters
13 if not supplied at invocation.
14
15 You can force rendering of fields if you define C<render_attribute> sub with
16 desired rendering as in:
17
18 sub render_pipe { 'radio' }
19
20 =cut
21
22 has 'class' => (
23 is => 'rw',
24 isa => 'Str',
25 required => 1,
26 );
27
28 has 'params' => (
29 is => 'rw',
30 isa => 'HashRef',
31 default => sub { {} },
32 );
33
34 has 'input_step_size' => (
35 documentation => 'Resize input fields by this step',
36 is => 'rw',
37 isa => 'Int',
38 # required => 1,
39 default => 20,
40 );
41
42 =head2 required
43
44 my @required_attributes = $self->required;
45 my $required_attributes = $self->required;
46 my $required_hash = $self->required('as_hash');
47
48 =cut
49
50 sub required {
51 my ($self,$param) = @_;
52 $self->load_class( $self->class );
53
54 my @required =
55 grep {
56 defined $_ && $_->can('name') &&
57 ! defined( $self->params->{ $_->name } ) &&
58 ! $_->is_lazy
59 }
60 map {
61 my $attr = $self->class->meta->get_attribute($_);
62 blessed $attr && $attr->is_required && $attr;
63 } $self->class->meta->get_attribute_list;
64
65 @required = map { $_->name } @required;
66 warn "## required = ",dump( @required ), " for ", $self->class if @required && $self->debug;
67
68 if ( $param eq 'as_hash' ) {
69 my $hash;
70 map { $hash->{$_}++ } @required;
71 return $hash;
72 }
73 return @required if wantarray;
74 return \@required;
75 }
76
77 =head2 attributes
78
79 Generated from attributes specified in code (extracted using L<Frey::PPI>)
80 and required atributes
81
82 my @class_attributes = $self->attributes;
83 my @class_attributes = $self->attributes;
84
85 =cut
86
87 sub attributes {
88 my ( $self ) = @_;
89 my $a;
90 my @attrs = @{ $self->attribute_order };
91 @attrs = map { $a->{$_}++; $_ } @attrs;
92 push @attrs, $_ foreach grep { ! $a->{$_} } @{ $self->required };
93 warn "# attributes = ",dump( @attrs ) if $self->debug;
94 return @attrs if wantarray;
95 return \@attrs;
96 }
97
98 =head2 params_form
99
100 my $html = $self->params_form;
101 my ($html,$default_params) = $self->params_form;
102
103 =cut
104
105 sub form_id {
106 my ($self) = @_;
107 my $form_id = $self->class;
108 $form_id =~ s{\W+}{_}g;
109 return $form_id;
110 }
111
112 sub select_values {
113 my ( $self, $name, $attr_type, $values ) = @_;
114
115 $attr_type ||= '?' and warn "$name doesn't have attr_type";
116
117 my $form_id = $self->form_id;
118 my $max_value_len = 0;
119 my @values;
120 my $display;
121 my $html = '';
122
123 foreach ( @$values ) {
124 my $v = ref($_) eq 'HASH' ? $_->{$name} : $_;
125 if ( $v =~ s/\t+(.+)$// ) {
126 $display->{$v} = $1;
127 }
128 warn "## value '$v'";
129 push @values, $v;
130 $max_value_len = length($v) if length($v) > $max_value_len;
131 }
132
133 warn "# max_value_len: $max_value_len";
134 #my $render = eval $class . '->render_' . $name;
135 my $call = 'render_' . $name;
136 my $render = $self->class->$call;
137 warn "## render $@";
138
139 if ( $#values > 3 && $render !~ m{radio} ) {
140 my $options = join("\n",
141 map {
142 my $d = $display->{$_} || $_;
143 qq|<option value="$_">$d</option>|;
144 } @values
145 );
146 # onchange="alert(this.options[this.selectedIndex].value);"
147 $html = qq|
148 <select title="$attr_type" name="$name">
149 $options
150 </select>
151 | if $options;
152 } else {
153 my $delimiter = $max_value_len > $self->input_step_size ? qq|<br>| : '';
154 my $radio =
155 # $delimiter .
156 join("\n",
157 map { strip(qq|
158 <span title="$attr_type">
159 <input type="radio" name="$name" value="$_">
160 $_
161 </span>
162 $delimiter
163 |) } @values
164 );
165 if ( $radio ) {
166
167 my $size = int( $max_value_len / $self->input_step_size ) + 1;
168 $size = 5 if $size > 5;
169 $size *= $self->input_step_size;
170 $radio .= qq|
171 <span>
172 <input type="radio" name="$name" value=" " onclick="document.getElementById('new-$name').focus();" >
173 <input type="text" name="new-$name" id="new-$name" onchange="clear_radio('$form_id','$name'); this.disable = false;" onblur="this.disable = true;" title="enter new value" size="$size">
174 </span>
175 |;
176 }
177 $html = qq|<div style="display: block;">$radio</div>|;
178 }
179
180 return
181 # qq|<input type="text" name="$name">| .
182 $html
183 }
184
185 sub params_form {
186 my ( $self ) = @_;
187
188 foreach my $checkbox ( split(/\s+/, $self->params->{'frey-checkboxes'} ) ) {
189 next if defined $self->params->{ $checkbox };
190
191 $self->params->{ $checkbox } = 0;
192 warn "# checkbox $checkbox not ticked";
193 }
194
195 my $required = $self->required('as_hash');
196 if ( $required ) {
197 warn $self->class, " required params ", dump( keys %$required ) if $self->debug;
198 } else {
199 warn "all params available ", dump( $self->params ), " not creating form" if $self->debug;
200 return (undef,$self->params) if wantarray;
201 return;
202 }
203
204 my $class = $self->class;
205
206 $self->load_class( $class );
207
208 my $default = clone $self->params; # XXX we really don't want to modify params!
209
210 my $params_config = {};
211 $params_config = $self->config($class);
212 warn "# $class config = ",dump( $params_config ) if $self->debug;
213
214 my $form;
215 my $form_id = $self->form_id;
216
217 my @checkboxes;
218
219 my $label_width = 1; # minimum
220
221 my @fields =
222 grep {
223 die "$_ doesn't have meta" unless $class->can('meta');
224 ! $class->meta->get_attribute($_)->is_lazy
225 # && ! defined $default->{$_} # XXX show fields with values
226 && ! m{^_} # skip _private
227 } $self->attributes;
228
229 my $fieldset;
230
231 my $last;
232 foreach my $name ( @fields ) {
233 my $set = $name;
234 $set =~ s{_[^_]+$}{};
235 push @{ $fieldset->{$set} }, $name;
236 }
237
238 delete( $fieldset->{$_} )
239 foreach ( grep { $#{ $fieldset->{$_} } == 0 } keys %$fieldset );
240
241 warn "# fieldset = ",dump( $fieldset );
242
243 foreach my $name ( @fields ) {
244 my $attr_type = '';
245 my $type = $name =~ m/^pass/ ? 'password' : 'text';
246 my $label = $name;
247 my $label_title = '';
248 my $value_html = '';
249
250 my $attr = $class->meta->get_attribute( $name );
251 $attr_type = $attr->type_constraint->name if $attr->has_type_constraint;
252
253 my $value =
254 defined $default->{$name} ? $default->{$name} :
255 $attr->has_default ? $attr->default( $name ) :
256 undef;
257
258 if ( ref($params_config) eq 'HASH' && defined $params_config->{$name} ) {
259 $value = $params_config->{$name};
260 } elsif ( ref($params_config) eq 'ARRAY' ) {
261 $value_html = $self->select_values( $name, $attr_type, $params_config );
262 $default->{$name} = $params_config->[0]->{$name};
263 } elsif ( $attr->has_type_constraint && $attr->type_constraint->can('values') ) {
264 $value_html = $self->select_values( $name, $attr_type, $attr->type_constraint->values );
265 } elsif ( $class->can( $name . '_available' ) ) {
266 my $available = $class->$name . '_available';
267 confess $@ if $@;
268 $available =~ s/^\s+//gs;
269 $available =~ s/\s+$//gs;
270 $value_html = $self->select_values( $name, $attr_type, [ split(/\n/,$available) ]);
271 } elsif ( $attr_type =~ m{^Bool} ) {
272 my $suffix = '';
273 $suffix = ' checked=1' if $value;
274 $value_html = qq|<input type="checkbox" name="$name" title="$attr_type" value=1$suffix>|;
275 push @checkboxes, $name;
276 } elsif ( ! defined $value && ! $required->{$name} ) {
277 $value_html = qq|<tt id="$name">undef</tt><!-- $name = undef -->|; # FIXME if $self->debug
278 } elsif ( $attr_type !~ m{^(Str|Int)$} || $value =~ $Frey::Web::re_html || $name =~ m{text} ) {
279 $value_html = qq|<textarea name="$name" title="$attr_type">$value</textarea>|;
280 }
281
282 $label_title = qq| title="| . $attr->documentation . qq|"| if $attr->has_documentation;
283
284 $default->{$name} = $value unless defined $default->{$name};
285
286 my $size = ( int( length($value) / $self->input_step_size ) + 1 ) * $self->input_step_size;
287 $value_html = qq|<input type="$type" name="$name" title="$attr_type" value="$value" size="$size">| unless $value_html;
288
289 # warn "# required $name ", $class->meta->get_attribute( $name )->dump( 2 );
290
291 if ( $required->{$name} ) {
292 $label_title .= qq| class="required"|;
293 $value_html =~ s{(<\S+)\s}{$1 class=required };
294 }
295 $label =~ s/_/ /g;
296
297 my $set = $name;
298 $set =~ s{_[^_]+$}{};
299
300 my ( $before, $after ) = ( '', '<br>' );
301
302 if ( my $s = $fieldset->{$set} ) {
303 if ($s->[0] eq $name) {
304 $before = qq|
305 <fieldset>
306 <legend>$set</legend>
307 |;
308 } elsif ( $s->[ -1 ] eq $name ) {
309 $after = qq|
310 </fieldset>
311 |;
312 }
313 $label =~ s{^\Q$set\E\s+}{};
314 }
315
316 $form .= qq|$before<label for="$name"$label_title>$label</label>$value_html $after|;
317 my $ll = length($label);
318 $label_width = $ll if $ll > $label_width;
319 }
320 $form .= qq|<input type="hidden" name="frey-checkboxes" value="| . join(' ', @checkboxes) . qq|">| if @checkboxes;
321
322 $label_width += 2; # XXX padding left+right em
323
324 $self->add_js('static/Frey/Action.js');
325
326 $self->add_css(qq|
327 label,input {
328 display: block;
329 float: left;
330 margin-bottom: 10px;
331 }
332
333 input:focus {
334 border-color: #cc0;
335 background: #ffc;
336 }
337
338 label {
339 text-align: right;
340 width: ${label_width}ex;
341 padding-right: 1ex;
342 white-space: nowrap;
343 }
344
345 label.required {
346 font-weight: bold;
347 }
348 input.required,
349 select.required {
350 border-color: #c00;
351 }
352
353 br {
354 clear: left;
355 }
356
357 fieldset {
358 margin: 0;
359 padding: 0;
360 margin-bottom: 0.5em;
361 }
362 |);
363
364 my $html;
365
366 # http://www.quirksmode.org/oddsandends/forms.html
367 # $form =~ s{<([^>]+)(name=")([^"]+)(")([^>]*)>}{<$1$2$3$4 id="$3" $5}gs;
368
369 $html = qq|
370 <h1>$class params</h1>
371 <form name="$form_id" id="$form_id" method="post">
372 $form
373 <input type="submit" value="Run $class">
374 </form>
375 | if $form;
376
377 $self->add_status({
378 $self->class => {
379 params => $self->params,
380 params_config => $params_config,
381 default => $default,
382 },
383 });
384
385 return ($html,$default) if wantarray;
386 return $html;
387 }
388
389 =head1 SEE ALSO
390
391 L<http://www.quirksmode.org/css/forms.html> for info on CSS2 forms
392
393 =cut
394
395 1;

  ViewVC Help
Powered by ViewVC 1.1.26