summaryrefslogtreecommitdiffstats
path: root/template/default/query/query.atml
diff options
context:
space:
mode:
Diffstat (limited to 'template/default/query/query.atml')
-rw-r--r--template/default/query/query.atml778
1 files changed, 778 insertions, 0 deletions
diff --git a/template/default/query/query.atml b/template/default/query/query.atml
new file mode 100644
index 000000000..ab1ce4d48
--- /dev/null
+++ b/template/default/query/query.atml
@@ -0,0 +1,778 @@
+[%# The contents of this file are subject to the Mozilla Public
+ # License Version 1.1 (the "License"); you may not use this file
+ # except in compliance with the License. You may obtain a copy of
+ # the License at http://www.mozilla.org/MPL/
+ #
+ # Software distributed under the License is distributed on an "AS
+ # IS" basis, WITHOUT WARRANTY OF ANY KIND, either express or
+ # implied. See the License for the specific language governing
+ # rights and limitations under the License.
+ #
+ # The Original Code is the Bugzilla Bug Tracking System.
+ #
+ # The Initial Developer of the Original Code is Netscape Communications
+ # Corporation. Portions created by Netscape are
+ # Copyright (C) 1998 Netscape Communications Corporation. All
+ # Rights Reserved.
+ #
+ # Contributor(s): Chris Lahey <clahey@ximian.com> [javascript fixes]
+ # Christian Reis <kiko@async.com.br> [javascript rewrite]
+ # Gervase Markham <gerv@gerv.net>
+ #%]
+
+[% INCLUDE global/header
+ title = "Search for bugs"
+ extra = " onLoad=\"selectProduct(document.forms['queryform']);\""
+%]
+
+[%# Note: use Template comments and not JS ones here, to avoid bloating
+ what we actually send to the browser %]
+
+<script language="JavaScript" type="text/javascript"> <!--
+
+var first_load = 1; [%# is this the first time we load the page? %]
+var last_sel = []; [%# caches last selection %]
+var usetms = [% IF Param('usetargetmilestone') %]true[% ELSE %]false[% END %];
+ [%# do we have target milestone? %]
+
+var cpts = new Object();
+var vers = new Object();
+var tms = new Object();
+
+[% FOREACH p = product %]
+ cpts['[% p FILTER js %]'] = [
+ [%- FOREACH item = componentsbyproduct.$p %]'[% item FILTER js %]', [%- END -%]]
+ vers['[% p FILTER js %]'] = [
+ [%- FOREACH item = versionsbyproduct.$p -%]'[% item FILTER js %]', [%- END -%]]
+ tms['[% p FILTER js %]'] = [
+ [%- FOREACH item = milestonesbyproduct.$p %]'[% item FILTER js %]', [%- END -%]]
+[% END %]
+
+[%# Adds to the target select object all elements in array that
+ # correspond to the elements selected in source.
+ # - array should be a array of arrays, indexed by product name. the
+ # array should contain the elements that correspont to that
+ # product. Example:
+ # var array = Array();
+ # array['ProductOne'] = [ 'ComponentA', 'ComponentB' ];
+ # updateSelect(array, source, target);
+ # - sel is a list of selected items, either whole or a diff
+ # depending on sel_is_diff.
+ # - sel_is_diff determines if we are sending in just a diff or the
+ # whole selection. a diff is used to optimize adding selections.
+ # - target should be the target select object.
+ # - single specifies if we selected a single item. if we did, no
+ # need to merge. %]
+function updateSelect(array, sel, target, sel_is_diff, single) {
+
+ var i, comp;
+
+ [%# if single, even if it's a diff (happens when you have nothing
+ selected and select one item alone), skip this. %]
+ if (!single) {
+ [%# array merging/sorting in the case of multiple selections %]
+ if (sel_is_diff) {
+ [%# merge in the current options with the first selection %]
+ comp = merge_arrays(array[sel[0]], target.options, 1);
+
+ [%# merge the rest of the selection with the results %]
+ for (i = 1 ; i < sel.length ; i++) {
+ comp = merge_arrays(array[sel[i]], comp, 0);
+ }
+ } else {
+ [%# here we micro-optimize for two arrays to avoid merging with a
+ null array %]
+ comp = merge_arrays(array[sel[0]],array[sel[1]], 0);
+
+ [%# merge the arrays. not very good for multiple selections. %]
+ for (i = 2; i < sel.length; i++) {
+ comp = merge_arrays(comp, array[sel[i]], 0);
+ }
+ }
+ } else {
+ [%# single item in selection, just get me the list %]
+ comp = array[sel[0]];
+ }
+
+ [%# clear select %]
+ target.options.length = 0;
+
+ [%# load elements of list into select %]
+ for (i = 0; i < comp.length; i++) {
+ target.options[i] = new Option(comp[i], comp[i]);
+ }
+}
+
+[%# Returns elements in a that are not in b.
+ # NOT A REAL DIFF: does not check the reverse.
+ # - a,b: arrays of values to be compare. %]
+function fake_diff_array(a, b) {
+ var newsel = new Array();
+
+ [%# do a boring array diff to see who's new %]
+ for (var ia in a) {
+ var found = 0;
+ for (var ib in b) {
+ if (a[ia] == b[ib]) {
+ found = 1;
+ }
+ }
+ if (!found) {
+ newsel[newsel.length] = a[ia];
+ }
+ found = 0;
+ }
+ return newsel;
+}
+
+[%# takes two arrays and sorts them by string, returning a new, sorted
+ # array. the merge removes dupes, too.
+ # - a, b: arrays to be merge.
+ # - b_is_select: if true, then b is actually an optionitem and as
+ # such we need to use item.value on it. %]
+function merge_arrays(a, b, b_is_select) {
+ var pos_a = 0;
+ var pos_b = 0;
+ var ret = new Array();
+ var bitem, aitem;
+
+ [%# iterate through both arrays and add the larger item to the return
+ list. remove dupes, too. Use toLowerCase to provide
+ case-insensitivity. %]
+ while ((pos_a < a.length) && (pos_b < b.length)) {
+ if (b_is_select) {
+ bitem = b[pos_b].value;
+ } else {
+ bitem = b[pos_b];
+ }
+ aitem = a[pos_a];
+
+ [%# smaller item in list a %]
+ if (aitem.toLowerCase() < bitem.toLowerCase()) {
+ ret[ret.length] = aitem;
+ pos_a++;
+ } else {
+ [%# smaller item in list b %]
+ if (aitem.toLowerCase() > bitem.toLowerCase()) {
+ ret[ret.length] = bitem;
+ pos_b++;
+ } else {
+ [%# list contents are equal, inc both counters. %]
+ ret[ret.length] = aitem;
+ pos_a++;
+ pos_b++;
+ }
+ }
+ }
+
+ [%# catch leftovers here. these sections are ugly code-copying. %]
+ if (pos_a < a.length) {
+ for (; pos_a < a.length ; pos_a++) {
+ ret[ret.length] = a[pos_a];
+ }
+ }
+
+ if (pos_b < b.length) {
+ for (; pos_b < b.length; pos_b++) {
+ if (b_is_select) {
+ bitem = b[pos_b].value;
+ } else {
+ bitem = b[pos_b];
+ }
+ ret[ret.length] = bitem;
+ }
+ }
+ return ret;
+}
+
+[%# selectProduct reads the selection from f.product and updates
+ # f.version, component and target_milestone accordingly.
+ # - f: a form containing product, component, varsion and
+ # target_milestone select boxes.
+ # globals (3vil!):
+ # - cpts, vers, tms: array of arrays, indexed by product name. the
+ # subarrays contain a list of names to be fed to the respective
+ # selectboxes. For bugzilla, these are generated with perl code
+ # at page start.
+ # - usetms: this is a global boolean that is defined if the
+ # bugzilla installation has it turned on. generated in perl too.
+ # - first_load: boolean, specifying if it is the first time we load
+ # the query page.
+ # - last_sel: saves our last selection list so we know what has
+ # changed, and optimize for additions. %]
+function selectProduct(f) {
+ [%# this is to avoid handling events that occur before the form
+ itself is ready, which happens in buggy browsers. %]
+ if ((!f) || (!f.product)) {
+ return;
+ }
+
+ [%# if this is the first load and nothing is selected, no need to
+ merge and sort all components; perl gives it to us sorted. %]
+ if ((first_load) && (f.product.selectedIndex == -1)) {
+ first_load = 0;
+ return;
+ }
+
+ [%# turn first_load off. this is tricky, since it seems to be
+ redundant with the above clause. It's not: if when we first load
+ the page there is _one_ element selected, it won't fall into that
+ clause, and first_load will remain 1. Then, if we unselect that
+ item, selectProduct will be called but the clause will be valid
+ (since selectedIndex == -1), and we will return - incorrectly -
+ without merge/sorting. %]
+ first_load = 0;
+
+ [%# - sel keeps the array of products we are selected.
+ - is_diff says if it is a full list or just a list of products that
+ were added to the current selection.
+ - single indicates if a single item was selected %]
+ var sel = Array();
+ var is_diff = 0;
+ var single;
+
+ [%# if nothing selected, pick all %]
+ if (f.product.selectedIndex == -1) {
+ for (var i = 0 ; i < f.product.length ; i++) {
+ sel[sel.length] = f.product.options[i].value;
+ }
+ single = 0;
+ } else {
+ for (i = 0 ; i < f.product.length ; i++) {
+ if (f.product.options[i].selected) {
+ sel[sel.length] = f.product.options[i].value;
+ }
+ }
+
+ single = (sel.length == 1);
+
+ [%# save last_sel before we kill it %]
+ var tmp = last_sel;
+ last_sel = sel;
+
+ [%# this is an optimization: if we have added components, no need
+ to remerge them; just merge the new ones with the existing
+ options. %]
+ if ((tmp) && (tmp.length < sel.length)) {
+ sel = fake_diff_array(sel, tmp);
+ is_diff = 1;
+ }
+ }
+
+ [%# do the actual fill/update %]
+ updateSelect(cpts, sel, f.component, is_diff, single);
+ updateSelect(vers, sel, f.version, is_diff, single);
+ if (usetms) {
+ updateSelect(tms, sel, f.target_milestone, is_diff, single);
+ }
+}
+
+// -->
+</script>
+
+[% query_variants = [
+ { value => "allwordssubstr", description => "contains all of the words/strings" },
+ { value => "anywordssubstr", description => "contains any of the words/strings" },
+ { value => "substring", description => "contains the string" },
+ { value => "casesubstring", description => "contains the string (exact case)" },
+ { value => "allwords", description => "contains all of the words" },
+ { value => "anywords", description => "contains any of the words" },
+ { value => "regexp", description => "matches the regexp" },
+ { value => "notregexp", description => "doesn&#8217;t match the regexp" } ] %]
+
+<form method="get" action="buglist.cgi" name="queryform">
+
+[%# *** Summary *** %]
+
+<table>
+ <tr>
+ <th align="right">Summary:</th>
+ <td>
+ <select name="short_desc_type">
+ [% FOREACH qv = query_variants %]
+ <option value="[% qv.value %]"
+ [% " selected" IF default.short_desc_type.0 == qv.value %]>[% qv.description %]</option>
+ [% END %]
+ </select>
+ </td>
+ <td>
+ <input name="short_desc" size="30" value="[% default.short_desc.0 FILTER html %]" />
+ </td>
+ <td>
+ <input type="submit" value="Search" />
+ </td>
+ </tr>
+</table>
+
+[%# *** Product Component Version Target *** %]
+
+<table>
+ <tr>
+ <td>
+ <table>
+ <tr valign="bottom">
+ <th align="left">Program:</th>
+ <th align="left"><a href="describecomponents.cgi">Component</a>:</th>
+ <th align="left">Version:</th>
+
+ [% IF (Param("usetargetmilestone")) %]
+ <th align="left">Target:</th>
+ [% END %]
+ </tr>
+
+ <tr valign="top">
+
+ [%# Can't use the select block here because of onChange and the fact that
+ 'component' is a toolkit reserved word - we use 'component_' instead. %]
+ <td align="left">
+ <select name="product" multiple size="5" onChange="selectProduct(this.form);">
+ [% FOREACH p = product %]
+ <option value="[% p FILTER html %]"
+ [% " selected" IF lsearch(default.product, p) != -1 %]>
+ [% p FILTER html %]</option>
+ [% END %]
+ </select>
+ </td>
+
+ <td align="left">
+ <select name="component" multiple size="5">
+ [% FOREACH c = component_ %]
+ <option value="[% c FILTER html %]"
+ [% " selected" IF lsearch(default.component, c) != -1 %]>
+ [% c FILTER html %]</option>
+ [% END %]
+ </select>
+ </td>
+
+ [% PROCESS select sel = { name => 'version', size => 5 } %]
+
+ [% IF target_milestone.size > 0 %]
+ [% PROCESS select sel = { name => 'target_milestone', size => 5 } %]
+ [% END %]
+ </tr>
+ </table>
+ </td>
+ </tr>
+
+[%# *** Comment URL Whiteboard Keywords *** %]
+
+ <tr>
+ <td>
+ <table border="0">
+ [% FOREACH field = [
+ { name => "long_desc", description => "A comment" },
+ { name => "bug_file_loc", description => "The URL" },
+ { name => "status_whiteboard", description => "Whiteboard" } ] %]
+
+ [% UNLESS field.name == 'status_whiteboard' AND NOT Param('usestatuswhiteboard') %]
+ <tr>
+ <th align="right">[% field.description %]:</th>
+ <td>
+ <select name="[% field.name %]_type">
+ [% FOREACH qv = query_variants %]
+ [% type = "${field.name}_type" %]
+ <option value="[% qv.value %]"
+ [% " selected" IF default.$type.0 == qv.value %]>[% qv.description %]</option>
+ [% END %]
+ </select>
+ </td>
+ <td><input name="[% field.name %]" size="40" value="
+ [% default.${field.name}.0 FILTER html %]" /></td>
+ </tr>
+ [% END %]
+ [% END %]
+
+ [% IF have_keywords %]
+ <tr>
+ <th align="right"><a href="describekeywords.cgi">Keywords</a>:</th>
+ <td>
+ <select name="keywords_type">
+ [% FOREACH qv = [
+ { name => "anywords", description => "contains any of the keywords" },
+ { name => "allwords", description => "contains all of the keywords" },
+ { name => "nowords", description => "contains none of the keywords" } ] %]
+
+ <option value="[% qv.name %]"
+ [% " selected" IF default.keywords_type.0 == qv.name %]>
+ [% qv.description %]</option>
+ [% END %]
+ </select>
+ </td>
+ <td>
+ <input name="keywords" size="40" value="[% default.keywords.0 FILTER html %]" />
+ </td>
+ </tr>
+ [% END %]
+
+ </table>
+ </td>
+ </tr>
+</table>
+
+<hr>
+
+[%# *** Status Resolution Severity Priority Hardware OS *** %]
+
+<table>
+ <tr>
+ <th align="left"><a href="queryhelp.cgi#status">Status</a>:</th>
+ <th align="left"><a href="queryhelp.cgi#resolution">Resolution</a>:</th>
+ <th align="left"><a href="queryhelp.cgi#severity">Severity</a>:</th>
+ <th align="left"><a href="queryhelp.cgi#priority">Priority</a>:</th>
+ <th align="left"><a href="queryhelp.cgi#platform">Hardware</a>:</th>
+ <th align="left"><a href="queryhelp.cgi#opsys">OS</a>:</th>
+ </tr>
+
+ <tr valign="top">
+ [% PROCESS select sel = { name => 'bug_status', size => 7 } %]
+ [% PROCESS select sel = { name => 'resolution', size => 7 } %]
+ [% PROCESS select sel = { name => 'bug_severity', size => 7 } %]
+ [% PROCESS select sel = { name => 'priority', size => 7 } %]
+ [% PROCESS select sel = { name => 'rep_platform', size => 7 } %]
+ [% PROCESS select sel = { name => 'op_sys', size => 7 } %]
+ </tr>
+</table>
+
+<p>
+
+[%# *** Email Numbering Votes *** %]
+
+<table>
+ <tr>
+ <td>
+ <fieldset>
+ <legend>
+ <strong>
+ <a href="queryhelp.cgi#peopleinvolved">Email</a> and Numbering
+ </strong>
+ </legend>
+
+<table>
+ <tr>
+ [% FOREACH n = [1, 2] %]
+ <td>
+
+
+<table cellspacing="0" cellpadding="0">
+ <tr>
+ <td>
+ Any of:
+ </td>
+ </tr>
+ <tr>
+ <td><input type="checkbox" name="emailassigned_to[% n %]" value="1"
+ [% " checked" IF default.emailassigned_to.$n %] />bug owner</td>
+ </tr>
+ <tr>
+ <td><input type="checkbox" name="emailreporter[% n %]" value="1"
+ [% " checked" IF default.emailreporter.$n %] />reporter</td>
+ </tr>
+ [% IF Param('useqacontact') %]
+ <tr>
+ <td><input type="checkbox" name="emailqa_contact[% n %]" value="1"
+ [% " checked" IF default.emailqa_contact.$n %] />QA Contact</td>
+ </tr>
+ [% END %]
+ <tr>
+ <td><input type="checkbox" name="emailcc[% n %]" value="1"
+ [% " checked" IF default.emailcc.$n %] />CC list member</td>
+ </tr>
+ <tr>
+ <td><input type="checkbox" name="emaillongdesc[% n %]" value="1"
+ [% " checked" IF default.emaillongdesc.$n %] />commenter</td>
+ </tr>
+ <tr>
+ <td>
+ <select name="emailtype[% n %]">
+ [% FOREACH qv = [
+ { name => "exact", description => "is" },
+ { name => "substring", description => "contains" },
+ { name => "regexp", description => "matches regexp" },
+ { name => "notregexp", description => "doesn&#8217;t match regexp" } ] %]
+
+ <option value="[% qv.name %]"
+ [% " selected" IF default.emailtype.$n == qv.name %]>[% qv.description %]</option>
+ [% END %]
+ </select>
+ </td>
+ </tr>
+ <tr>
+ <td>
+ <input name="email[% n %]" size="20" value="[% default.email.$n FILTER html %]" />
+ </td>
+ </tr>
+</table>
+
+
+ </td>
+ [% END %]
+ </tr>
+</table>
+<hr>
+<table>
+ <tr>
+ <td>
+ <select name="bugidtype">
+ <option value="include"[% " selected" IF default.bugidtype.0 == "include" %]>Only include</option>
+ <option value="exclude"[% " selected" IF default.bugidtype.0 == "exclude" %]>Exclude</option>
+ </select>
+ bugs numbered:
+ </td>
+ <td>
+ <input type="text" name="bug_id" value="[% default.bug_id.0 FILTER html %]" size="20" />
+ </td>
+ </tr>
+ <tr>
+ <td></td>
+ <td>(comma-separated list)</td>
+ </tr>
+ <tr>
+ <td align="right">
+ Only bugs with at least:
+ </td>
+ <td>
+ <input name="votes" size="3" value="[% default.votes.0 FILTER html %]" /> votes
+ </td>
+ </tr>
+</table>
+
+ </fieldset>
+ </td>
+
+[%# *** Bug Changes *** %]
+
+ <td valign="top">
+ <fieldset>
+ <legend><strong>Bug Changes</strong></legend>
+
+<dl>
+ <dt>Only bugs changed in the last </dt>
+ <dd><input name=changedin size=3 value="[% default.changedin.0 FILTER html %]" /> days</dd>
+</dl>
+
+<dl>
+ <dt>Only bugs where any of the fields</dt>
+ <dd>
+ <select name="chfield" multiple size="4">
+ [% FOREACH field = chfield %]
+ <option value="[% field FILTER html %]"
+ [% " selected" IF lsearch(default.chfield, field) != -1 %]>
+ [% field FILTER html %]</option>
+ [% END %]
+ </select>
+ </dd>
+
+ <dt>were changed between</dt>
+ <dd>
+ <input name="chfieldfrom" size="10" value="[% default.chfieldfrom.0 FILTER html %]" />
+ and <input name="chfieldto" size="10" value="[% default.chfieldto.0 FILTER html %]" />
+ <br>(YYYY-MM-DD)
+ </dd>
+ <dt>to this value: (optional)</dt>
+ <dd>
+ <input name="chfieldvalue" size="20" value="[% default.chfieldvalue.0 FILTER html %]" />
+ </dd>
+</dl>
+
+ </fieldset>
+ </td>
+ </tr>
+
+[%# *** Action Selection *** %]
+
+ <tr>
+ <td colspan="2">
+
+ [% IF NOT userid %]
+ <input type="hidden" name="cmdtype" value="doit" />
+ [% ELSE %]
+ <br>
+ <input type="radio" name="cmdtype" value="doit" checked /> Run this query
+ <br>
+
+ [% IF namedqueries.size > 0 %]
+ <p>
+ <table cellspacing="0" cellpadding="0">
+ <tr>
+ <td>
+ <input type="radio" name="cmdtype" value="editnamed" />
+ Load my remembered query:
+ </td>
+ <td rowspan="3">
+ <select name="namedcmd">
+ [% FOREACH query = namedqueries %]
+ <option value="[% query FILTER html %]">[% query FILTER html %]</option>
+ [% END %]
+ </select>
+ </td>
+ </tr>
+ <tr>
+ <td>
+ <input type="radio" name="cmdtype" value="runnamed" />
+ Run my remembered query:
+ </td>
+ </tr>
+ <tr>
+ <td>
+ <input type="radio" name="cmdtype" value="forgetnamed" />
+ Forget my remembered query:
+ </td>
+ </tr>
+ </table>
+ </p>
+ [% END %]
+
+ <input type="radio" name="cmdtype" value="asdefault" />
+ Remember this as my default query
+ <br>
+ <input type="radio" name="cmdtype" value="asnamed" />
+ Remember this query, and name it:
+ <input type="text" name="newqueryname">
+ <br>&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;<input type="checkbox" name="tofooter" value="1" />
+ and put it in my page footer
+ <br>
+ [% END %]
+ <p>
+ Sort results by:
+ <select name="order">
+ [% FOREACH order = orders %]
+ <option value="[% order FILTER html %]"
+ [% " selected" IF default.order.0 == order %]>[% order FILTER html %]</option>
+ [% END %]
+ </select>
+
+ <input type="submit" value="Submit query" />
+ [% IF userdefaultquery %]
+ <p>
+ <a href="query.cgi?nukedefaultquery=1">
+ Set my default query back to the system default</a>
+ </p>
+ [% END %]
+ </p>
+ </td>
+ </tr>
+</table>
+
+[%# *** Boolean Charts *** %]
+
+<hr>
+
+[% types = [
+ { name => "noop", description => "---" },
+ { name => "equals", description => "is equal to" },
+ { name => "notequals", description => "is not equal to" },
+ { name => "substring", description => "contains the string" },
+ { name => "casesubstring", description => "contains the string (exact case)" },
+ { name => "notsubstring", description => "does not contain the string" },
+ { name => "allwordssubstr", description => "contains all of the strings" },
+ { name => "anywordssubstr", description => "contains any of the strings" },
+ { name => "regexp", description => "contains regexp" },
+ { name => "notregexp", description => "does not contain regexp" },
+ { name => "lessthan", description => "is less than" },
+ { name => "greaterthan", description => "is greater than" },
+ { name => "anywords", description => "contains any of the words" },
+ { name => "allwords", description => "contains all of the words" },
+ { name => "nowords", description => "contains none of the words" },
+ { name => "changedbefore", description => "changed before" },
+ { name => "changedafter", description => "changed after" },
+ { name => "changedfrom", description => "changed from" },
+ { name => "changedto", description => "changed to" },
+ { name => "changedby", description => "changed by" } ] %]
+
+ <p>
+ <strong>
+ <a name="chart" href="queryhelp.cgi#advancedquerying">
+ Advanced Querying Using Boolean Charts</a>:
+ </strong>
+ </p>
+
+[%# Whoever wrote the original version of boolean charts had a seriously twisted mind %]
+
+[% jsmagic = "onclick=\"document.forms[0].action='query.cgi#chart'; document.forms[0].method='POST'; return 1;\"" %]
+
+[% FOREACH chart = default.charts %]
+ [% chartnum = loop.count - 1 %]
+ <table>
+ [% FOREACH row = chart %]
+ [% rownum = loop.count - 1 %]
+ <tr>
+ [% FOREACH col = row %]
+ [% colnum = loop.count - 1 %]
+ <td>
+ <select name="[% "field${chartnum}-${rownum}-${colnum}" %]">
+ [% FOREACH field = fields %]
+ <option value="[% field.name %]"
+ [%- " selected" IF field.name == col.field %]>[% field.description %]</option>
+ [% END %]
+ </select>
+
+ <select name="[% "type${chartnum}-${rownum}-${colnum}" %]">
+ [% FOREACH type = types %]
+ <option value="[% type.name %]"
+ [%- " selected" IF type.name == col.type %]>[% type.description %]</option>
+ [% END %]
+ </select>
+
+ <input name="[% "value${chartnum}-${rownum}-${colnum}" %]"
+ value="[% col.value FILTER html %]" />
+ </td>
+
+ [% IF NOT col == row.last %]
+ <td align="center">
+ Or
+ </td>
+ [% ELSE %]
+ <td>
+ [% newor = colnum + 1 %]
+ <input type="submit" value="Or"
+ name="cmd-add[% "${chartnum}-${rownum}-${newor}" %]" [% $jsmagic %] />
+ </td>
+ [% END %]
+
+ [% END %]
+ </tr>
+
+ [% IF NOT row == chart.last %]
+ <tr>
+ <td>And</td>
+ </tr>
+ [% ELSE %]
+ <tr>
+ <td>
+ [% newand = rownum + 1; newchart = chartnum + 1 %]
+ <input type="submit" value="And"
+ name="cmd-add[% "${chartnum}-${newand}-0" %]"[% $jsmagic %] />
+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
+ <input type="submit" value="Add another boolean chart"
+ name="cmd-add[% newchart %]-0-0" [% $jsmagic %] />
+ &nbsp;&nbsp;&nbsp;&nbsp;&nbsp;
+ </td>
+ </tr>
+ [% END %]
+
+ [% END %]
+ </table>
+ <hr>
+[% END %]
+
+<p>Give me a <a href="queryhelp.cgi">clue</a> about how to use this form.</p>
+
+</FORM>
+
+[% INCLUDE global/footer %]
+
+[%############################################################################%]
+[%# Block for SELECT fields #%]
+[%############################################################################%]
+
+[% BLOCK select %]
+ <td align="left">
+ <select name="[% sel.name %]" multiple size="[% sel.size %]">
+ [% FOREACH name = ${sel.name} %]
+ <option value="[% name FILTER html %]"
+ [% " selected" IF lsearch(default.${sel.name}, name) != -1 %]>
+ [% name FILTER html %]</option>
+ [% END %]
+ </select>
+ </td>
+[% END %]