2006-12-07 13:41:40 +00:00
|
|
|
##############################################################################
|
|
|
|
#
|
2008-06-16 11:00:21 +00:00
|
|
|
# Copyright (c) 2004-2008 TINY SPRL. (http://tiny.be) All Rights Reserved.
|
2006-12-07 13:41:40 +00:00
|
|
|
#
|
2008-06-16 06:44:24 +00:00
|
|
|
# $Id$
|
2006-12-07 13:41:40 +00:00
|
|
|
#
|
|
|
|
# WARNING: This program as such is intended to be used by professional
|
|
|
|
# programmers who take the whole responsability of assessing all potential
|
|
|
|
# consequences resulting from its eventual inadequacies and bugs
|
|
|
|
# End users who are looking for a ready-to-use solution with commercial
|
|
|
|
# garantees and support are strongly adviced to contract a Free Software
|
|
|
|
# Service Company
|
|
|
|
#
|
|
|
|
# This program is Free Software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU General Public License
|
|
|
|
# as published by the Free Software Foundation; either version 2
|
|
|
|
# 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 General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU General Public License
|
|
|
|
# along with this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
|
|
|
|
#
|
|
|
|
##############################################################################
|
|
|
|
|
|
|
|
from osv import fields, osv
|
|
|
|
from osv.orm import browse_null, browse_record
|
2008-07-11 11:42:24 +00:00
|
|
|
import re
|
2006-12-07 13:41:40 +00:00
|
|
|
|
|
|
|
def one_in(setA, setB):
|
2008-07-22 14:24:36 +00:00
|
|
|
"""Check the presence of an element of setA in setB
|
|
|
|
"""
|
|
|
|
for x in setA:
|
|
|
|
if x in setB:
|
|
|
|
return True
|
|
|
|
return False
|
2006-12-07 13:41:40 +00:00
|
|
|
|
|
|
|
icons = map(lambda x: (x,x), ['STOCK_ABOUT', 'STOCK_ADD', 'STOCK_APPLY', 'STOCK_BOLD',
|
|
|
|
'STOCK_CANCEL', 'STOCK_CDROM', 'STOCK_CLEAR', 'STOCK_CLOSE', 'STOCK_COLOR_PICKER',
|
|
|
|
'STOCK_CONNECT', 'STOCK_CONVERT', 'STOCK_COPY', 'STOCK_CUT', 'STOCK_DELETE',
|
|
|
|
'STOCK_DIALOG_AUTHENTICATION', 'STOCK_DIALOG_ERROR', 'STOCK_DIALOG_INFO',
|
|
|
|
'STOCK_DIALOG_QUESTION', 'STOCK_DIALOG_WARNING', 'STOCK_DIRECTORY', 'STOCK_DISCONNECT',
|
|
|
|
'STOCK_DND', 'STOCK_DND_MULTIPLE', 'STOCK_EDIT', 'STOCK_EXECUTE', 'STOCK_FILE',
|
|
|
|
'STOCK_FIND', 'STOCK_FIND_AND_REPLACE', 'STOCK_FLOPPY', 'STOCK_GOTO_BOTTOM',
|
|
|
|
'STOCK_GOTO_FIRST', 'STOCK_GOTO_LAST', 'STOCK_GOTO_TOP', 'STOCK_GO_BACK',
|
|
|
|
'STOCK_GO_DOWN', 'STOCK_GO_FORWARD', 'STOCK_GO_UP', 'STOCK_HARDDISK',
|
|
|
|
'STOCK_HELP', 'STOCK_HOME', 'STOCK_INDENT', 'STOCK_INDEX', 'STOCK_ITALIC',
|
|
|
|
'STOCK_JUMP_TO', 'STOCK_JUSTIFY_CENTER', 'STOCK_JUSTIFY_FILL',
|
|
|
|
'STOCK_JUSTIFY_LEFT', 'STOCK_JUSTIFY_RIGHT', 'STOCK_MEDIA_FORWARD',
|
|
|
|
'STOCK_MEDIA_NEXT', 'STOCK_MEDIA_PAUSE', 'STOCK_MEDIA_PLAY',
|
|
|
|
'STOCK_MEDIA_PREVIOUS', 'STOCK_MEDIA_RECORD', 'STOCK_MEDIA_REWIND',
|
|
|
|
'STOCK_MEDIA_STOP', 'STOCK_MISSING_IMAGE', 'STOCK_NETWORK', 'STOCK_NEW',
|
|
|
|
'STOCK_NO', 'STOCK_OK', 'STOCK_OPEN', 'STOCK_PASTE', 'STOCK_PREFERENCES',
|
|
|
|
'STOCK_PRINT', 'STOCK_PRINT_PREVIEW', 'STOCK_PROPERTIES', 'STOCK_QUIT',
|
|
|
|
'STOCK_REDO', 'STOCK_REFRESH', 'STOCK_REMOVE', 'STOCK_REVERT_TO_SAVED',
|
|
|
|
'STOCK_SAVE', 'STOCK_SAVE_AS', 'STOCK_SELECT_COLOR', 'STOCK_SELECT_FONT',
|
|
|
|
'STOCK_SORT_ASCENDING', 'STOCK_SORT_DESCENDING', 'STOCK_SPELL_CHECK',
|
|
|
|
'STOCK_STOP', 'STOCK_STRIKETHROUGH', 'STOCK_UNDELETE', 'STOCK_UNDERLINE',
|
|
|
|
'STOCK_UNDO', 'STOCK_UNINDENT', 'STOCK_YES', 'STOCK_ZOOM_100',
|
|
|
|
'STOCK_ZOOM_FIT', 'STOCK_ZOOM_IN', 'STOCK_ZOOM_OUT',
|
2007-08-09 11:20:56 +00:00
|
|
|
'terp-account', 'terp-crm', 'terp-mrp', 'terp-product', 'terp-purchase',
|
|
|
|
'terp-sale', 'terp-tools', 'terp-administration', 'terp-hr', 'terp-partner',
|
|
|
|
'terp-project', 'terp-report', 'terp-stock', 'terp-calendar', 'terp-graph',
|
2006-12-07 13:41:40 +00:00
|
|
|
])
|
|
|
|
|
2007-08-09 06:06:27 +00:00
|
|
|
class many2many_unique(fields.many2many):
|
2008-07-22 14:24:36 +00:00
|
|
|
def set(self, cr, obj, id, name, values, user=None, context=None):
|
|
|
|
if not values:
|
|
|
|
return
|
|
|
|
val = values[:]
|
|
|
|
for act in values:
|
|
|
|
if act[0]==4:
|
|
|
|
cr.execute('SELECT * FROM '+self._rel+' \
|
|
|
|
WHERE '+self._id1+'=%d AND '+self._id2+'=%d', (id, act[1]))
|
|
|
|
if cr.fetchall():
|
|
|
|
val.remove(act)
|
|
|
|
return super(many2many_unique, self).set(cr, obj, id, name, val, user=user,
|
|
|
|
context=context)
|
2007-08-09 06:06:27 +00:00
|
|
|
|
2006-12-07 13:41:40 +00:00
|
|
|
|
|
|
|
class ir_ui_menu(osv.osv):
|
2008-07-22 14:24:36 +00:00
|
|
|
_name = 'ir.ui.menu'
|
|
|
|
def search(self, cr, uid, args, offset=0, limit=2000, order=None,
|
|
|
|
context=None, count=False):
|
|
|
|
if context is None:
|
|
|
|
context = {}
|
|
|
|
ids = osv.orm.orm.search(self, cr, uid, args, offset, limit, order,
|
|
|
|
context=context)
|
|
|
|
if uid==1:
|
|
|
|
return ids
|
|
|
|
user_groups = self.pool.get('res.users').read(cr, uid, [uid])[0]['groups_id']
|
|
|
|
result = []
|
|
|
|
for menu in self.browse(cr, uid, ids):
|
|
|
|
if not len(menu.groups_id):
|
|
|
|
result.append(menu.id)
|
|
|
|
continue
|
|
|
|
for g in menu.groups_id:
|
|
|
|
if g.id in user_groups:
|
|
|
|
result.append(menu.id)
|
|
|
|
break
|
|
|
|
return result
|
2006-12-07 13:41:40 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
def _get_full_name(self, cr, uid, ids, name, args, context):
|
|
|
|
res = {}
|
|
|
|
for m in self.browse(cr, uid, ids, context=context):
|
|
|
|
res[m.id] = self._get_one_full_name(m)
|
|
|
|
return res
|
2006-12-07 13:41:40 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
def _get_one_full_name(self, menu, level=6):
|
|
|
|
if level<=0:
|
|
|
|
return '...'
|
|
|
|
if menu.parent_id:
|
|
|
|
parent_path = self._get_one_full_name(menu.parent_id, level-1) + "/"
|
|
|
|
else:
|
|
|
|
parent_path = ''
|
|
|
|
return parent_path + menu.name
|
2006-12-07 13:41:40 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
def copy(self, cr, uid, id, default=None, context=None):
|
|
|
|
ir_values_obj = self.pool.get('ir.values')
|
|
|
|
res = super(ir_ui_menu, self).copy(cr, uid, id, context=context)
|
|
|
|
datas=self.read(cr,uid,[res],['name'])[0]
|
|
|
|
rex=re.compile('\([0-9]+\)')
|
|
|
|
concat=rex.findall(datas['name'])
|
|
|
|
if concat:
|
|
|
|
next_num=eval(concat[0])+1
|
|
|
|
datas['name']=rex.sub(('(%d)'%next_num),datas['name'])
|
|
|
|
else:
|
|
|
|
datas['name']=datas['name']+'(1)'
|
|
|
|
self.write(cr,uid,[res],{'name':datas['name']})
|
|
|
|
ids = ir_values_obj.search(cr, uid, [
|
|
|
|
('model', '=', 'ir.ui.menu'),
|
|
|
|
('res_id', '=', id),
|
|
|
|
])
|
|
|
|
for iv in ir_values_obj.browse(cr, uid, ids):
|
|
|
|
new_id = ir_values_obj.copy(cr, uid, iv.id,
|
|
|
|
default={'res_id': res}, context=context)
|
|
|
|
return res
|
2006-12-07 13:41:40 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
def _action(self, cursor, user, ids, name, arg, context=None):
|
|
|
|
res = {}
|
|
|
|
values_obj = self.pool.get('ir.values')
|
|
|
|
value_ids = values_obj.search(cursor, user, [
|
|
|
|
('model', '=', self._name), ('key', '=', 'action'),
|
|
|
|
('key2', '=', 'tree_but_open'), ('res_id', 'in', ids)],
|
|
|
|
context=context)
|
|
|
|
values_action = {}
|
|
|
|
for value in values_obj.browse(cursor, user, value_ids,
|
|
|
|
context=context):
|
|
|
|
values_action[value.res_id] = value.value
|
|
|
|
for menu_id in ids:
|
|
|
|
res[menu_id] = values_action.get(menu_id, False)
|
|
|
|
return res
|
2007-10-19 12:14:58 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
def _action_inv(self, cursor, user, menu_id, name, value, arg, context=None):
|
|
|
|
if context is None:
|
|
|
|
context = {}
|
|
|
|
ctx = context.copy()
|
|
|
|
if 'read_delta' in ctx:
|
|
|
|
del ctx['read_delta']
|
|
|
|
values_obj = self.pool.get('ir.values')
|
|
|
|
values_ids = values_obj.search(cursor, user, [
|
|
|
|
('model', '=', self._name), ('key', '=', 'action'),
|
|
|
|
('key2', '=', 'tree_but_open'), ('res_id', '=', menu_id)],
|
|
|
|
context=context)
|
|
|
|
if values_ids:
|
|
|
|
values_obj.write(cursor, user, values_ids[0], {'value': value},
|
|
|
|
context=ctx)
|
|
|
|
else:
|
|
|
|
values_obj.create(cursor, user, {
|
|
|
|
'name': 'Menuitem',
|
|
|
|
'model': self._name,
|
|
|
|
'value': value,
|
|
|
|
'object': True,
|
|
|
|
'key': 'action',
|
|
|
|
'key2': 'tree_but_open',
|
|
|
|
'res_id': menu_id,
|
|
|
|
}, context=ctx)
|
2007-10-19 12:14:58 +00:00
|
|
|
|
2008-07-22 14:24:36 +00:00
|
|
|
_columns = {
|
|
|
|
'name': fields.char('Menu', size=64, required=True, translate=True),
|
|
|
|
'sequence': fields.integer('Sequence'),
|
|
|
|
'child_id' : fields.one2many('ir.ui.menu', 'parent_id','Child ids'),
|
|
|
|
'parent_id': fields.many2one('ir.ui.menu', 'Parent Menu', select=True),
|
|
|
|
'groups_id': many2many_unique('res.groups', 'ir_ui_menu_group_rel',
|
|
|
|
'menu_id', 'gid', 'Groups'),
|
|
|
|
'complete_name': fields.function(_get_full_name, method=True,
|
|
|
|
string='Complete Name', type='char', size=128),
|
|
|
|
'icon': fields.selection(icons, 'Icon', size=64),
|
|
|
|
'action': fields.function(_action, fnct_inv=_action_inv,
|
|
|
|
method=True, type='reference', string='Action',
|
|
|
|
selection=[
|
|
|
|
('ir.actions.report.custom', 'ir.actions.report.custom'),
|
|
|
|
('ir.actions.report.xml', 'ir.actions.report.xml'),
|
|
|
|
('ir.actions.act_window', 'ir.actions.act_window'),
|
|
|
|
('ir.actions.wizard', 'ir.actions.wizard'),
|
2008-06-06 13:13:42 +00:00
|
|
|
('ir.actions.url', 'ir.actions.act_url'),
|
2008-07-22 14:24:36 +00:00
|
|
|
]),
|
|
|
|
}
|
|
|
|
_defaults = {
|
|
|
|
'icon' : lambda *a: 'STOCK_OPEN',
|
|
|
|
'sequence' : lambda *a: 10
|
|
|
|
}
|
|
|
|
_order = "sequence,id"
|
2006-12-07 13:41:40 +00:00
|
|
|
ir_ui_menu()
|
|
|
|
|
|
|
|
|