/[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

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

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

revision 595 by dpavlin, Fri Nov 28 17:18:56 2008 UTC revision 1119 by dpavlin, Tue Jun 30 10:09:43 2009 UTC
# Line 12  use Data::Dump qw/dump/; Line 12  use Data::Dump qw/dump/;
12  Invoke any L<Frey> object creating html for with various default parameters  Invoke any L<Frey> object creating html for with various default parameters
13  if not supplied at invocation.  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  =cut
21    
22  has 'class' => (  has 'class' => (
# Line 26  has 'params' => ( Line 31  has 'params' => (
31          default => sub { {} },          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  =head2 required
43    
44    my @required_attributes = $self->required;    my @required_attributes = $self->required;
45    my $required_attributes = $self->required;    my $required_attributes = $self->required;
46      my $required_hash       = $self->required('as_hash');
47    
48  =cut  =cut
49    
50  sub required {  sub required {
51          my ( $self ) = @_;          my ($self,$param) = @_;
52          $self->load_class( $self->class );          $self->load_class( $self->class );
53    
54          my @required =          my @required =
55                  grep {                  grep {
56                          defined $_ && $_->can('name') &&                          defined $_ && $_->can('name') &&
# Line 47  sub required { Line 62  sub required {
62                          blessed $attr && $attr->is_required && $attr;                          blessed $attr && $attr->is_required && $attr;
63                  } $self->class->meta->get_attribute_list;                  } $self->class->meta->get_attribute_list;
64    
65          warn "## required = ",dump( map { $_->name } @required ), " for ", $self->class if @required && $self->debug;          @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;          return @required if wantarray;
74          return \@required;          return \@required;
75  }  }
# Line 67  sub attributes { Line 89  sub attributes {
89          my $a;          my $a;
90          my @attrs = @{ $self->attribute_order };          my @attrs = @{ $self->attribute_order };
91          @attrs = map {  $a->{$_}++; $_ } @attrs;          @attrs = map {  $a->{$_}++; $_ } @attrs;
92          push @attrs, $_ foreach grep { ! $a->{$_} } map { $_->name } @{ $self->required };          push @attrs, $_ foreach grep { ! $a->{$_} } @{ $self->required };
93          warn "# attributes = ",dump( @attrs ) if $self->debug;          warn "# attributes = ",dump( @attrs ) if $self->debug;
94          return @attrs if wantarray;          return @attrs if wantarray;
95          return \@attrs;          return \@attrs;
# Line 80  sub attributes { Line 102  sub attributes {
102    
103  =cut  =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 {  sub params_form {
186          my ( $self ) = @_;          my ( $self ) = @_;
187          my @required = $self->required;  
188          if ( ! @required ) {          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;                  warn "all params available ", dump( $self->params ), " not creating form" if $self->debug;
200                  return (undef,$self->params) if wantarray;                  return (undef,$self->params) if wantarray;
201                  return;                  return;
         } else {  
                 warn $self->class, " required params ", map { $_->dump(2) } @required if $self->debug;  
202          }          }
203    
204          my $class = $self->class;          my $class = $self->class;
# Line 97  sub params_form { Line 207  sub params_form {
207    
208          my $default = clone $self->params; # XXX we really don't want to modify params!          my $default = clone $self->params; # XXX we really don't want to modify params!
209    
210          my $config_params = {};          my $params_config = {};
211          $config_params = $self->config($class);          $params_config = $self->config($class);
212          warn "# $class config = ",dump( $config_params ) if $self->debug;          warn "# $class config = ",dump( $params_config ) if $self->debug;
213    
214          my $form;          my $form;
215            my $form_id = $self->form_id;
216    
217          sub select_values {          my @checkboxes;
218                  my ( $name, $attr_type, $values ) = @_;  
219                  my $options = join("\n",          my $label_width = 1; # minimum
220                          map {  
221                                  my $v = ref($_) eq 'HASH' ? $_->{$name} : $_;          my @fields =
222                                  qq|<option value="$v">$v</option>| if $v;                  grep {
223                          } @$values                          die "$_ doesn't have meta" unless $class->can('meta');
224                  );                          ! $class->meta->get_attribute($_)->is_lazy
225                  qq|<select title="$attr_type" name="$name">$options</select>| if $options;  #                       && ! 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          foreach my $name ( grep { ! $class->meta->get_attribute($_)->is_lazy } $self->attributes ) {          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 = '';                  my $attr_type = '';
245                  my $type = $name =~ m/^pass/ ? 'password' : 'text';                  my $type = $name =~ m/^pass/ ? 'password' : 'text';
246                  my $label = $name;                  my $label = $name;
                 my $value = '';  
247                  my $label_title = '';                  my $label_title = '';
248                  my $value_html = '';                  my $value_html = '';
249    
250                  my $attr = $class->meta->get_attribute( $name );                  my $attr = $class->meta->get_attribute( $name );
251                  $attr_type = $attr->type_constraint->name if $attr->has_type_constraint;                  $attr_type = $attr->type_constraint->name if $attr->has_type_constraint;
252    
253                  $value = $attr->default( $name ) if ! $value && $attr->has_default;                  my $value =
254                            defined $default->{$name} ? $default->{$name}       :
255                  if ( ref($config_params) eq 'HASH' ) {                          $attr->has_default        ? $attr->default( $name ) :
256                          $value = $config_params->{$name};                          undef;
257                  } elsif ( ref($config_params) eq 'ARRAY' ) {  
258                          $value_html = select_values( $name, $attr_type, $config_params );                  if ( ref($params_config) eq 'HASH' && defined $params_config->{$name} ) {
259                          $default->{$name} = $config_params->[0]->{$name};                          $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') ) {                  } elsif ( $attr->has_type_constraint && $attr->type_constraint->can('values') ) {
264                                  $value_html = select_values( $name, $attr_type, $attr->type_constraint->values );                          $value_html = $self->select_values( $name, $attr_type, $attr->type_constraint->values );
265                  } elsif ( $attr_type !~ m{^(Str|Int)$} ) {                  } elsif ( $class->can( $name . '_available' ) ) {
266                                  $value_html = qq|<textarea name="$name" title="$attr_type">$value</textarea>|;                          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|Email)$} || $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;                  $label_title = qq| title="| . $attr->documentation . qq|"| if $attr->has_documentation;
283    
284                  $default->{$name} = $value unless defined $default->{$name};                  $default->{$name} = $value unless defined $default->{$name};
285    
286                  if ( ! $value_html ) {                  my $size = ( int( length($value) / $self->input_step_size ) + 1 ) * $self->input_step_size;
287                          my $suffix = '';                  $value_html = qq|<input type="$type" name="$name" title="$attr_type" value="$value" size="$size">| unless $value_html;
288                          if ( $attr_type =~ m{^Bool$} ) {  
289                                  $type = 'checkbox';  #               warn "# required $name ", $class->meta->get_attribute( $name )->dump( 2 );
290                                  $suffix = ' checked' if $value;  
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                          $value_html = qq|<input type="$type" name="$name" title="$attr_type" value="$value"$suffix>|;                          $label =~ s{^\Q$set\E\s+}{};
314                  }                  }
315    
316  #               warn "# required $name ", $class->meta->get_attribute( $name )->dump( 2 );                  $form .= qq|$before<label for="$name"$label_title>$label</label>$value_html $after|;
317                  $form .= qq|<label for="$name"$label_title>$label</label>| . $value_html;                  my $ll = length($label);
318                    $label_width = $ll if $ll > $label_width;
319          }          }
320          my $html = qq|<h1>$class params</h1><form method="post">$form<input type="submit" value="Run $class"></form>|;          $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({          $self->add_status({
378                  $self->class => {                  $self->class => {
379                          params  => $self->params,                          params  => $self->params,
380                          config_params  => $config_params,                          params_config  => $params_config,
381                          default => $default                          default => $default,
382                  },                  },
383          });          });
384    
# Line 167  sub params_form { Line 386  sub params_form {
386          return $html;          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;  1;

Legend:
Removed from v.595  
changed lines
  Added in v.1119

  ViewVC Help
Powered by ViewVC 1.1.26