0c83b3bb2591fa7e9ce03bc16c6595091ff7f720
[odoo/odoo.git] / addons / website_forum / controllers / main.py
1 # -*- coding: utf-8 -*-
2
3 from datetime import datetime
4 import werkzeug.urls
5 import simplejson
6
7 from openerp import tools
8 from openerp import SUPERUSER_ID
9 from openerp.addons.web import http
10 from openerp.addons.web.controllers.main import login_redirect
11 from openerp.addons.web.http import request
12 from openerp.addons.website.controllers.main import Website as controllers
13 from openerp.addons.website.models.website import slug
14 from openerp.tools import html2plaintext
15
16 controllers = controllers()
17
18
19 class WebsiteForum(http.Controller):
20     _post_per_page = 10
21     _user_per_page = 30
22
23     def _get_notifications(self):
24         cr, uid, context = request.cr, request.uid, request.context
25         Message = request.registry['mail.message']
26         badge_st_id = request.registry['ir.model.data'].xmlid_to_res_id(cr, uid, 'gamification.mt_badge_granted')
27         if badge_st_id:
28             msg_ids = Message.search(cr, uid, [('subtype_id', '=', badge_st_id), ('to_read', '=', True)], context=context)
29             msg = Message.browse(cr, uid, msg_ids, context=context)
30         else:
31             msg = list()
32         return msg
33
34     def _prepare_forum_values(self, forum=None, **kwargs):
35         user = request.registry['res.users'].browse(request.cr, request.uid, request.uid, context=request.context)
36         public_uid = request.registry['website'].get_public_user(request.cr, request.uid, request.context)
37         values = {'user': user, 'is_public_user': user.id == public_uid,
38                   'notifications': self._get_notifications(),
39                   'header': kwargs.get('header', dict()),
40                   'searches': kwargs.get('searches', dict())}
41         if forum:
42             values['forum'] = forum
43         elif kwargs.get('forum_id'):
44             values['forum'] = request.registry['forum.forum'].browse(request.cr, request.uid, kwargs.pop('forum_id'), context=request.context)
45         values.update(kwargs)
46         return values
47
48     # Forum
49     # --------------------------------------------------
50
51     @http.route(['/forum'], type='http', auth="public", website=True, multilang=True)
52     def forum(self, **kwargs):
53         cr, uid, context = request.cr, request.uid, request.context
54         Forum = request.registry['forum.forum']
55         obj_ids = Forum.search(cr, uid, [], context=context)
56         forums = Forum.browse(cr, uid, obj_ids, context=context)
57         return request.website.render("website_forum.forum_all", {'forums': forums})
58
59     @http.route('/forum/new', type='http', auth="user", multilang=True, website=True)
60     def forum_create(self, forum_name="New Forum", **kwargs):
61         forum_id = request.registry['forum.forum'].create(request.cr, request.uid, {
62             'name': forum_name,
63         }, context=request.context)
64         return request.redirect("/forum/%s" % slug(forum_id))
65
66     @http.route('/forum/notification_read', type='json', auth="user", multilang=True, methods=['POST'], website=True)
67     def notification_read(self, **kwargs):
68         request.registry['mail.message'].set_message_read(request.cr, request.uid, [int(kwargs.get('notification_id'))], read=True, context=request.context)
69         return True
70
71     @http.route(['/forum/<model("forum.forum"):forum>',
72                  '/forum/<model("forum.forum"):forum>/page/<int:page>',
73                  '/forum/<model("forum.forum"):forum>/tag/<model("forum.tag"):tag>/questions'
74                  ], type='http', auth="public", website=True, multilang=True)
75     def questions(self, forum, tag=None, page=1, filters='all', sorting='date', search='', **post):
76         cr, uid, context = request.cr, request.uid, request.context
77         Post = request.registry['forum.post']
78         user = request.registry['res.users'].browse(cr, uid, uid, context=context)
79
80         domain = [('forum_id', '=', forum.id), ('parent_id', '=', False)]
81         if search:
82             domain += ['|', ('name', 'ilike', search), ('content', 'ilike', search)]
83         if tag:
84             domain += [('tag_ids', 'in', tag.id)]
85         if filters == 'unanswered':
86             domain += [('child_ids', '=', False)]
87         elif filters == 'followed':
88             domain += [('message_follower_ids', '=', user.partner_id.id)]
89         else:
90             filters = 'all'
91
92         if sorting == 'answered':
93             order = 'child_count desc'
94         elif sorting == 'vote':
95             order = 'vote_count desc'
96         else:
97             sorting = 'date'
98             order = 'write_date desc'
99
100         question_count = Post.search(cr, uid, domain, count=True, context=context)
101         pager = request.website.pager(url="/forum/%s" % slug(forum), total=question_count, page=page, step=self._post_per_page, scope=self._post_per_page)
102
103         obj_ids = Post.search(cr, uid, domain, limit=self._post_per_page, offset=pager['offset'], order=order, context=context)
104         question_ids = Post.browse(cr, uid, obj_ids, context=context)
105
106         values = self._prepare_forum_values(forum=forum, searches=post)
107         values.update({
108             'question_ids': question_ids,
109             'pager': pager,
110             'tag': tag,
111             'filters': filters,
112             'sorting': sorting,
113             'search': search,
114         })
115         return request.website.render("website_forum.forum_index", values)
116
117     @http.route(['/forum/<model("forum.forum"):forum>/faq'], type='http', auth="public", website=True, multilang=True)
118     def forum_faq(self, forum, **post):
119         values = self._prepare_forum_values(forum=forum, searches=dict(), **post)
120         return request.website.render("website_forum.faq", values)
121
122     @http.route('/forum/get_tags', type='http', auth="public", multilang=True, methods=['GET'], website=True)
123     def tag_read(self, **post):
124         tags = request.registry['forum.tag'].search_read(request.cr, request.uid, [], ['name'], context=request.context)
125         data = [tag['name'] for tag in tags]
126         return simplejson.dumps(data)
127
128     @http.route(['/forum/<model("forum.forum"):forum>/tag'], type='http', auth="public", website=True, multilang=True)
129     def tags(self, forum, page=1, **post):
130         cr, uid, context = request.cr, request.uid, request.context
131         Tag = request.registry['forum.tag']
132         obj_ids = Tag.search(cr, uid, [('forum_id', '=', forum.id)], limit=None, context=context)
133         tags = Tag.browse(cr, uid, obj_ids, context=context)
134         values = self._prepare_forum_values(forum=forum, searches={'tags': True}, **post)
135         values.update({
136             'tags': tags,
137         })
138         return request.website.render("website_forum.tag", values)
139
140     # Questions
141     # --------------------------------------------------
142
143     @http.route(['/forum/<model("forum.forum"):forum>/ask'], type='http', auth="public", website=True, multilang=True)
144     def question_ask(self, forum, **post):
145         if not request.session.uid:
146             return login_redirect()
147         values = self._prepare_forum_values(forum=forum, searches={},  header={'ask_hide': True})
148         return request.website.render("website_forum.ask_question", values)
149
150     @http.route('/forum/<model("forum.forum"):forum>/question/new', type='http', auth="user", multilang=True, methods=['POST'], website=True)
151     def question_create(self, forum, **post):
152         cr, uid, context = request.cr, request.uid, request.context
153         Tag = request.registry['forum.tag']
154         question_tag_ids = []
155         if post.get('question_tags').strip('[]'):
156             tags = post.get('question_tags').strip('[]').replace('"', '').split(",")
157             for tag in tags:
158                 tag_ids = Tag.search(cr, uid, [('name', '=', tag)], context=context)
159                 if tag_ids:
160                     question_tag_ids.append((4, tag_ids[0]))
161                 else:
162                     question_tag_ids.append((0, 0, {'name': tag, 'forum_id': forum.id}))
163
164         new_question_id = request.registry['forum.post'].create(
165             request.cr, request.uid, {
166                 'forum_id': forum.id,
167                 'name': post.get('question_name'),
168                 'content': post.get('content'),
169                 'tag_ids': question_tag_ids,
170             }, context=context)
171         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), new_question_id))
172
173     def prepare_question_values(self, forum=None, **kwargs):
174         '''Overwrite value in website_doc'''
175         values = self._prepare_forum_values(forum=forum, searches=kwargs)
176         return values
177
178     @http.route(['/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>'], type='http', auth="public", website=True, multilang=True)
179     def question(self, forum, question, **post):
180         cr, uid, context = request.cr, request.uid, request.context
181         # increment view counter
182         request.registry['forum.post'].set_viewed(cr, SUPERUSER_ID, [question.id], context=context)
183
184         filters = 'question'
185         values = self.prepare_question_values(forum=forum, kwargs=post)
186         values.update({
187             'question': question,
188             'header': {'question_data': True},
189             'filters': filters,
190             'reversed': reversed,
191         })
192         return request.website.render("website_forum.post_description_full", values)
193
194     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/toggle_favourite', type='json', auth="user", multilang=True, methods=['POST'], website=True)
195     def question_toggle_favorite(self, forum, question, **post):
196         if not request.session.uid:
197             return {'error': 'anonymous_user'}
198         # TDE: add check for not public
199         favourite = False if question.user_favourite else True
200         if favourite:
201             favourite_ids = [(4, request.uid)]
202         else:
203             favourite_ids = [(3, request.uid)]
204         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'favourite_ids': favourite_ids}, context=request.context)
205         return favourite
206
207     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/ask_for_close', type='http', auth="user", multilang=True, website=True)
208     def question_ask_for_close(self, forum, question, **post):
209         cr, uid, context = request.cr, request.uid, request.context
210         Reason = request.registry['forum.post.reason']
211         reason_ids = Reason.search(cr, uid, [], context=context)
212         reasons = Reason.browse(cr, uid, reason_ids, context)
213
214         values = self._prepare_forum_values(**post)
215         values.update({
216             'post': question,
217             'forum': forum,
218             'reasons': reasons,
219         })
220         return request.website.render("website_forum.close_question", values)
221
222     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/edit_answer', type='http', auth="user", website=True, multilang=True)
223     def question_edit_answer(self, forum, question, **kwargs):
224         for record in question.child_ids:
225             if record.create_uid.id == request.uid:
226                 answer = record
227                 break
228         return werkzeug.utils.redirect("/forum/%s/post/%s/edit" % (slug(forum), slug(answer)))
229
230     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/close', type='http', auth="user", multilang=True, methods=['POST'], website=True)
231     def question_close(self, forum, question, **post):
232         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {
233             'state': 'close',
234             'closed_uid': request.uid,
235             'closed_date': datetime.today().strftime(tools.DEFAULT_SERVER_DATETIME_FORMAT),
236             'closed_reason_id': post.get('reason_id', False),
237         }, context=request.context)
238         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
239
240     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/reopen', type='http', auth="user", multilang=True, website=True)
241     def question_reopen(self, forum, question, **kwarg):
242         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'state': 'active'}, context=request.context)
243         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
244
245     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/delete', type='http', auth="user", multilang=True, website=True)
246     def question_delete(self, forum, question, **kwarg):
247         #instead of unlink record just change 'active' to false so user can undelete it.
248         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': False}, context=request.context)
249         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
250
251     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/undelete', type='http', auth="user", multilang=True, website=True)
252     def question_undelete(self, forum, question, **kwarg):
253         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': True}, context=request.context)
254         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
255
256     # Post
257     # --------------------------------------------------
258
259     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/new', type='http', auth="public", multilang=True, methods=['POST'], website=True)
260     def post_new(self, forum, post, **kwargs):
261         if not request.session.uid:
262             return login_redirect()
263         request.registry['forum.post'].create(
264             request.cr, request.uid, {
265                 'forum_id': forum.id,
266                 'parent_id': post.id,
267                 'content': kwargs.get('content'),
268             }, context=request.context)
269         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(post)))
270
271     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/comment', type='http', auth="public", methods=['POST'], website=True)
272     def post_comment(self, forum, post, **kwargs):
273         if not request.session.uid:
274             return login_redirect()
275         question = post.parent_id if post.parent_id else post
276         cr, uid, context = request.cr, request.uid, request.context
277         if kwargs.get('comment') and post.forum_id.id == forum.id:
278             # TDE FIXME: check that post_id is the question or one of its answers
279             if request.registry['res.users'].has_group(cr, uid, 'website_mail.group_comment'):
280                 request.registry['forum.post'].message_post(
281                     cr, uid, post.id,
282                     body=kwargs.get('comment'),
283                     type='comment',
284                     subtype='mt_comment',
285                     context=dict(context, mail_create_nosubcribe=True))
286         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
287
288     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/toggle_correct', type='json', auth="public", website=True)
289     def post_toggle_correct(self, forum, post, **kwargs):
290         cr, uid, context = request.cr, request.uid, request.context
291         if not request.session.uid:
292             return {'error': 'anonymous_user'}
293         # if user have not access to accept answer then reise warning
294         if post.parent_id is False or post.parent_id.create_uid.id != uid:
295             return {'error': 'own_post'}
296
297         # set all answers to False, only one can be accepted
298         request.registry['forum.post'].write(cr, uid, [c.id for c in post.parent_id.child_ids], {'is_correct': False}, context=context)
299         request.registry['forum.post'].write(cr, uid, [post.id, post.parent_id.id], {'is_correct': not post.is_correct}, context=context)
300         return not post.is_correct
301
302     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/delete', type='http', auth="user", multilang=True, website=True)
303     def post_delete(self, forum, post, **kwargs):
304         question = post.parent_id
305         request.registry['forum.post'].unlink(request.cr, request.uid, [post.id], context=request.context)
306         if question:
307             werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
308         return werkzeug.utils.redirect("/forum/%s" % slug(forum))
309
310     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/edit', type='http', auth="user", website=True, multilang=True)
311     def post_edit(self, forum, post, **kwargs):
312         tags = ""
313         for tag_name in post.tag_ids:
314             tags += tag_name.name + ","
315         values = self._prepare_forum_values(forum=forum)
316         values.update({
317             'tags': tags,
318             'post': post,
319             'is_answer': bool(post.parent_id),
320             'searches': kwargs
321         })
322         return request.website.render("website_forum.edit_post", values)
323
324     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/save', type='http', auth="user", multilang=True, methods=['POST'], website=True)
325     def post_save(self, forum, post, **kwargs):
326         cr, uid, context = request.cr, request.uid, request.context
327         question_tags = []
328         if kwargs.get('question_tag') and kwargs.get('question_tag').strip('[]'):
329             Tag = request.registry['forum.tag']
330             tags = kwargs.get('question_tag').strip('[]').replace('"', '').split(",")
331             for tag in tags:
332                 tag_ids = Tag.search(cr, uid, [('name', '=', tag)], context=context)
333                 if tag_ids:
334                     question_tags += tag_ids
335                 else:
336                     new_tag = Tag.create(cr, uid, {'name': tag, 'forum_id': forum.id}, context=context)
337                     question_tags.append(new_tag)
338         vals = {
339             'tag_ids': [(6, 0, question_tags)],
340             'name': kwargs.get('question_name'),
341             'content': kwargs.get('content'),
342         }
343         request.registry['forum.post'].write(cr, uid, [post.id], vals, context=context)
344         question = post.parent_id if post.parent_id else post
345         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
346
347     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/upvote', type='json', auth="public", multilang=True, website=True)
348     def post_upvote(self, forum, post, **kwargs):
349         # check for karma and not self vote
350         if not request.session.uid:
351             return {'error': 'anonymous_user'}
352         if request.uid == post.create_uid.id:
353             return {'error': 'own_post'}
354         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
355         if user.karma <= 5:
356             return {'error': 'not_enough_karma', 'karma': 1}
357         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=True, context=request.context)
358
359     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/downvote', type='json', auth="public", multilang=True, website=True)
360     def post_downvote(self, forum, post, **kwargs):
361         if not request.session.uid:
362             return {'error': 'anonymous_user'}
363         if request.uid == post.create_uid.id:
364             return {'error': 'own_post'}
365         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
366         if user.karma <= 50:
367             return {'error': 'not_enough_karma', 'karma': 50}
368         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=False, context=request.context)
369
370     # User
371     # --------------------------------------------------
372
373     @http.route('/forum/<model("forum.forum"):forum>/users', type='http', auth="public", website=True, multilang=True)
374     def users(self, forum, page=1, **searches):
375         cr, uid, context = request.cr, request.uid, request.context
376         User = request.registry['res.users']
377
378         step = 30
379         tag_count = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], count=True, context=context)
380         pager = request.website.pager(url="/forum/users", total=tag_count, page=page, step=step, scope=30)
381
382         obj_ids = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], limit=step, offset=pager['offset'], context=context)
383         users = User.browse(cr, SUPERUSER_ID, obj_ids, context=context)
384         searches['users'] = 'True'
385
386         values = self._prepare_forum_values(forum=forum, searches=searches)
387         values .update({
388             'users': users,
389             'notifications': self._get_notifications(),
390             'pager': pager,
391         })
392
393         return request.website.render("website_forum.users", values)
394
395     @http.route(['/forum/<model("forum.forum"):forum>/user/<int:user_id>'], type='http', auth="public", website=True, multilang=True)
396     def open_user(self, forum, user_id=0, **post):
397         cr, uid, context = request.cr, request.uid, request.context
398         User = request.registry['res.users']
399         Post = request.registry['forum.post']
400         Vote = request.registry['forum.post.vote']
401         Activity = request.registry['mail.message']
402         Followers = request.registry['mail.followers']
403         Data = request.registry["ir.model.data"]
404
405         user_id = User.search(cr, SUPERUSER_ID, [('id', '=', user_id), ('karma', '>', '1')], context=context)
406         if not user_id:
407             return werkzeug.utils.redirect("/forum/%s" % slug(forum))
408         user = User.browse(cr, SUPERUSER_ID, user_id[0], context=context)
409
410         # questions and answers by user
411         user_questions, user_answers = [], []
412         user_post_ids = Post.search(
413             cr, uid, [
414                 ('forum_id', '=', forum.id), ('create_uid', '=', user.id),
415                 '|', ('active', '=', False), ('active', '=', True)], context=context)
416         user_posts = Post.browse(cr, uid, user_post_ids, context=context)
417         for record in user_posts:
418             if record.parent_id:
419                 user_answers.append(record)
420             else:
421                 user_questions.append(record)
422
423         # showing questions which user following
424         obj_ids = Followers.search(cr, SUPERUSER_ID, [('res_model', '=', 'forum.post'), ('partner_id', '=', user.partner_id.id)], context=context)
425         post_ids = [follower.res_id for follower in Followers.browse(cr, SUPERUSER_ID, obj_ids, context=context)]
426         que_ids = Post.search(cr, uid, [('id', 'in', post_ids), ('forum_id', '=', forum.id), ('parent_id', '=', False)], context=context)
427         followed = Post.browse(cr, uid, que_ids, context=context)
428
429         #showing Favourite questions of user.
430         fav_que_ids = Post.search(cr, uid, [('favourite_ids', '=', user.id), ('forum_id', '=', forum.id), ('parent_id', '=', False)], context=context)
431         favourite = Post.browse(cr, uid, fav_que_ids, context=context)
432
433         #votes which given on users questions and answers.
434         data = Vote.read_group(cr, uid, [('post_id.forum_id', '=', forum.id), ('post_id.create_uid', '=', user.id)], ["vote"], groupby=["vote"], context=context)
435         up_votes, down_votes = 0, 0
436         for rec in data:
437             if rec['vote'] == '1':
438                 up_votes = rec['vote_count']
439             elif rec['vote'] == '-1':
440                 down_votes = rec['vote_count']
441         total_votes = up_votes + down_votes
442
443         #Votes which given by users on others questions and answers.
444         post_votes = Vote.search(cr, uid, [('user_id', '=', user.id)], context=context)
445         vote_ids = Vote.browse(cr, uid, post_votes, context=context)
446
447         #activity by user.
448         model, comment = Data.get_object_reference(cr, uid, 'mail', 'mt_comment')
449         activity_ids = Activity.search(cr, uid, [('res_id', 'in', user_post_ids), ('model', '=', 'forum.post'), ('subtype_id', '!=', comment)], context=context)
450         activities = Activity.browse(cr, uid, activity_ids, context=context)
451
452         posts = {}
453         for act in activities:
454             posts[act.res_id] = True
455         posts_ids = Post.browse(cr, uid, posts.keys(), context=context)
456         posts = dict(map(lambda x: (x.id, (x.parent_id or x, x.parent_id and x or False)), posts_ids))
457
458         post['users'] = 'True'
459
460         values = self._prepare_forum_values(**post)
461         values.update({
462             'uid': uid,
463             'user': user,
464             'main_object': user,
465             'searches': post,
466             'forum': forum,
467             'questions': user_questions,
468             'answers': user_answers,
469             'followed': followed,
470             'favourite': favourite,
471             'total_votes': total_votes,
472             'up_votes': up_votes,
473             'down_votes': down_votes,
474             'activities': activities,
475             'posts': posts,
476             'vote_post': vote_ids,
477         })
478         return request.website.render("website_forum.user_detail_full", values)
479
480     @http.route('/forum/<model("forum.forum"):forum>/user/<model("res.users"):user>/edit', type='http', auth="user", multilang=True, website=True)
481     def edit_profile(self, forum, user, **kwargs):
482         country = request.registry['res.country']
483         country_ids = country.search(request.cr, SUPERUSER_ID, [], context=request.context)
484         countries = country.browse(request.cr, SUPERUSER_ID, country_ids, context=request.context)
485         values = self._prepare_forum_values(forum=forum, searches=kwargs)
486         values.update({
487             'countries': countries,
488             'notifications': self._get_notifications(),
489         })
490         return request.website.render("website_forum.edit_profile", values)
491
492     @http.route('/forum/<model("forum.forum"):forum>/user/<model("res.users"):user>/save', type='http', auth="user", multilang=True, website=True)
493     def save_edited_profile(self, forum, user, **kwargs):
494         request.registry['res.users'].write(request.cr, request.uid, [user.id], {
495             'name': kwargs.get('name'),
496             'website': kwargs.get('website'),
497             'email': kwargs.get('email'),
498             'city': kwargs.get('city'),
499             'country_id': kwargs.get('country'),
500             'website_description': kwargs.get('description'),
501         }, context=request.context)
502         return werkzeug.utils.redirect("/forum/%s/user/%d" % (slug(forum), user.id))
503
504     # Badges
505     # --------------------------------------------------
506
507     @http.route('/forum/<model("forum.forum"):forum>/badge', type='http', auth="public", website=True, multilang=True)
508     def badges(self, forum, **searches):
509         cr, uid, context = request.cr, request.uid, request.context
510         Badge = request.registry['gamification.badge']
511         badge_ids = Badge.search(cr, SUPERUSER_ID, [('challenge_ids.category', '=', 'forum')], context=context)
512         badges = Badge.browse(cr, uid, badge_ids, context=context)
513         values = self._prepare_forum_values(forum=forum, searches={'badges': True})
514         values.update({
515             'badges': badges,
516         })
517         return request.website.render("website_forum.badge", values)
518
519     @http.route(['/forum/<model("forum.forum"):forum>/badge/<model("gamification.badge"):badge>'], type='http', auth="public", website=True, multilang=True)
520     def badge_users(self, forum, badge, **kwargs):
521         user_ids = [badge_user.user_id.id for badge_user in badge.owner_ids]
522         users = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, user_ids, context=request.context)
523
524         values = self._prepare_forum_values(forum=forum, searches={'badges': True})
525         values.update({
526             'badge': badge,
527             'users': users,
528         })
529         return request.website.render("website_forum.badge_user", values)
530
531     # Messaging
532     # --------------------------------------------------
533
534     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/comment/<model("mail.message"):comment>/convert_to_answer', type='http', auth="public", multilang=True, website=True)
535     def convert_comment_to_answer(self, forum, post, comment, **kwarg):
536         values = {
537             'content': comment.body,
538         }
539         request.registry['mail.message'].unlink(request.cr, request.uid, [comment.id], context=request.context)
540         question = post.parent_id if post.parent_id else post
541         return self.post_new(forum, question, **values)
542
543     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/convert_to_comment', type='http', auth="user", multilang=True, website=True)
544     def convert_answer_to_comment(self, forum, post, **kwarg):
545         values = {
546             'comment': html2plaintext(post.content),
547         }
548         question = post.parent_id
549         request.registry['forum.post'].unlink(request.cr, SUPERUSER_ID, [post.id], context=request.context)
550         return self.post_comment(forum, question, **values)
551
552     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/comment/<model("mail.message"):comment>/delete', type='json', auth="user", multilang=True, website=True)
553     def delete_comment(self, forum, post, comment, **kwarg):
554         request.registry['mail.message'].unlink(request.cr, SUPERUSER_ID, [comment.id], context=request.context)
555         return True