source: text-formbuilder/trunk/lib/Text/FormBuilder/grammar @ 66

Last change on this file since 66 was 66, checked in by peichman, 20 years ago

worked on the rules for parsing lists in the grammar
allow end of field line comments
growable fields can specify a limit; e.g. "person*4" to limit to 4
use '+other' to specify that a field should have an 'Other' option (FB 3.02)
rearranged TODOs in the main documentation

File size: 7.0 KB
RevLine 
[66]1{
2    #$::RD_TRACE = 1;
[21]3    my (
4        $context,      # line or group
[29]5        @sections,     # master data structure
6        $section_head,
7        $section_id,
8        @lines,        # lines in each section
[21]9        $title,
10        $author,
11        $description,
12        %lists,
13        %patterns,
[39]14        %subs,         # validation subs
[21]15        @group,        # current group
16        %groups,       # stored groups of fields
17        $type,
18        @options,
[24]19        $required,
[21]20        $list_var,
21        $size,
[56]22        $maxlength,
[21]23        $rows,
24        $cols,
25    );
26    $context = 'line';
27}
[1]28
[50]29form_spec: (list_def | description_def | validate_def | group_def | note | line)(s)
[1]30    {
[29]31        # grab the last section, if there is any
32        if (@lines) {
33            push @sections,
34                {
35                    id   => $section_id,
36                    head => $section_head,
37                    lines => [ @lines ],
38                };
39        }
40       
41        $section_id = $item{identifier};
42        $section_head = $item[3];
43        @lines = ();
[1]44        $return = {
45            title    => $title,
46            author   => $author,
[14]47            description => $description,
[28]48            lists    => \%lists,
49            patterns => \%patterns,
[39]50            subs     => \%subs,
[28]51            groups   => \%groups,
[29]52            sections => \@sections,
[1]53        }
54    }
55
[16]56list_def: '!list' var_name (static_list | dynamic_list)
57    { $lists{$item{var_name}} = [ @options ]; @options = () }
[1]58
[66]59static_list: '{' /\s*/ option(s /\s*,\s*/) /,?/ /\s*/ '}'
[10]60
61dynamic_list: '&' <perl_codeblock>
62    {
[64]63        warn "[Text::FormBuilder] Dynamic lists have been removed from the formspec grammar";
[10]64    }
65
[57]66description_def: '!description' block
[64]67    {
68        warn "[Text::FormBuilder] Description redefined at input text line $thisline\n" if defined $description;
69        $description = $item{block};
[14]70    }
71
[39]72validate_def: '!validate' var_name <perl_codeblock>
[42]73    { $subs{$item{var_name}} = eval "sub $item[3]" }
[39]74
[21]75group_def: '!group' { $context = 'group' } var_name '{' field_line(s) '}' { $context = 'line' }
76    {
77        #warn "$item{var_name} group; context $context\n"
78        $groups{$item{var_name}} = [ @group ];
79        @group = ();
80    }
[1]81
[64]82note: '!note' block  { push @lines, [ 'note', $item{block} ]; }
[50]83
[64]84
85#TODO: allow \ escape for [] {} in these blocks
86
[57]87# curly-brace delimited block, that can contain properly
[64]88# nested curly braces, along with any other characters
89# inner blocks return with the '{...}' so that nested
90# blocks get the braces treated as literals
91block: '{' <skip:''> block_content(s) '}' { join('', @{ $item[3] }) }
92inner_block: '{' <skip:''> block_content(s) '}'  { '{' . join('', @{ $item[3] }) . '}' }
93block_content: /[^\{\}]+?/ | inner_block
[50]94
[64]95# square brace delimited block, that can contain properly
96# nested square brackets, along with any other characters
97# inner bracket blocks return with the '[...]' so that nested
98# blocks get the braces treated as literals
99bracket_block: '[' <skip:''> bracket_block_content(s) ']' { join('', @{ $item[3] }) }
100inner_bracket_block: '[' <skip:''> bracket_block_content(s) ']' { '[' . join('', @{ $item[3] }) . ']'; }
101bracket_block_content: /[^\[\]]+?/ | inner_bracket_block
[57]102
103
[64]104# field lines are the subset of lines that are allowed in a !group directive
[21]105field_line: <skip:'[ \t]*'> ( field | comment | blank ) "\n"
[64]106
[63]107line: <skip:'[ \t]*'> ( title | author | pattern_def | section_head | heading | group_field | field_group | unknown_directive | field | comment | blank ) "\n"
[21]108
[1]109title: '!title' /.*/
[28]110    {
[57]111        warn "[Text::FormBuilder] Title redefined at input text line $thisline\n" if defined $title;
[28]112        $title = $item[2];
113    }
[1]114
115author: '!author' /.*/
[28]116    {
[57]117        warn "[Text::FormBuilder] Author redefined at input text line $thisline\n" if defined $author;
[28]118        $author = $item[2];
119    }
[1]120
[16]121pattern_def: '!pattern' var_name pattern
122    { $patterns{$item{var_name}} = $item{pattern} }
[1]123
124pattern: /.*/
125
[29]126section_head: '!section' identifier /.*/
127    {
128        #warn "starting section $item{identifier}\n";
129        #warn "  with heading $item[3]\n" if $item[3];
130       
131        if (@lines) {
132            push @sections,
133                {
134                    id   => $section_id,
135                    head => $section_head,
136                    lines => [ @lines ],
137                };
138        }
139       
140        $section_id = $item{identifier};
141        $section_head = $item[3];
142        @lines = ();
143    }
144
[28]145heading: '!head' /.*/    { push @lines, [ 'head', $item[2] ] }
[11]146
[21]147group_field: '!field' group_name name label(?)
[28]148    {
149        push @lines, [ 'group', { name => $item{name}, label => $item{'label(?)'}[0], group => $item{group_name} } ];
[21]150    }
151
152group_name: /%[A-Z_]+/
153
[63]154field_group: name label(?) group_type
155    {
156        #warn "[$thisline] field $item{name} is $item{group_type}\n";
157        push @lines, [ 'group', { name => $item{name}, label => $item{'label(?)'}[0], group => $item{group_type} } ];
158    }
159
160group_type: ':' var_name
161
[64]162# this is the real heart of the thing
[66]163field: name field_size(?) growable(?) label(?) hint(?) type(?) other(?) default(?) option_list(?) validate(?) comment(?)
[1]164    {
165        my $field = {
166            name     => $item{name},
[66]167            growable => $item{'growable(?)'}[0],
[1]168            label    => $item{'label(?)'}[0],
169            comment  => $item{'hint(?)'}[0],
170            type     => $item{'type(?)'}[0],
[66]171            other    => $item{'other(?)'}[0],
[1]172            value    => $item{'default(?)'}[0],
[63]173            list     => $list_var,
174            validate => $item{'validate(?)'}[0],
[24]175            required => $required || 0,
[1]176        };
177       
178        $$field{options} = [ @options ] if @options;
179       
180        $$field{rows} = $rows if defined $rows;
181        $$field{cols} = $cols if defined $cols;
182        $$field{size} = $size if defined $size;
[56]183        $$field{maxlength} = $maxlength if defined $maxlength;
[1]184       
[28]185        #warn "[$thisline] field $item{name}; context $context\n";
[21]186        if ($context eq 'group') {
187            push @group, $field;
188        } else {
189            push @lines, [ 'field', $field ];
190        }
[1]191       
192        $type = undef;
[24]193        $required = 0;
[1]194        $list_var = undef;
195        $size = undef;
196        $rows = undef;
197        $cols = undef;
[56]198        $maxlength = undef;
[1]199        @options = ();
[63]200       
[62]201        $field;
[1]202    }
203   
204name: identifier
205
[16]206var_name: /[A-Z_]+/
207
[1]208field_size: '[' ( row_col | size ) ']'
209
[56]210size: /\d+/ bang(?)
211    { $maxlength = $item[1] if $item[2][0]; $size = $item[1] }
[1]212
[56]213bang: '!'
214
[1]215row_col: /\d+/ /,\s*/ /\d+/
216    { $rows = $item[1]; $cols = $item[3] }
217
[66]218growable: '*' limit(?) { $item{'limit(?)'}[0] || 1 }
[61]219
[66]220limit: /\d+/
221
[22]222label: '|' (simple_multiword | quoted_string) { $item[2] }
[1]223
[64]224hint: bracket_block
[1]225
[63]226type: ':' builtin_field
[1]227
[55]228builtin_field: /textarea|text|password|file|checkbox|radio|select|hidden|static/
229
[66]230other: '+' 'other' { 1 }
[55]231
[22]232default: '=' (simple_multiword | quoted_string) { $item[2] }
[1]233
[22]234# for simple multiword values not involving punctuation
[66]235simple_multiword: <skip:''> /\w[\w\t ]+/ { $item[2] }
[22]236
237# my attempt at a single-quoted, non-interpolating string
238# where the backslash can escape literal single quotes
239quoted_string: <skip:''> "'" /(\\'|[^'])*/ "'"
240    { $item[3] =~ s/\\'/'/g; $item[3] }
241
[1]242option_list: options | list_var
243   
[66]244options: '{' option(s /,/) '}'
[1]245
246list_var: /@[A-Z_]+/ { $list_var = $item[1] }
247
[66]248option: (simple_multiword | quoted_string) display_text(?)
[23]249    { push @options, { $item[1] => $item{'display_text(?)'}[0] } }
[1]250
251value: identifier
252
[64]253display_text: bracket_block
[1]254
[59]255
[64]256validate: '//' (optional_pattern | required_pattern)
[59]257
[64]258optional_pattern: var_name '?'  { $required = 0; $item[1] }
[1]259
[64]260required_pattern: var_name { $required = 1; $item[1] }
[24]261
[1]262comment: '#' /.*/
263blank:
264
265identifier: /\w+/
[16]266
267# skip unknown directives with a warning
268unknown_directive: /\!\S*/ /.*/
269    { warn "[Text::Formbuilder] Skipping unknown directive '$item[1]' at input text line $thisline\n"; }
Note: See TracBrowser for help on using the repository browser.