/[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 832 - (show annotations)
Sat Dec 13 16:37:07 2008 UTC (15 years, 4 months ago) by dpavlin
File size: 6908 byte(s)
 r919@eeepy:  dpavlin | 2008-12-13 17:35:22 +0100
 render attributes with text in name as textarea

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 =cut
16
17 has 'class' => (
18 is => 'rw',
19 isa => 'Str',
20 required => 1,
21 );
22
23 has 'params' => (
24 is => 'rw',
25 isa => 'HashRef',
26 default => sub { {} },
27 );
28
29 has 'input_step_size' => (
30 documentation => 'Resize input fields by this step',
31 is => 'rw',
32 isa => 'Int',
33 # required => 1,
34 default => 20,
35 );
36
37 =head2 required
38
39 my @required_attributes = $self->required;
40 my $required_attributes = $self->required;
41
42 =cut
43
44 sub required {
45 my ( $self ) = @_;
46 $self->load_class( $self->class );
47
48 my @required =
49 grep {
50 defined $_ && $_->can('name') &&
51 ! defined( $self->params->{ $_->name } ) &&
52 ! $_->is_lazy
53 }
54 map {
55 my $attr = $self->class->meta->get_attribute($_);
56 blessed $attr && $attr->is_required && $attr;
57 } $self->class->meta->get_attribute_list;
58
59 warn "## required = ",dump( map { $_->name } @required ), " for ", $self->class if @required && $self->debug;
60 return @required if wantarray;
61 return \@required;
62 }
63
64 =head2 attributes
65
66 Generated from attributes specified in code (extracted using L<Frey::PPI>)
67 and required atributes
68
69 my @class_attributes = $self->attributes;
70 my @class_attributes = $self->attributes;
71
72 =cut
73
74 sub attributes {
75 my ( $self ) = @_;
76 my $a;
77 my @attrs = @{ $self->attribute_order };
78 @attrs = map { $a->{$_}++; $_ } @attrs;
79 push @attrs, $_ foreach grep { ! $a->{$_} } map { $_->name } @{ $self->required };
80 warn "# attributes = ",dump( @attrs ) if $self->debug;
81 return @attrs if wantarray;
82 return \@attrs;
83 }
84
85 =head2 params_form
86
87 my $html = $self->params_form;
88 my ($html,$default_params) = $self->params_form;
89
90 =cut
91
92 sub params_form {
93 my ( $self ) = @_;
94 my @required = $self->required;
95 if ( ! @required ) {
96 warn "all params available ", dump( $self->params ), " not creating form" if $self->debug;
97 return (undef,$self->params) if wantarray;
98 return;
99 } else {
100 warn $self->class, " required params ", map { $_->dump(2) } @required if $self->debug;
101 }
102
103 my $class = $self->class;
104
105 $self->load_class( $class );
106
107 my $default = clone $self->params; # XXX we really don't want to modify params!
108
109 my $params_config = {};
110 $params_config = $self->config($class);
111 warn "# $class config = ",dump( $params_config ) if $self->debug;
112
113 my $form;
114
115 sub select_values {
116 my ( $name, $attr_type, $values ) = @_;
117
118 my $max_value_len = 0;
119 my @values;
120 my $html = '';
121
122 foreach ( @$values ) {
123 my $v = ref($_) eq 'HASH' ? $_->{$name} : $_;
124 warn "## value '$v'";
125 push @values, $v;
126 $max_value_len = length($v) if length($v) > $max_value_len;
127 }
128
129 warn "# max_value_len: $max_value_len";
130
131 if ( $#values > 3 ) {
132 my $options = join("\n",
133 map {
134 qq|<option value="$_">$_</option>|;
135 } @values
136 );
137 # onchange="alert(this.options[this.selectedIndex].value);"
138 $html = qq|
139 <select title="$attr_type" name="$name">
140 $options
141 </select>
142 | if $options;
143 } else {
144 my $delimiter = $max_value_len > $self->input_step_size ? qq|<br>| : '';
145 my $radio =
146 # $delimiter .
147 join("\n",
148 map { strip(qq|
149 <span title="$attr_type">
150 <input type="radio" name="$name" value="$_">
151 $_
152 </span>
153 $delimiter
154 |) } @values
155 );
156 $html = qq|<div style="display: block;">$radio</div>|;
157 }
158
159 return
160 # qq|<input type="text" name="$name">| .
161 $html
162 }
163
164 foreach my $checkbox ( split(/\s+/, $default->{'frey-checkboxes'} ) ) {
165 next if defined $default->{ $checkbox };
166
167 $default->{ $checkbox } = 0;
168 $self->params->{ $checkbox } = 0;
169 warn "# checkbox $checkbox not ticked";
170 }
171
172 my @checkboxes;
173
174 my $label_width = 1; # minimum
175
176 foreach my $name (
177 grep {
178 ! $class->meta->get_attribute($_)->is_lazy
179 && ! defined $default->{$_}
180 && ! m{^_} # skip _private
181 } $self->attributes
182 ) {
183 my $attr_type = '';
184 my $type = $name =~ m/^pass/ ? 'password' : 'text';
185 my $label = $name;
186 my $label_title = '';
187 my $value_html = '';
188
189 my $attr = $class->meta->get_attribute( $name );
190 $attr_type = $attr->type_constraint->name if $attr->has_type_constraint;
191
192 my $value =
193 defined $default->{$name} ? $default->{$name} :
194 $attr->has_default ? $attr->default( $name ) :
195 '';
196
197 if ( ref($params_config) eq 'HASH' && defined $params_config->{$name} ) {
198 $value = $params_config->{$name};
199 } elsif ( ref($params_config) eq 'ARRAY' ) {
200 $value_html = select_values( $name, $attr_type, $params_config );
201 $default->{$name} = $params_config->[0]->{$name};
202 } elsif ( $attr->has_type_constraint && $attr->type_constraint->can('values') ) {
203 $value_html = select_values( $name, $attr_type, $attr->type_constraint->values );
204 } elsif ( $attr_type =~ m{^Bool} ) {
205 my $suffix = '';
206 $suffix = ' checked' if $value;
207 $value_html = qq|<input type="checkbox" name="$name" title="$attr_type" value="$value"$suffix>|;
208 push @checkboxes, $name;
209 } elsif ( ! defined $value ) {
210 $value_html = qq|<tt id="$name">undef</tt><!-- $name = undef -->|; # FIXME if $self->debug
211 } elsif ( $attr_type !~ m{^(Str|Int)$} || $value =~ $Frey::Web::re_html || $name =~ m{text} ) {
212 $value_html = qq|<textarea name="$name" title="$attr_type">$value</textarea>|;
213 }
214
215 $label_title = qq| title="| . $attr->documentation . qq|"| if $attr->has_documentation;
216
217 $default->{$name} = $value unless defined $default->{$name};
218
219 my $size = ( int( length($value) / $self->input_step_size ) + 1 ) * $self->input_step_size;
220 $value_html = qq|<input type="$type" name="$name" title="$attr_type" value="$value" size="$size">| unless $value_html;
221
222 # warn "# required $name ", $class->meta->get_attribute( $name )->dump( 2 );
223 $form .= qq|<label for="$name"$label_title>$label</label>$value_html<br>|;
224 my $ll = length($label);
225 $label_width = $ll if $ll > $label_width;
226 }
227 $form .= qq|<input type="hidden" name="frey-checkboxes" value="| . join(' ', @checkboxes) . qq|">| if @checkboxes;
228
229 $self->add_css(qq|
230 label,input {
231 display: block;
232 float: left;
233 margin-bottom: 10px;
234 }
235
236 label {
237 text-align: right;
238 width: ${label_width}ex;
239 padding-right: 20px;
240 }
241
242 br {
243 clear: left;
244 }
245 |);
246
247 my $html;
248
249 # http://www.quirksmode.org/oddsandends/forms.html
250 # $form =~ s{<([^>]+)(name=")([^"]+)(")([^>]*)>}{<$1$2$3$4 id="$3" $5}gs;
251
252 $html = qq|
253 <h1>$class params</h1>
254 <form method="post">
255 $form
256 <input type="submit" value="Run $class">
257 </form>
258 | if $form;
259
260 $self->add_status({
261 $self->class => {
262 params => $self->params,
263 params_config => $params_config,
264 default => $default,
265 },
266 });
267
268 return ($html,$default) if wantarray;
269 return $html;
270 }
271
272 =head1 SEE ALSO
273
274 L<http://www.quirksmode.org/css/forms.html> for info on CSS2 forms
275
276 =cut
277
278 1;

  ViewVC Help
Powered by ViewVC 1.1.26