Commit da12cec6 authored by Byron Jones's avatar Byron Jones

Bug 780820: Allows for multiple custom search criteria to match one field

r=dkl,a=LpSolit
parent ceafb43c
...@@ -385,8 +385,10 @@ sub sql_string_until { ...@@ -385,8 +385,10 @@ sub sql_string_until {
} }
sub sql_in { sub sql_in {
my ($self, $column_name, $in_list_ref) = @_; my ($self, $column_name, $in_list_ref, $negate) = @_;
return " $column_name IN (" . join(',', @$in_list_ref) . ") "; return " $column_name "
. ($negate ? "NOT " : "")
. "IN (" . join(',', @$in_list_ref) . ") ";
} }
sub sql_fulltext_search { sub sql_fulltext_search {
......
...@@ -203,16 +203,16 @@ sub sql_position { ...@@ -203,16 +203,16 @@ sub sql_position {
} }
sub sql_in { sub sql_in {
my ($self, $column_name, $in_list_ref) = @_; my ($self, $column_name, $in_list_ref, $negate) = @_;
my @in_list = @$in_list_ref; my @in_list = @$in_list_ref;
return $self->SUPER::sql_in($column_name, $in_list_ref) if $#in_list < 1000; return $self->SUPER::sql_in($column_name, $in_list_ref, $negate) if $#in_list < 1000;
my @in_str; my @in_str;
while (@in_list) { while (@in_list) {
my $length = $#in_list + 1; my $length = $#in_list + 1;
my $splice = $length > 1000 ? 1000 : $length; my $splice = $length > 1000 ? 1000 : $length;
my @sub_in_list = splice(@in_list, 0, $splice); my @sub_in_list = splice(@in_list, 0, $splice);
push(@in_str, push(@in_str,
$self->SUPER::sql_in($column_name, \@sub_in_list)); $self->SUPER::sql_in($column_name, \@sub_in_list, $negate));
} }
return "( " . join(" OR ", @in_str) . " )"; return "( " . join(" OR ", @in_str) . " )";
} }
......
...@@ -23,6 +23,7 @@ use Bugzilla::Group; ...@@ -23,6 +23,7 @@ use Bugzilla::Group;
use Bugzilla::User; use Bugzilla::User;
use Bugzilla::Field; use Bugzilla::Field;
use Bugzilla::Search::Clause; use Bugzilla::Search::Clause;
use Bugzilla::Search::ClauseGroup;
use Bugzilla::Search::Condition qw(condition); use Bugzilla::Search::Condition qw(condition);
use Bugzilla::Status; use Bugzilla::Status;
use Bugzilla::Keyword; use Bugzilla::Keyword;
...@@ -1152,7 +1153,7 @@ sub _translate_join { ...@@ -1152,7 +1153,7 @@ sub _translate_join {
die "join with no table: " . Dumper($join_info) if !$join_info->{table}; die "join with no table: " . Dumper($join_info) if !$join_info->{table};
die "join with no 'as': " . Dumper($join_info) if !$join_info->{as}; die "join with no 'as': " . Dumper($join_info) if !$join_info->{as};
my $from_table = "bugs"; my $from_table = $join_info->{bugs_table} || "bugs";
my $from = $join_info->{from} || "bug_id"; my $from = $join_info->{from} || "bug_id";
if ($from =~ /^(\w+)\.(\w+)$/) { if ($from =~ /^(\w+)\.(\w+)$/) {
($from_table, $from) = ($1, $2); ($from_table, $from) = ($1, $2);
...@@ -1557,7 +1558,7 @@ sub _charts_to_conditions { ...@@ -1557,7 +1558,7 @@ sub _charts_to_conditions {
my $clause = $self->_charts; my $clause = $self->_charts;
my @joins; my @joins;
$clause->walk_conditions(sub { $clause->walk_conditions(sub {
my ($condition) = @_; my ($clause, $condition) = @_;
return if !$condition->translated; return if !$condition->translated;
push(@joins, @{ $condition->translated->{joins} }); push(@joins, @{ $condition->translated->{joins} });
}); });
...@@ -1577,7 +1578,7 @@ sub _params_to_data_structure { ...@@ -1577,7 +1578,7 @@ sub _params_to_data_structure {
my ($self) = @_; my ($self) = @_;
# First we get the "special" charts, representing all the normal # First we get the "special" charts, representing all the normal
# field son the search page. This may modify _params, so it needs to # fields on the search page. This may modify _params, so it needs to
# happen first. # happen first.
my $clause = $self->_special_charts; my $clause = $self->_special_charts;
...@@ -1636,13 +1637,19 @@ sub _custom_search { ...@@ -1636,13 +1637,19 @@ sub _custom_search {
my @field_ids = $self->_field_ids; my @field_ids = $self->_field_ids;
return unless scalar @field_ids; return unless scalar @field_ids;
my $current_clause = new Bugzilla::Search::Clause($params->{j_top}); my $joiner = $params->{j_top} || '';
my $current_clause = $joiner eq 'AND_G'
? new Bugzilla::Search::ClauseGroup()
: new Bugzilla::Search::Clause($joiner);
my @clause_stack; my @clause_stack;
foreach my $id (@field_ids) { foreach my $id (@field_ids) {
my $field = $params->{"f$id"}; my $field = $params->{"f$id"};
if ($field eq 'OP') { if ($field eq 'OP') {
my $joiner = $params->{"j$id"}; my $joiner = $params->{"j$id"} || '';
my $new_clause = new Bugzilla::Search::Clause($joiner); my $new_clause = $joiner eq 'AND_G'
? new Bugzilla::Search::ClauseGroup()
: new Bugzilla::Search::Clause($joiner);
$new_clause->negate($params->{"n$id"}); $new_clause->negate($params->{"n$id"});
$current_clause->add($new_clause); $current_clause->add($new_clause);
push(@clause_stack, $current_clause); push(@clause_stack, $current_clause);
...@@ -1681,14 +1688,12 @@ sub _field_ids { ...@@ -1681,14 +1688,12 @@ sub _field_ids {
} }
sub _handle_chart { sub _handle_chart {
my ($self, $chart_id, $condition) = @_; my ($self, $chart_id, $clause, $condition) = @_;
my $dbh = Bugzilla->dbh; my $dbh = Bugzilla->dbh;
my $params = $self->_params; my $params = $self->_params;
my ($field, $operator, $value) = $condition->fov; my ($field, $operator, $value) = $condition->fov;
$field = FIELD_MAP->{$field} || $field;
return if (!defined $field or !defined $operator or !defined $value); return if (!defined $field or !defined $operator or !defined $value);
$field = FIELD_MAP->{$field} || $field;
my $string_value; my $string_value;
if (ref $value eq 'ARRAY') { if (ref $value eq 'ARRAY') {
...@@ -1727,7 +1732,11 @@ sub _handle_chart { ...@@ -1727,7 +1732,11 @@ sub _handle_chart {
value => $string_value, value => $string_value,
all_values => $value, all_values => $value,
joins => [], joins => [],
bugs_table => 'bugs',
table_suffix => '',
); );
$clause->update_search_args(\%search_args);
$search_args{quoted} = $self->_quote_unless_numeric(\%search_args); $search_args{quoted} = $self->_quote_unless_numeric(\%search_args);
# This should add a "term" selement to %search_args. # This should add a "term" selement to %search_args.
$self->do_search_function(\%search_args); $self->do_search_function(\%search_args);
...@@ -1744,6 +1753,11 @@ sub _handle_chart { ...@@ -1744,6 +1753,11 @@ sub _handle_chart {
value => $string_value, term => $search_args{term}, value => $string_value, term => $search_args{term},
}); });
foreach my $join (@{ $search_args{joins} }) {
$join->{bugs_table} = $search_args{bugs_table};
$join->{table_suffix} = $search_args{table_suffix};
}
$condition->translated(\%search_args); $condition->translated(\%search_args);
} }
...@@ -1899,8 +1913,14 @@ sub _quote_unless_numeric { ...@@ -1899,8 +1913,14 @@ sub _quote_unless_numeric {
} }
sub build_subselect { sub build_subselect {
my ($outer, $inner, $table, $cond) = @_; my ($outer, $inner, $table, $cond, $negate) = @_;
return "$outer IN (SELECT $inner FROM $table WHERE $cond)"; # Execute subselects immediately to avoid dependent subqueries, which are
# large performance hits on MySql
my $q = "SELECT DISTINCT $inner FROM $table WHERE $cond";
my $dbh = Bugzilla->dbh;
my $list = $dbh->selectcol_arrayref($q);
return $negate ? "1=1" : "1=2" unless @$list;
return $dbh->sql_in($outer, $list, $negate);
} }
# Used by anyexact to get the list of input values. This allows us to # Used by anyexact to get the list of input values. This allows us to
...@@ -2653,8 +2673,7 @@ sub _multiselect_term { ...@@ -2653,8 +2673,7 @@ sub _multiselect_term {
my $term = $args->{term}; my $term = $args->{term};
$term .= $args->{_extra_where} || ''; $term .= $args->{_extra_where} || '';
my $select = $args->{_select_field} || 'bug_id'; my $select = $args->{_select_field} || 'bug_id';
my $not_sql = $not ? "NOT " : ''; return build_subselect("$args->{bugs_table}.bug_id", $select, $table, $term, $not);
return "bugs.bug_id ${not_sql}IN (SELECT $select FROM $table WHERE $term)";
} }
############################### ###############################
......
...@@ -30,6 +30,11 @@ sub children { ...@@ -30,6 +30,11 @@ sub children {
return $self->{children}; return $self->{children};
} }
sub update_search_args {
my ($self, $search_args) = @_;
# abstract
}
sub joiner { return $_[0]->{joiner} } sub joiner { return $_[0]->{joiner} }
sub has_translated_conditions { sub has_translated_conditions {
...@@ -71,7 +76,7 @@ sub walk_conditions { ...@@ -71,7 +76,7 @@ sub walk_conditions {
my ($self, $callback) = @_; my ($self, $callback) = @_;
foreach my $child (@{ $self->children }) { foreach my $child (@{ $self->children }) {
if ($child->isa('Bugzilla::Search::Condition')) { if ($child->isa('Bugzilla::Search::Condition')) {
$callback->($child); $callback->($self, $child);
} }
else { else {
$child->walk_conditions($callback); $child->walk_conditions($callback);
......
# This Source Code Form is subject to the terms of the Mozilla Public
# License, v. 2.0. If a copy of the MPL was not distributed with this
# file, You can obtain one at http://mozilla.org/MPL/2.0/.
#
# This Source Code Form is "Incompatible With Secondary Licenses", as
# defined by the Mozilla Public License, v. 2.0.
package Bugzilla::Search::ClauseGroup;
use 5.10.1;
use strict;
use base qw(Bugzilla::Search::Clause);
use Bugzilla::Error;
use Bugzilla::Search::Condition qw(condition);
use Bugzilla::Util qw(trick_taint);
use List::MoreUtils qw(uniq);
use constant UNSUPPORTED_FIELDS => qw(
attach_data.thedata
classification
commenter
component
longdescs.count
product
owner_idle_time
);
sub new {
my ($class) = @_;
my $self = bless({ joiner => 'AND' }, $class);
# Add a join back to the bugs table which will be used to group conditions
# for this clause
my $condition = Bugzilla::Search::Condition->new({});
$condition->translated({
joins => [{
table => 'bugs',
as => 'bugs_g0',
from => 'bug_id',
to => 'bug_id',
extra => [],
}],
term => '1 = 1',
});
$self->SUPER::add($condition);
$self->{group_condition} = $condition;
return $self;
}
sub add {
my ($self, @args) = @_;
my $field = scalar(@args) == 3 ? $args[0] : $args[0]->{field};
# We don't support nesting of conditions under this clause
if (scalar(@args) == 1 && !$args[0]->isa('Bugzilla::Search::Condition')) {
ThrowUserError('search_grouped_invalid_nesting');
}
# Ensure all conditions use the same field
if (!$self->{_field}) {
$self->{_field} = $field;
} elsif ($field ne $self->{_field}) {
ThrowUserError('search_grouped_field_mismatch');
}
# Unsupported fields
if (grep { $_ eq $field } UNSUPPORTED_FIELDS ) {
ThrowUserError('search_grouped_field_invalid', { field => $field });
}
$self->SUPER::add(@args);
}
sub update_search_args {
my ($self, $search_args) = @_;
# No need to change things if there's only one child condition
return unless scalar(@{ $self->children }) > 1;
# we want all the terms to use the same join table
if (!exists $self->{_first_chart_id}) {
$self->{_first_chart_id} = $search_args->{chart_id};
} else {
$search_args->{chart_id} = $self->{_first_chart_id};
}
my $suffix = '_g' . $self->{_first_chart_id};
$self->{group_condition}->{translated}->{joins}->[0]->{as} = "bugs$suffix";
$search_args->{full_field} =~ s/^bugs\./bugs$suffix\./;
$search_args->{table_suffix} = $suffix;
$search_args->{bugs_table} = "bugs$suffix";
}
1;
...@@ -28,7 +28,7 @@ function custom_search_new_row() { ...@@ -28,7 +28,7 @@ function custom_search_new_row() {
var row = document.getElementById('custom_search_last_row'); var row = document.getElementById('custom_search_last_row');
var clone = row.cloneNode(true); var clone = row.cloneNode(true);
_cs_fix_ids(clone); _cs_fix_row_ids(clone);
// We only want one copy of the buttons, in the new row. So the old // We only want one copy of the buttons, in the new row. So the old
// ones get deleted. // ones get deleted.
...@@ -43,13 +43,28 @@ function custom_search_new_row() { ...@@ -43,13 +43,28 @@ function custom_search_new_row() {
// Always make sure there's only one row with this id. // Always make sure there's only one row with this id.
row.id = null; row.id = null;
row.parentNode.appendChild(clone); row.parentNode.appendChild(clone);
cs_reconfigure(row);
fix_query_string(row); fix_query_string(row);
return clone; return clone;
} }
var _cs_source_any_all;
function custom_search_open_paren() { function custom_search_open_paren() {
var row = document.getElementById('custom_search_last_row'); var row = document.getElementById('custom_search_last_row');
// create a copy of j_top and use that as the source, so we can modify
// j_top if required
if (!_cs_source_any_all) {
var j_top = document.getElementById('j_top');
_cs_source_any_all = j_top.cloneNode(true);
}
// find the parent any/all select, and remove the grouped option
var structure = _cs_build_structure(row);
var old_id = _cs_get_row_id(row);
var parent_j = document.getElementById(_cs_get_join(structure, 'f' + old_id));
_cs_remove_and_g(parent_j);
// If there's an "Any/All" select in this row, it needs to stay as // If there's an "Any/All" select in this row, it needs to stay as
// part of the parent paren set. // part of the parent paren set.
var old_any_all = _remove_any_all(row); var old_any_all = _remove_any_all(row);
...@@ -66,21 +81,20 @@ function custom_search_open_paren() { ...@@ -66,21 +81,20 @@ function custom_search_open_paren() {
var not_for_paren = new_not[0].cloneNode(true); var not_for_paren = new_not[0].cloneNode(true);
// Preserve the values when modifying the row. // Preserve the values when modifying the row.
var id = _cs_fix_ids(row, true); var id = _cs_fix_row_ids(row, true);
var prev_id = id - 1; var prev_id = id - 1;
var paren_row = row.cloneNode(false); var paren_row = row.cloneNode(false);
paren_row.id = null; paren_row.id = null;
paren_row.innerHTML = '(<input type="hidden" name="f' + prev_id paren_row.innerHTML = '(<input type="hidden" name="f' + prev_id
+ '" value="OP">'; + '" id="f' + prev_id + '" value="OP">';
paren_row.insertBefore(not_for_paren, paren_row.firstChild); paren_row.insertBefore(not_for_paren, paren_row.firstChild);
row.parentNode.insertBefore(paren_row, row); row.parentNode.insertBefore(paren_row, row);
// New paren set needs a new "Any/All" select. // New paren set needs a new "Any/All" select.
var any_all_container = document.createElement('div'); var any_all_container = document.createElement('div');
YAHOO.util.Dom.addClass(any_all_container, ANY_ALL_SELECT_CLASS); YAHOO.util.Dom.addClass(any_all_container, ANY_ALL_SELECT_CLASS);
var j_top = document.getElementById('j_top'); var any_all = _cs_source_any_all.cloneNode(true);
var any_all = j_top.cloneNode(true);
any_all.name = 'j' + prev_id; any_all.name = 'j' + prev_id;
any_all.id = any_all.name; any_all.id = any_all.name;
any_all_container.appendChild(any_all); any_all_container.appendChild(any_all);
...@@ -92,6 +106,7 @@ function custom_search_open_paren() { ...@@ -92,6 +106,7 @@ function custom_search_open_paren() {
YAHOO.util.Dom.setStyle(row, 'margin-left', new_margin + 'em'); YAHOO.util.Dom.setStyle(row, 'margin-left', new_margin + 'em');
YAHOO.util.Dom.removeClass('cp_container', 'bz_default_hidden'); YAHOO.util.Dom.removeClass('cp_container', 'bz_default_hidden');
cs_reconfigure(any_all_container);
fix_query_string(any_all_container); fix_query_string(any_all_container);
} }
...@@ -100,7 +115,7 @@ function custom_search_close_paren() { ...@@ -100,7 +115,7 @@ function custom_search_close_paren() {
// We need to up the new row's id by one more, because we're going // We need to up the new row's id by one more, because we're going
// to insert a "CP" before it. // to insert a "CP" before it.
var id = _cs_fix_ids(new_row); var id = _cs_fix_row_ids(new_row);
var margin = YAHOO.util.Dom.getStyle(new_row, 'margin-left'); var margin = YAHOO.util.Dom.getStyle(new_row, 'margin-left');
var int_match = margin.match(/\d+/); var int_match = margin.match(/\d+/);
...@@ -110,7 +125,7 @@ function custom_search_close_paren() { ...@@ -110,7 +125,7 @@ function custom_search_close_paren() {
var paren_row = new_row.cloneNode(false); var paren_row = new_row.cloneNode(false);
paren_row.id = null; paren_row.id = null;
paren_row.innerHTML = ')<input type="hidden" name="f' + (id - 1) paren_row.innerHTML = ')<input type="hidden" name="f' + (id - 1)
+ '" value="CP">'; + '" id="f' + (id - 1) + '" value="CP">';
new_row.parentNode.insertBefore(paren_row, new_row); new_row.parentNode.insertBefore(paren_row, new_row);
...@@ -118,6 +133,7 @@ function custom_search_close_paren() { ...@@ -118,6 +133,7 @@ function custom_search_close_paren() {
YAHOO.util.Dom.addClass('cp_container', 'bz_default_hidden'); YAHOO.util.Dom.addClass('cp_container', 'bz_default_hidden');
} }
cs_reconfigure(new_row);
fix_query_string(new_row); fix_query_string(new_row);
} }
...@@ -160,19 +176,17 @@ function redirect_html4_browsers() { ...@@ -160,19 +176,17 @@ function redirect_html4_browsers() {
document.location = url; document.location = url;
} }
function _cs_fix_ids(parent, preserve_values) { function _cs_fix_row_ids(row, preserve_values) {
// Update the label of the checkbox. // Update the label of the checkbox.
var label = YAHOO.util.Dom.getElementBy(function() { return true }, var label = YAHOO.util.Dom.getElementBy(function() { return true }, 'label', row);
'label', parent);
var id_match = label.htmlFor.match(/\d+$/); var id_match = label.htmlFor.match(/\d+$/);
var id = parseInt(id_match[0]) + 1; var id = parseInt(id_match[0]) + 1;
label.htmlFor = label.htmlFor.replace(/\d+$/, id); label.htmlFor = label.htmlFor.replace(/\d+$/, id);
// Sets all the inputs in the parent back to their default // Sets all the inputs in the row back to their default
// and fixes their id. // and fixes their id.
var fields = var fields =
YAHOO.util.Dom.getElementsByClassName('custom_search_form_field', null, YAHOO.util.Dom.getElementsByClassName('custom_search_form_field', null, row);
parent);
for (var i = 0; i < fields.length; i++) { for (var i = 0; i < fields.length; i++) {
var field = fields[i]; var field = fields[i];
...@@ -185,7 +199,7 @@ function _cs_fix_ids(parent, preserve_values) { ...@@ -185,7 +199,7 @@ function _cs_fix_ids(parent, preserve_values) {
} }
} }
// Update the numeric id for the new row. // Update the numeric id for the row.
field.name = field.name.replace(/\d+$/, id); field.name = field.name.replace(/\d+$/, id);
field.id = field.name; field.id = field.name;
} }
...@@ -193,6 +207,132 @@ function _cs_fix_ids(parent, preserve_values) { ...@@ -193,6 +207,132 @@ function _cs_fix_ids(parent, preserve_values) {
return id; return id;
} }
function _cs_build_structure(form_member) {
// build a map of the structure of the custom fields
var form = YAHOO.util.Dom.getAncestorByTagName(form_member, 'form');
var last_id = _get_last_cs_row_id(form);
var structure = [ 'j_top' ];
var nested = [ structure ];
for (var id = 1; id <= last_id; id++) {
var f = form['f' + id];
if (!f || !f.parentNode.parentNode) continue;
if (f.value == 'OP') {
var j = [ 'j' + id ];
nested[nested.length - 1].push(j);
nested.push(j);
continue;
} else if (f.value == 'CP') {
nested.pop();
continue;
} else {
nested[nested.length - 1].push('f' + id);
}
}
return structure;
}
function cs_reconfigure(form_member) {
var structure = _cs_build_structure(form_member);
_cs_add_listeners(structure);
_cs_trigger_j_listeners(structure);
fix_query_string(form_member);
var j = _cs_get_join(structure, 'f' + _get_last_cs_row_id());
document.getElementById('op_button').disabled = document.getElementById(j).value == 'AND_G';
}
function _cs_add_listeners(parents) {
for (var i = 0, l = parents.length; i < l; i++) {
if (typeof(parents[i]) == 'object') {
// nested
_cs_add_listeners(parents[i]);
} else if (i == 0) {
// joiner
YAHOO.util.Event.removeListener(parents[i], 'change', _cs_j_change);
YAHOO.util.Event.addListener(parents[i], 'change', _cs_j_change, parents);
} else {
// field
YAHOO.util.Event.removeListener(parents[i], 'change', _cs_f_change);
YAHOO.util.Event.addListener(parents[i], 'change', _cs_f_change, parents);
}
}
}
function _cs_trigger_j_listeners(fields) {
var has_children = false;
for (var i = 0, l = fields.length; i < l; i++) {
if (typeof(fields[i]) == 'undefined') {
continue;
} else if (typeof(fields[i]) == 'object') {
// nested
_cs_trigger_j_listeners(fields[i]);
has_children = true;
} else if (i == 0) {
_cs_j_change(undefined, fields);
}
}
if (has_children) {
_cs_remove_and_g(document.getElementById(fields[0]));
}
}
function _cs_get_join(parents, field) {
for (var i = 0, l = parents.length; i < l; i++) {
if (typeof(parents[i]) == 'object') {
// nested
var result = _cs_get_join(parents[i], field);
if (result) return result;
} else if (parents[i] == field) {
return parents[0];
}
}
return false;
}
function _cs_remove_and_g(join_field) {
var index = bz_optionIndex(join_field, 'AND_G');
join_field.options[index] = null;
join_field.options[bz_optionIndex(join_field, 'AND')].innerHTML = cs_and_label;
join_field.options[bz_optionIndex(join_field, 'OR')].innerHTML = cs_or_label;
}
function _cs_j_change(evt, fields, field) {
var j = document.getElementById(fields[0]);
if (j && j.value == 'AND_G') {
for (var i = 1, l = fields.length; i < l; i++) {
if (typeof(fields[i]) == 'object') continue;
if (!field) {
field = document.getElementById(fields[i]).value;
} else {
document.getElementById(fields[i]).value = field;
}
}
if (evt) {
fix_query_string(j);
}
if ('f' + _get_last_cs_row_id() == fields[fields.length - 1]) {
document.getElementById('op_button').style.display = 'none';
}
} else {
document.getElementById('op_button').style.display = '';
}
}
function _cs_f_change(evt, args) {
var field = YAHOO.util.Event.getTarget(evt);
_cs_j_change(evt, args, field.value);
}
function _get_last_cs_row_id() {
return _cs_get_row_id('custom_search_last_row');
}
function _cs_get_row_id(row) {
var label = YAHOO.util.Dom.getElementBy(function() { return true }, 'label', row);
return parseInt(label.htmlFor.match(/\d+$/)[0]);
}
function _remove_any_all(parent) { function _remove_any_all(parent) {
var any_all = YAHOO.util.Dom.getElementsByClassName( var any_all = YAHOO.util.Dom.getElementsByClassName(
ANY_ALL_SELECT_CLASS, null, parent); ANY_ALL_SELECT_CLASS, null, parent);
......
...@@ -1597,6 +1597,17 @@ ...@@ -1597,6 +1597,17 @@
and the "matches" search can only be used with the "content" and the "matches" search can only be used with the "content"
field. field.
[% ELSIF error == "search_grouped_field_invalid" %]
[% terms.Bugzilla %] does not support using the
"[%+ field_descs.$field FILTER html %]" ([% field FILTER html %])
field with grouped search conditions.
[% ELSIF error == "search_grouped_invalid_nesting" %]
You cannot nest clauses within grouped search conditions.
[% ELSIF error == "search_grouped_field_mismatch" %]
All conditions under a groups search must use the same field.
[% ELSIF error == "search_field_operator_invalid" %] [% ELSIF error == "search_field_operator_invalid" %]
[% terms.Bugzilla %] does not support using the [% terms.Bugzilla %] does not support using the
"[%+ field_descs.$field FILTER html %]" ([% field FILTER html %]) "[%+ field_descs.$field FILTER html %]" ([% field FILTER html %])
......
...@@ -64,6 +64,10 @@ ...@@ -64,6 +64,10 @@
<script type="text/javascript" src="[% 'js/history.js/native.history.js' FILTER mtime %]"></script> <script type="text/javascript" src="[% 'js/history.js/native.history.js' FILTER mtime %]"></script>
<script type="text/javascript"> <script type="text/javascript">
redirect_html4_browsers(); redirect_html4_browsers();
[%# These are alternative labels for the AND and OR options in and_all_select %]
var cs_and_label = 'Match ALL of the following:';
var cs_or_label = 'Match ANY of the following:';
cs_reconfigure('custom_search_last_row');
</script> </script>
</div> </div>
...@@ -120,7 +124,8 @@ ...@@ -120,7 +124,8 @@
( (
[% indent_level = indent_level + 1 %] [% indent_level = indent_level + 1 %]
[% ELSIF condition.f == "CP" %] [% ELSIF condition.f == "CP" %]
<input type="hidden" name="f[% cond_num FILTER html %]" value="CP"> <input type="hidden" name="f[% cond_num FILTER html %]"
id="f[% cond_num FILTER html %]" value="CP">
) )
[% ELSE %] [% ELSE %]
<select name="f[% cond_num FILTER html %]" title="Field" <select name="f[% cond_num FILTER html %]" title="Field"
...@@ -164,9 +169,11 @@ ...@@ -164,9 +169,11 @@
<div class="any_all_select"> <div class="any_all_select">
<select name="[% name FILTER html %]" id="[% name FILTER html %]" <select name="[% name FILTER html %]" id="[% name FILTER html %]"
onchange="fix_query_string(this)"> onchange="fix_query_string(this)">
<option value="AND">Match ALL of the following:</option> <option value="AND">Match ALL of the following separately:</option>
<option value="OR" [% ' selected="selected"' IF selected == "OR" %]> <option value="OR" [% ' selected="selected"' IF selected == "OR" %]>
Match ANY of the following:</option> Match ANY of the following separately:</option>
<option value="AND_G" [% ' selected' IF selected == "AND_G" %]>
Match ALL of the following against the same field:</option>
</select> </select>
[% IF with_advanced_link %] [% IF with_advanced_link %]
<a id="custom_search_advanced_controller" <a id="custom_search_advanced_controller"
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment