3
0
Fork 0

Adapt to v9 API.

- Use the new domain widget.
- Use modules system.
12.0
Jairo Llopis 2017-02-16 11:36:04 +01:00 committed by Pedro M. Baeza
parent 7b5740727c
commit ece4af9bb4
6 changed files with 163 additions and 192 deletions

View File

@ -17,6 +17,7 @@ To use this module, you need to:
* select a one2many, many2many or many2one field * select a one2many, many2many or many2one field
* select operator `is equal to` or `is not equal to` * select operator `is equal to` or `is not equal to`
* the textfield changes to a many2one selection field where you can search for the record in question * the textfield changes to a many2one selection field where you can search for the record in question
* click *Apply*
To search for properties of linked records (ie invoices for customers with a credit limit higher than X): To search for properties of linked records (ie invoices for customers with a credit limit higher than X):
@ -24,11 +25,8 @@ To search for properties of linked records (ie invoices for customers with a cre
* select a one2many, many2many or many2one field * select a one2many, many2many or many2one field
* select operator `is in selection` * select operator `is in selection`
* in the search view that pops up, select the criteria * in the search view that pops up, select the criteria
* click `Use criteria` * select the records you want, or select the top corner box to select all matching records with that criteria
* if you're only interested in certain records, mark them en click `Select` * click *Select*
* if you want to change your selection afterwards, click the search symbol right of the selection term
In both cases, don't forget to click `Apply` to actually execute the search.
Note that you can stack searching for properties: Simply add another advanced search in the selection search window. You can do this indefinetely, so it is possible to search for moves belonging to a journal which has a user who is member of a certain group etc. Note that you can stack searching for properties: Simply add another advanced search in the selection search window. You can do this indefinetely, so it is possible to search for moves belonging to a journal which has a user who is member of a certain group etc.
@ -36,6 +34,13 @@ Note that you can stack searching for properties: Simply add another advanced se
:alt: Try me on Runbot :alt: Try me on Runbot
:target: https://runbot.odoo-community.org/runbot/154/9.0 :target: https://runbot.odoo-community.org/runbot/154/9.0
Known issues / Roadmap
======================
* When you use *is [not] equal to* search system and click on an option with
the mouse, the search dropdown gets hidden and you have to reopen it and
reapply the filter. As a workaround, use the keyboard to select the option
you want.
Credits Credits
======= =======
@ -45,6 +50,7 @@ Contributors
* Holger Brunn <hbrunn@therp.nl> * Holger Brunn <hbrunn@therp.nl>
* Vicent Cubells <vicent.cubells@tecnativa.com> * Vicent Cubells <vicent.cubells@tecnativa.com>
* Jairo Llopis <jairo.llopis@tecnativa.com>
Maintainer Maintainer
---------- ----------

View File

@ -7,6 +7,7 @@
"name": "Search x2x fields", "name": "Search x2x fields",
"version": "9.0.1.0.0", "version": "9.0.1.0.0",
"author": "Therp BV, " "author": "Therp BV, "
"Tecnativa, "
"Odoo Community Association (OCA)", "Odoo Community Association (OCA)",
"license": "AGPL-3", "license": "AGPL-3",
"category": "Usability", "category": "Usability",

View File

@ -1,36 +0,0 @@
.openerp .searchview_extended_prop_value .oe_form_field_with_button
{
position: relative;
}
.openerp .oe_searchview_drawer .web_advanced_search_x2x_domain
{
max-width: 20em;
overflow: hidden;
text-overflow: ellipsis;
display: inline-block;
}
/* copy search view's button style */
.openerp .oe_searchview_drawer .web_advanced_search_x2x_search:before
{
font: 21px "mnmliconsRegular";
content: "r";
color: #a3a3a3;
margin-left: 5px;
}
.openerp .oe_searchview_drawer .web_advanced_search_x2x_search
{
font-size: 1px;
letter-spacing: -1px;
color: transparent;
text-shadow: none;
font-weight: normal;
-moz-box-shadow: none;
-webkit-box-shadow: none;
box-shadow: none;
-moz-border-radius: 0;
-webkit-border-radius: 0;
border-radius: 0;
padding: 0;
border: none;
background: transparent;
}

View File

@ -1,41 +1,23 @@
//-*- coding: utf-8 -*- /* Copyright 2015 Therp BV <http://therp.nl>
//############################################################################ * Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
// * License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). */
// OpenERP, Open Source Management Solution
// This module copyright (C) 2015 Therp BV <http://therp.nl>.
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
//############################################################################
odoo.define('web_advanced_search_x2x.search_filters', function (require) { odoo.define('web_advanced_search_x2x.search_filters', function (require) {
"use strict"; "use strict";
var filters = require('web.search_filters'); require('web.form_relational');
require('web.form_widgets');
var search_filters = require('web.search_filters');
var form_common = require('web.form_common'); var form_common = require('web.form_common');
var SearchView = require('web.SearchView'); var SearchView = require('web.SearchView');
var data = require('web.data'); var data = require('web.data');
var session = require('web.session');
var core = require('web.core'); var core = require('web.core');
var searchfilters = filters.ExtendedSearchProposition.Char.extend( var X2XAdvancedSearchPropositionMixin = {
form_common.FieldManagerMixin, template: "web_advanced_search_x2x.proposition",
{
template: 'web_advanced_search_x2x.extended_search.proposition.many2one',
searchfield: null,
init: function() init: function()
{ {
// Make equal and not equal appear 1st and 2nd
this.operators = _.sortBy( this.operators = _.sortBy(
this.operators, this.operators,
function(op) function(op)
@ -50,99 +32,120 @@ odoo.define('web_advanced_search_x2x.search_filters', function (require) {
return 0; return 0;
} }
}); });
// Append domain operator
this.operators.push({ this.operators.push({
'value': 'domain', 'text': data._lt('is in selection'), 'value': 'domain', 'text': core._lt('is in selection'),
}); });
return this._super.apply(this, arguments); return this._super.apply(this, arguments);
}, },
start: function()
{
this.getParent().$('.searchview_extended_prop_op')
.on('change', this.proxy('operator_changed'));
return this._super.apply(this, arguments).then(
this.proxy(this.operator_changed));
},
get_field_desc: function() get_field_desc: function()
{ {
return this.field; return this.field;
}, },
create_searchfield_node: function() /**
{ * Add the right relational field to the template.
*/
renderElement: function () {
try {
this._x2x_field.destroy();
} catch (error) {}
this.relational = this.x2x_widget_name();
this._super.apply(this, arguments);
if (this.relational) {
this.x2x_field().appendTo(this.$el);
}
delete this.relational;
},
/**
* Create a relational field for the user.
*
* @return {Field}
*/
x2x_field: function () {
if (this._x2x_field) {
this._x2x_field.destroy();
delete this._x2x_field;
}
var widget = this.x2x_widget();
if (!widget) return;
this._x2x_field = new widget(
this,
this.x2x_field_create_options()
);
this._x2x_field.on(
"change:value",
this,
this.proxy("x2x_value_changed")
);
return this._x2x_field;
},
x2x_field_create_options: function () {
return { return {
attrs: { attrs: {
name: this.field.name, name: this.field.name,
options: '{"no_create": true}', options: JSON.stringify({
no_create: true,
model: this.field.relation,
}),
}, },
};
},
x2x_value_changed: function () {
switch (this.x2x_widget_name()) {
case "char_domain":
// Apply domain when selected
this.getParent().getParent().commit_search();
break;
} }
}, },
create_searchfield: function() x2x_widget: function () {
{ var name = this.x2x_widget_name();
if(this.searchfield) return name && core.form_widget_registry.get(name);
{
this.searchfield.destroy();
}
this.searchfield = new form_common.FieldMany2One(
this, this.create_searchfield_node());
return this.searchfield;
}, },
operator_changed: function(e) x2x_widget_name: function () {
{ switch (this.get_operator()) {
if(this.searchfield) case "=":
{ case "!=":
this.searchfield.destroy(); return "many2one";
} case "domain":
this.renderElement(); return "char_domain";
if(this.show_searchfield())
{
this.create_searchfield().appendTo(this.$el.empty());
}
if(this.show_domain_selection())
{
this.$el.filter('input').remove();
this.$el.filter('button.web_advanced_search_x2x_search').click(
this.proxy(this.popup_domain_selection));
this.popup_domain_selection();
} }
}, },
get_operator: function() get_domain: function () {
{ // Special way to get domain if user chose "domain" filter
if(this.isDestroyed()) if (this.get_operator() == "domain") {
{ var value = this._x2x_field.get_value();
return false; var domain = new data.CompoundDomain(),
name = this.field.name;
$.map(value, function (el) {
domain.add([[
_.str.sprintf("%s.%s", name, el[0]),
el[1],
el[2],
]]);
});
return domain;
} else {
return this._super.apply(this, arguments);
} }
return this.getParent().$('.searchview_extended_prop_op').val();
}, },
show_searchfield: function() get_operator: function () {
{ return !this.isDestroyed() &&
var operator = this.get_operator() this.getParent().$('.searchview_extended_prop_op').val();
return operator == '=' || operator == '!=';
}, },
show_domain_selection: function() get_value: function () {
{ try {
return this.get_operator() == 'domain'; return this._x2x_field.get_value();
}, } catch (error) {
get_value: function() return this._super.apply(this, arguments);
{
if(this.show_searchfield() && this.searchfield)
{
return this.searchfield.get_value();
} }
return this._super.apply(this, arguments);
}, },
format_label: function(format, field, operator) format_label: function (format, field, operator) {
{ if (this.x2x_widget()) {
var value = null; var value = String(this._x2x_field.get_value());
if(this.show_searchfield() && this.searchfield) if (this._x2x_field.display_value) {
{ value = this._x2x_field.display_value[value];
value = this.searchfield.display_value[ }
String(this.searchfield.get_value())];
}
if(this.show_domain_selection() && this.domain_representation)
{
value = this.domain_representation;
}
if(value)
{
return _.str.sprintf( return _.str.sprintf(
format, format,
{ {
@ -151,57 +154,46 @@ odoo.define('web_advanced_search_x2x.search_filters', function (require) {
value: value, value: value,
} }
); );
} else {
return this._super.apply(this, arguments);
}
},
};
var ExtendedSearchProposition = search_filters.ExtendedSearchProposition,
Char = ExtendedSearchProposition.Char,
affected_types = ["one2many", "many2one", "many2many"],
X2XAdvancedSearchProposition = Char.extend(
form_common.FieldManagerMixin,
X2XAdvancedSearchPropositionMixin
);
ExtendedSearchProposition.include({
/**
* Force re-rendering the value widget if needed.
*/
operator_changed: function (event) {
if (this.value instanceof X2XAdvancedSearchProposition) {
this.value_rerender();
} }
return this._super.apply(this, arguments); return this._super.apply(this, arguments);
}, },
get_domain: function() /**
{ * Re-render proposition's value widget.
if(this.show_domain_selection()) *
{ * @return {jQuery.Deferred}
var self = this; */
if(!this.domain || this.domain.length == 0) value_rerender: function () {
{ return this.value.appendTo(
throw new filters.Invalid( this.$(".searchview_extended_prop_value").show().empty()
this.field.string, this.domain_representation, );
data._lt('invalid search domain'));
}
return _.extend(new data.CompoundDomain(), {
__domains: [
_.map(this.domain, function(leaf)
{
if(_.isArray(leaf) && leaf.length == 3)
{
return [
self.field.name + '.' + leaf[0],
leaf[1],
leaf[2]
]
}
return leaf;
}),
],
})
}
return this._super.apply(this, arguments);
},
popup_domain_selection: function()
{
var self = this,
popup = new form_common.SelectCreatePopup(this);
popup.on('domain_selected', this, function(domain, domain_representation)
{
self.$el.filter('.web_advanced_search_x2x_domain').text(
domain_representation);
self.domain = domain;
self.domain_representation = domain_representation;
});
popup.select_element(
this.field.relation, {}, this.field.domain,
new data.CompoundContext(
session.user_context, this.field.context));
}, },
}); });
// Register this search proposition for relational fields
$.each(affected_types, function (index, value) {
core.search_filters_registry.add(value, X2XAdvancedSearchProposition);
});
SearchView.include({ SearchView.include({
build_search_data: function() build_search_data: function()
@ -245,5 +237,10 @@ odoo.define('web_advanced_search_x2x.search_filters', function (require) {
}); });
return result; return result;
}, },
}) });
return {
X2XAdvancedSearchPropositionMixin: X2XAdvancedSearchPropositionMixin,
X2XAdvancedSearchProposition: X2XAdvancedSearchProposition,
};
}); });

View File

@ -1,12 +1,14 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
<templates> <templates>
<t t-name="web_advanced_search_x2x.extended_search.proposition.many2one"> <t t-name="web_advanced_search_x2x.proposition">
<t t-if="!widget.show_searchfield()"> <t t-if="widget.relational">
<!-- This wrapper fixes CSS styiling -->
<div class="oe_form"/>
</t>
<t t-if="!widget.relational">
<t t-call="SearchView.extended_search.proposition.char" /> <t t-call="SearchView.extended_search.proposition.char" />
</t> </t>
<t t-if="widget.show_domain_selection()">
<span class="web_advanced_search_x2x_domain" /><button class="web_advanced_search_x2x_search" type="button">Search</button>
</t>
<span />
</t> </t>
</templates> </templates>

View File

@ -1,10 +1,11 @@
<?xml version="1.0" encoding="UTF-8"?> <?xml version="1.0" encoding="UTF-8"?>
<!-- Copyright 2017 Jairo Llopis <jairo.llopis@tecnativa.com>
License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl). -->
<odoo> <odoo>
<template id="assets_backend" name="web_advanced_search_x2x assets" inherit_id="web.assets_backend"> <template id="assets_backend" name="web_advanced_search_x2x assets" inherit_id="web.assets_backend">
<xpath expr="." position="inside"> <xpath expr="." position="inside">
<script type="text/javascript" src="/web_advanced_search_x2x/static/src/js/web_advanced_search_x2x.js"></script> <script type="text/javascript" src="/web_advanced_search_x2x/static/src/js/web_advanced_search_x2x.js"></script>
<link rel="stylesheet" href="/web_advanced_search_x2x/static/src/css/web_advanced_search_x2x.css"/>
</xpath> </xpath>
</template> </template>