#
##############################################################################
-import datetime as DT
import openerp
import openerp.tools as tools
-from operator import itemgetter
-from osv import osv
-from osv import fields
-import tools
-from tools.translate import _
-from lxml import etree
+from openerp.osv import osv
+from openerp.osv import fields
+from openerp import SUPERUSER_ID
+
class mail_group(osv.Model):
- """
- A mail_group is a collection of users sharing messages in a discussion
- group. Group users are users that follow the mail group, using the
- subscription/follow mechanism of OpenSocial. A mail group has nothing
- in common with res.users.group.
- """
-
+ """ A mail_group is a collection of users sharing messages in a discussion
+ group. The group mechanics are based on the followers. """
_description = 'Discussion group'
_name = 'mail.group'
+ _mail_flat_thread = False
_inherit = ['mail.thread']
- _inherits = {'mail.alias': 'alias_id', 'ir.ui.menu': 'menu_id'}
+ _inherits = {'mail.alias': 'alias_id'}
def _get_image(self, cr, uid, ids, name, args, context=None):
result = dict.fromkeys(ids, False)
for obj in self.browse(cr, uid, ids, context=context):
result[obj.id] = tools.image_get_resized_images(obj.image)
return result
-
+
def _set_image(self, cr, uid, id, name, value, args, context=None):
return self.write(cr, uid, [id], {'image': tools.image_resize_image_big(value)}, context=context)
-
- def _get_last_month_msg_nbr(self, cr, uid, ids, name, args, context=None):
- result = {}
- message_obj = self.pool.get('mail.message')
- for id in ids:
- lower_date = (DT.datetime.now() - DT.timedelta(days=30)).strftime(tools.DEFAULT_SERVER_DATE_FORMAT)
- result[id] = self.message_search(cr, uid, [id], limit=None, domain=[('date', '>=', lower_date)], count=True, context=context)
- return result
-
- def _get_default_image(self, cr, uid, context=None):
- image_path = openerp.modules.get_module_resource('mail', 'static/src/img', 'groupdefault.png')
- return tools.image_resize_image_big(open(image_path, 'rb').read().encode('base64'))
-
+
_columns = {
+ 'name': fields.char('Name', size=64, required=True, translate=True),
'description': fields.text('Description'),
'menu_id': fields.many2one('ir.ui.menu', string='Related Menu', required=True, ondelete="cascade"),
- 'responsible_id': fields.many2one('res.users', string='Responsible',
- ondelete='set null', required=True, select=1,
- help="Responsible of the group that has all rights on the record."),
- 'public': fields.selection([('public', 'Public'), ('private', 'Private'), ('groups', 'Selected Group Only')],
- string='Privacy', required=True,
- help='This group is visible by non members. '\
- 'Invisible groups can add members through the invite button.'),
+ 'public': fields.selection([('public', 'Public'), ('private', 'Private'), ('groups', 'Selected Group Only')], 'Privacy', required=True,
+ help='This group is visible by non members. \
+ Invisible groups can add members through the invite button.'),
'group_public_id': fields.many2one('res.groups', string='Authorized Group'),
'group_ids': fields.many2many('res.groups', rel='mail_group_res_group_rel',
id1='mail_group_id', id2='groups_id', string='Auto Subscription',
help="Members of those groups will automatically added as followers. "\
"Note that they will be able to manage their subscription manually "\
"if necessary."),
+ # image: all image fields are base64 encoded and PIL-supported
'image': fields.binary("Photo",
- help="This field holds the image used as photo for the "\
- "user. The image is base64 encoded, and PIL-supported. "\
- "It is limited to a 12024x1024 px image."),
+ help="This field holds the image used as photo for the group, limited to 1024x1024px."),
'image_medium': fields.function(_get_image, fnct_inv=_set_image,
string="Medium-sized photo", type="binary", multi="_get_image",
- store = {
+ store={
'mail.group': (lambda self, cr, uid, ids, c={}: ids, ['image'], 10),
},
help="Medium-sized photo of the group. It is automatically "\
- "resized as a 180x180px image, with aspect ratio preserved. "\
+ "resized as a 128x128px image, with aspect ratio preserved. "\
"Use this field in form views or some kanban views."),
'image_small': fields.function(_get_image, fnct_inv=_set_image,
string="Small-sized photo", type="binary", multi="_get_image",
- store = {
+ store={
'mail.group': (lambda self, cr, uid, ids, c={}: ids, ['image'], 10),
},
help="Small-sized photo of the group. It is automatically "\
- "resized as a 50x50px image, with aspect ratio preserved. "\
+ "resized as a 64x64px image, with aspect ratio preserved. "\
"Use this field anywhere a small image is required."),
- 'last_month_msg_nbr': fields.function(_get_last_month_msg_nbr, type='integer',
- string='Messages count for last month'),
- 'alias_id': fields.many2one('mail.alias', 'Alias', ondelete="cascade",
- help="The email address associated with this group. New emails received will automatically "
- "create new topics."),
+ 'alias_id': fields.many2one('mail.alias', 'Alias', ondelete="cascade", required=True,
+ help="The email address associated with this group. New emails received will automatically "
+ "create new topics."),
}
def _get_default_employee_group(self, cr, uid, context=None):
ref = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'base', 'group_user')
return ref and ref[1] or False
- def _get_menu_parent(self, cr, uid, context=None):
- ref = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'mail', 'mail_group_root')
- return ref and ref[1] or False
+ def _get_default_image(self, cr, uid, context=None):
+ image_path = openerp.modules.get_module_resource('mail', 'static/src/img', 'groupdefault.png')
+ return tools.image_resize_image_big(open(image_path, 'rb').read().encode('base64'))
_defaults = {
'public': 'groups',
'group_public_id': _get_default_employee_group,
- 'responsible_id': (lambda s, cr, uid, ctx: uid),
'image': _get_default_image,
- 'parent_id': _get_menu_parent,
+ 'alias_domain': False, # always hide alias during creation
}
- def _subscribe_user_with_group_m2m_command(self, cr, uid, ids, group_ids_command, context=None):
- # form: {'group_ids': [(3, 10), (3, 3), (4, 10), (4, 3)]} or {'group_ids': [(6, 0, [ids]}
- user_group_ids = [command[1] for command in group_ids_command if command[0] == 4]
- user_group_ids += [id for command in group_ids_command if command[0] == 6 for id in command[2]]
- # retrieve the user member of those groups
- user_ids = []
- res_groups_obj = self.pool.get('res.groups')
- for group in res_groups_obj.browse(cr, uid, user_group_ids, context=context):
- user_ids += [user.id for user in group.users]
- # subscribe the users
- return self.message_subscribe(cr, uid, ids, user_ids, context=context)
+ def _generate_header_description(self, cr, uid, group, context=None):
+ header = ''
+ if group.description:
+ header = '%s' % group.description
+ if group.alias_id and group.alias_id.alias_name and group.alias_id.alias_domain:
+ if header:
+ header = '%s<br/>' % header
+ return '%sGroup email gateway: %s@%s' % (header, group.alias_id.alias_name, group.alias_id.alias_domain)
+ return header
+
+ def _subscribe_users(self, cr, uid, ids, context=None):
+ for mail_group in self.browse(cr, uid, ids, context=context):
+ partner_ids = []
+ for group in mail_group.group_ids:
+ partner_ids += [user.partner_id.id for user in group.users]
+ self.message_subscribe(cr, uid, ids, partner_ids, context=context)
def create(self, cr, uid, vals, context=None):
- alias_pool = self.pool.get('mail.alias')
+ mail_alias = self.pool.get('mail.alias')
if not vals.get('alias_id'):
- name = vals.get('alias_name') or vals['name']
- alias_id = alias_pool.create_unique_alias(cr, uid,
- {'alias_name': "group_"+name},
- model_name=self._name, context=context)
+ vals.pop('alias_name', None) # prevent errors during copy()
+ alias_id = mail_alias.create_unique_alias(cr, uid,
+ # Using '+' allows using subaddressing for those who don't
+ # have a catchall domain setup.
+ {'alias_name': "group+" + vals['name']},
+ model_name=self._name, context=context)
vals['alias_id'] = alias_id
- mail_group_id = super(mail_group, self).create(cr, uid, vals, context)
+ # get parent menu
+ menu_parent = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'mail', 'mail_group_root')
+ menu_parent = menu_parent and menu_parent[1] or False
+
+ # Create menu id
+ mobj = self.pool.get('ir.ui.menu')
+ menu_id = mobj.create(cr, SUPERUSER_ID, {'name': vals['name'], 'parent_id': menu_parent}, context=context)
+ vals['menu_id'] = menu_id
+
+ # Create group and alias
+ mail_group_id = super(mail_group, self).create(cr, uid, vals, context=context)
+ mail_alias.write(cr, uid, [vals['alias_id']], {"alias_force_thread_id": mail_group_id}, context)
+ group = self.browse(cr, uid, mail_group_id, context=context)
# Create client action for this group and link the menu to it
ref = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'mail', 'action_mail_group_feeds')
if ref:
- search_ref = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'mail', 'view_message_search_wall')
+ search_ref = self.pool.get('ir.model.data').get_object_reference(cr, uid, 'mail', 'view_message_search')
params = {
'search_view_id': search_ref and search_ref[1] or False,
- 'domain': [('model','=','mail.group'),('res_id','=',mail_group_id)],
- 'res_model': 'mail.group',
- 'res_id': mail_group_id,
- 'thread_level': 2
+ 'domain': [('model', '=', 'mail.group'), ('res_id', '=', mail_group_id)],
+ 'context': {'default_model': 'mail.group', 'default_res_id': mail_group_id, 'search_default_message_unread': True},
+ 'res_model': 'mail.message',
+ 'thread_level': 1,
+ 'header_description': self._generate_header_description(cr, uid, group, context=context)
}
cobj = self.pool.get('ir.actions.client')
- newref = cobj.copy(cr, uid, ref[1], default={'params': str(params), 'name': vals['name']}, context=context)
- self.write(cr, uid, [mail_group_id], {'action': 'ir.actions.client,'+str(newref), 'mail_group_id': mail_group_id}, context=context)
+ newref = cobj.copy(cr, SUPERUSER_ID, ref[1], default={'params': str(params), 'name': vals['name']}, context=context)
+ mobj.write(cr, SUPERUSER_ID, menu_id, {'action': 'ir.actions.client,' + str(newref), 'mail_group_id': mail_group_id}, context=context)
- alias_pool.write(cr, uid, [vals['alias_id']], {"alias_force_thread_id": mail_group_id}, context)
if vals.get('group_ids'):
- self._subscribe_user_with_group_m2m_command(cr, uid, [mail_group_id], vals.get('group_ids'), context=context)
-
+ self._subscribe_users(cr, uid, [mail_group_id], context=context)
return mail_group_id
def unlink(self, cr, uid, ids, context=None):
+ groups = self.browse(cr, uid, ids, context=context)
# Cascade-delete mail aliases as well, as they should not exist without the mail group.
mail_alias = self.pool.get('mail.alias')
- alias_ids = [group.alias_id.id for group in self.browse(cr, uid, ids, context=context) if group.alias_id]
+ alias_ids = [group.alias_id.id for group in groups if group.alias_id]
+ # Delete mail_group
res = super(mail_group, self).unlink(cr, uid, ids, context=context)
- mail_alias.unlink(cr, uid, alias_ids, context=context)
+ # Delete alias
+ mail_alias.unlink(cr, SUPERUSER_ID, alias_ids, context=context)
+ # Cascade-delete menu entries as well
+ self.pool.get('ir.ui.menu').unlink(cr, SUPERUSER_ID, [group.menu_id.id for group in groups if group.menu_id], context=context)
return res
def write(self, cr, uid, ids, vals, context=None):
+ result = super(mail_group, self).write(cr, uid, ids, vals, context=context)
if vals.get('group_ids'):
- self._subscribe_user_with_group_m2m_command(cr, uid, ids, vals.get('group_ids'), context=context)
- return super(mail_group, self).write(cr, uid, ids, vals, context=context)
+ self._subscribe_users(cr, uid, ids, context=context)
+ # if description, name or alias is changed: update client action
+ if vals.get('description') or vals.get('name') or vals.get('alias_id') or vals.get('alias_name'):
+ cobj = self.pool.get('ir.actions.client')
+ for action in [group.menu_id.action for group in self.browse(cr, uid, ids, context=context)]:
+ new_params = action.params
+ new_params['header_description'] = self._generate_header_description(cr, uid, group, context=context)
+ cobj.write(cr, SUPERUSER_ID, [action.id], {'params': str(new_params)}, context=context)
+ # if name is changed: update menu
+ if vals.get('name'):
+ mobj = self.pool.get('ir.ui.menu')
+ mobj.write(cr, SUPERUSER_ID,
+ [group.menu_id.id for group in self.browse(cr, uid, ids, context=context)],
+ {'name': vals.get('name')}, context=context)
+
+ return result
- def action_group_join(self, cr, uid, ids, context=None):
- return self.message_subscribe(cr, uid, ids, context=context)
+ def action_follow(self, cr, uid, ids, context=None):
+ """ Wrapper because message_subscribe_users take a user_ids=None
+ that receive the context without the wrapper. """
+ return self.message_subscribe_users(cr, uid, ids, context=context)
- def action_group_leave(self, cr, uid, ids, context=None):
- return self.message_unsubscribe(cr, uid, ids, context=context)
+ def action_unfollow(self, cr, uid, ids, context=None):
+ """ Wrapper because message_unsubscribe_users take a user_ids=None
+ that receive the context without the wrapper. """
+ return self.message_unsubscribe_users(cr, uid, ids, context=context)