One morning I shot an elephant in my pajamas. How he got in my pajamas, I don't know.
[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     @http.route(['/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>'], type='http', auth="public", website=True, multilang=True)
174     def question(self, forum, question, **post):
175         cr, uid, context = request.cr, request.uid, request.context
176         # increment view counter
177         request.registry['forum.post'].set_viewed(cr, SUPERUSER_ID, [question.id], context=context)
178
179         filters = 'question'
180         values = self._prepare_forum_values(forum=forum, searches=post)
181         values.update({
182             'question': question,
183             'header': {'question_data': True},
184             'filters': filters,
185             'reversed': reversed,
186         })
187         return request.website.render("website_forum.post_description_full", values)
188
189     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/toggle_favourite', type='json', auth="user", multilang=True, methods=['POST'], website=True)
190     def question_toggle_favorite(self, forum, question, **post):
191         if not request.session.uid:
192             return {'error': 'anonymous_user'}
193         # TDE: add check for not public
194         favourite = False if question.user_favourite else True
195         if favourite:
196             favourite_ids = [(4, request.uid)]
197         else:
198             favourite_ids = [(3, request.uid)]
199         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'favourite_ids': favourite_ids}, context=request.context)
200         return favourite
201
202     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/ask_for_close', type='http', auth="user", multilang=True, website=True)
203     def question_ask_for_close(self, forum, question, **post):
204         cr, uid, context = request.cr, request.uid, request.context
205         Reason = request.registry['forum.post.reason']
206         reason_ids = Reason.search(cr, uid, [], context=context)
207         reasons = Reason.browse(cr, uid, reason_ids, context)
208
209         values = self._prepare_forum_values(**post)
210         values.update({
211             'post': question,
212             'forum': forum,
213             'reasons': reasons,
214         })
215         return request.website.render("website_forum.close_question", values)
216
217     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/edit_answer', type='http', auth="user", website=True, multilang=True)
218     def question_edit_answer(self, forum, question, **kwargs):
219         for record in question.child_ids:
220             if record.create_uid.id == request.uid:
221                 answer = record
222                 break
223         return werkzeug.utils.redirect("/forum/%s/post/%s/edit" % (slug(forum), slug(answer)))
224
225     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/close', type='http', auth="user", multilang=True, methods=['POST'], website=True)
226     def question_close(self, forum, question, **post):
227         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {
228             'state': 'close',
229             'closed_uid': request.uid,
230             'closed_date': datetime.today().strftime(tools.DEFAULT_SERVER_DATETIME_FORMAT),
231             'closed_reason_id': post.get('reason_id', False),
232         }, context=request.context)
233         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
234
235     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/reopen', type='http', auth="user", multilang=True, website=True)
236     def question_reopen(self, forum, question, **kwarg):
237         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'state': 'active'}, 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>/delete', type='http', auth="user", multilang=True, website=True)
241     def question_delete(self, forum, question, **kwarg):
242         #instead of unlink record just change 'active' to false so user can undelete it.
243         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': False}, context=request.context)
244         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
245
246     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/undelete', type='http', auth="user", multilang=True, website=True)
247     def question_undelete(self, forum, question, **kwarg):
248         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': True}, context=request.context)
249         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
250
251     # Post
252     # --------------------------------------------------
253
254     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/new', type='http', auth="public", multilang=True, methods=['POST'], website=True)
255     def post_new(self, forum, post, **kwargs):
256         if not request.session.uid:
257             return login_redirect()
258         request.registry['forum.post'].create(
259             request.cr, request.uid, {
260                 'forum_id': forum.id,
261                 'parent_id': post.id,
262                 'content': kwargs.get('content'),
263             }, context=request.context)
264         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(post)))
265
266     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/comment', type='http', auth="public", methods=['POST'], website=True)
267     def post_comment(self, forum, post, **kwargs):
268         if not request.session.uid:
269             return login_redirect()
270         question = post.parent_id if post.parent_id else post
271         cr, uid, context = request.cr, request.uid, request.context
272         if kwargs.get('comment') and post.forum_id.id == forum.id:
273             # TDE FIXME: check that post_id is the question or one of its answers
274             if request.registry['res.users'].has_group(cr, uid, 'website_mail.group_comment'):
275                 request.registry['forum.post'].message_post(
276                     cr, uid, post.id,
277                     body=kwargs.get('comment'),
278                     type='comment',
279                     subtype='mt_comment',
280                     context=dict(context, mail_create_nosubcribe=True))
281         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
282
283     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/toggle_correct', type='json', auth="public", website=True)
284     def post_toggle_correct(self, forum, post, **kwargs):
285         cr, uid, context = request.cr, request.uid, request.context
286         if not request.session.uid:
287             return {'error': 'anonymous_user'}
288         # if user have not access to accept answer then reise warning
289         if post.parent_id is False or post.parent_id.create_uid.id != uid:
290             return {'error': 'own_post'}
291
292         # set all answers to False, only one can be accepted
293         request.registry['forum.post'].write(cr, uid, [c.id for c in post.parent_id.child_ids], {'is_correct': False}, context=context)
294         request.registry['forum.post'].write(cr, uid, [post.id, post.parent_id.id], {'is_correct': not post.is_correct}, context=context)
295         return not post.is_correct
296
297     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/delete', type='http', auth="user", multilang=True, website=True)
298     def post_delete(self, forum, post, **kwargs):
299         question = post.parent_id
300         request.registry['forum.post'].unlink(request.cr, request.uid, [post.id], context=request.context)
301         if question:
302             werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
303         return werkzeug.utils.redirect("/forum/%s" % slug(forum))
304
305     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/edit', type='http', auth="user", website=True, multilang=True)
306     def post_edit(self, forum, post, **kwargs):
307         tags = ""
308         for tag_name in post.tag_ids:
309             tags += tag_name.name + ","
310         values = self._prepare_forum_values(forum=forum)
311         values.update({
312             'tags': tags,
313             'post': post,
314             'is_answer': bool(post.parent_id),
315             'searches': kwargs
316         })
317         return request.website.render("website_forum.edit_post", values)
318
319     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/save', type='http', auth="user", multilang=True, methods=['POST'], website=True)
320     def post_save(self, forum, post, **kwargs):
321         cr, uid, context = request.cr, request.uid, request.context
322         question_tags = []
323         if kwargs.get('question_tag') and kwargs.get('question_tag').strip('[]'):
324             Tag = request.registry['forum.tag']
325             tags = kwargs.get('question_tag').strip('[]').replace('"', '').split(",")
326             for tag in tags:
327                 tag_ids = Tag.search(cr, uid, [('name', '=', tag)], context=context)
328                 if tag_ids:
329                     question_tags += tag_ids
330                 else:
331                     new_tag = Tag.create(cr, uid, {'name': tag, 'forum_id': forum.id}, context=context)
332                     question_tags.append(new_tag)
333         vals = {
334             'tag_ids': [(6, 0, question_tags)],
335             'name': kwargs.get('question_name'),
336             'content': kwargs.get('content'),
337         }
338         request.registry['forum.post'].write(cr, uid, [post.id], vals, context=context)
339         question = post.parent_id if post.parent_id else post
340         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
341
342     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/upvote', type='json', auth="public", multilang=True, website=True)
343     def post_upvote(self, forum, post, **kwargs):
344         # check for karma and not self vote
345         if not request.session.uid:
346             return {'error': 'anonymous_user'}
347         if request.uid == post.create_uid.id:
348             return {'error': 'own_post'}
349         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
350         if user.karma <= 5:
351             return {'error': 'not_enough_karma', 'karma': 1}
352         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=True, context=request.context)
353
354     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/downvote', type='json', auth="public", multilang=True, website=True)
355     def post_downvote(self, forum, post, **kwargs):
356         if not request.session.uid:
357             return {'error': 'anonymous_user'}
358         if request.uid == post.create_uid.id:
359             return {'error': 'own_post'}
360         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
361         if user.karma <= 50:
362             return {'error': 'not_enough_karma', 'karma': 50}
363         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=False, context=request.context)
364
365     # User
366     # --------------------------------------------------
367
368     @http.route('/forum/<model("forum.forum"):forum>/users', type='http', auth="public", website=True, multilang=True)
369     def users(self, forum, page=1, **searches):
370         cr, uid, context = request.cr, request.uid, request.context
371         User = request.registry['res.users']
372
373         step = 30
374         tag_count = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], count=True, context=context)
375         pager = request.website.pager(url="/forum/users", total=tag_count, page=page, step=step, scope=30)
376
377         obj_ids = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], limit=step, offset=pager['offset'], context=context)
378         users = User.browse(cr, SUPERUSER_ID, obj_ids, context=context)
379         searches['users'] = 'True'
380
381         values = self._prepare_forum_values(forum=forum, searches=searches)
382         values .update({
383             'users': users,
384             'notifications': self._get_notifications(),
385             'pager': pager,
386         })
387
388         return request.website.render("website_forum.users", values)
389
390     @http.route(['/forum/<model("forum.forum"):forum>/user/<int:user_id>'], type='http', auth="public", website=True, multilang=True)
391     def open_user(self, forum, user_id=0, **post):
392         cr, uid, context = request.cr, request.uid, request.context
393         User = request.registry['res.users']
394         Post = request.registry['forum.post']
395         Vote = request.registry['forum.post.vote']
396         Activity = request.registry['mail.message']
397         Followers = request.registry['mail.followers']
398         Data = request.registry["ir.model.data"]
399
400         user_id = User.search(cr, SUPERUSER_ID, [('id', '=', user_id), ('karma', '>', '1')], context=context)
401         if not user_id:
402             return werkzeug.utils.redirect("/forum/%s" % slug(forum))
403         user = User.browse(cr, SUPERUSER_ID, user_id[0], context=context)
404
405         # questions and answers by user
406         user_questions, user_answers = [], []
407         user_post_ids = Post.search(
408             cr, uid, [
409                 ('forum_id', '=', forum.id), ('create_uid', '=', user.id),
410                 '|', ('active', '=', False), ('active', '=', True)], context=context)
411         user_posts = Post.browse(cr, uid, user_post_ids, context=context)
412         for record in user_posts:
413             if record.parent_id:
414                 user_answers.append(record)
415             else:
416                 user_questions.append(record)
417
418         # showing questions which user following
419         obj_ids = Followers.search(cr, SUPERUSER_ID, [('res_model', '=', 'forum.post'), ('partner_id', '=', user.partner_id.id)], context=context)
420         post_ids = [follower.res_id for follower in Followers.browse(cr, SUPERUSER_ID, obj_ids, context=context)]
421         que_ids = Post.search(cr, uid, [('id', 'in', post_ids), ('forum_id', '=', forum.id), ('parent_id', '=', False)], context=context)
422         followed = Post.browse(cr, uid, que_ids, context=context)
423
424         #showing Favourite questions of user.
425         fav_que_ids = Post.search(cr, uid, [('favourite_ids', '=', user.id), ('forum_id', '=', forum.id), ('parent_id', '=', False)], context=context)
426         favourite = Post.browse(cr, uid, fav_que_ids, context=context)
427
428         #votes which given on users questions and answers.
429         data = Vote.read_group(cr, uid, [('post_id.forum_id', '=', forum.id), ('post_id.create_uid', '=', user.id)], ["vote"], groupby=["vote"], context=context)
430         up_votes, down_votes = 0, 0
431         for rec in data:
432             if rec['vote'] == '1':
433                 up_votes = rec['vote_count']
434             elif rec['vote'] == '-1':
435                 down_votes = rec['vote_count']
436         total_votes = up_votes + down_votes
437
438         #Votes which given by users on others questions and answers.
439         post_votes = Vote.search(cr, uid, [('user_id', '=', user.id)], context=context)
440         vote_ids = Vote.browse(cr, uid, post_votes, context=context)
441
442         #activity by user.
443         model, comment = Data.get_object_reference(cr, uid, 'mail', 'mt_comment')
444         activity_ids = Activity.search(cr, uid, [('res_id', 'in', user_post_ids), ('model', '=', 'forum.post'), ('subtype_id', '!=', comment)], context=context)
445         activities = Activity.browse(cr, uid, activity_ids, context=context)
446
447         posts = {}
448         for act in activities:
449             posts[act.res_id] = True
450         posts_ids = Post.browse(cr, uid, posts.keys(), context=context)
451         posts = dict(map(lambda x: (x.id, (x.parent_id or x, x.parent_id and x or False)), posts_ids))
452
453         post['users'] = 'True'
454
455         values = self._prepare_forum_values(**post)
456         values.update({
457             'uid': uid,
458             'user': user,
459             'main_object': user,
460             'searches': post,
461             'forum': forum,
462             'questions': user_questions,
463             'answers': user_answers,
464             'followed': followed,
465             'favourite': favourite,
466             'total_votes': total_votes,
467             'up_votes': up_votes,
468             'down_votes': down_votes,
469             'activities': activities,
470             'posts': posts,
471             'vote_post': vote_ids,
472         })
473         return request.website.render("website_forum.user_detail_full", values)
474
475     @http.route('/forum/<model("forum.forum"):forum>/user/<model("res.users"):user>/edit', type='http', auth="user", multilang=True, website=True)
476     def edit_profile(self, forum, user, **kwargs):
477         country = request.registry['res.country']
478         country_ids = country.search(request.cr, SUPERUSER_ID, [], context=request.context)
479         countries = country.browse(request.cr, SUPERUSER_ID, country_ids, context=request.context)
480         values = self._prepare_forum_values(forum=forum, searches=kwargs)
481         values.update({
482             'countries': countries,
483             'notifications': self._get_notifications(),
484         })
485         return request.website.render("website_forum.edit_profile", values)
486
487     @http.route('/forum/<model("forum.forum"):forum>/user/<model("res.users"):user>/save', type='http', auth="user", multilang=True, website=True)
488     def save_edited_profile(self, forum, user, **kwargs):
489         request.registry['res.users'].write(request.cr, request.uid, [user.id], {
490             'name': kwargs.get('name'),
491             'website': kwargs.get('website'),
492             'email': kwargs.get('email'),
493             'city': kwargs.get('city'),
494             'country_id': kwargs.get('country'),
495             'website_description': kwargs.get('description'),
496         }, context=request.context)
497         return werkzeug.utils.redirect("/forum/%s/user/%d" % (slug(forum), user.id))
498
499     # Badges
500     # --------------------------------------------------
501
502     @http.route('/forum/<model("forum.forum"):forum>/badge', type='http', auth="public", website=True, multilang=True)
503     def badges(self, forum, **searches):
504         cr, uid, context = request.cr, request.uid, request.context
505         Badge = request.registry['gamification.badge']
506         badge_ids = Badge.search(cr, SUPERUSER_ID, [('challenge_ids.category', '=', 'forum')], context=context)
507         badges = Badge.browse(cr, uid, badge_ids, context=context)
508         values = self._prepare_forum_values(forum=forum, searches={'badges': True})
509         values.update({
510             'badges': badges,
511         })
512         return request.website.render("website_forum.badge", values)
513
514     @http.route(['/forum/<model("forum.forum"):forum>/badge/<model("gamification.badge"):badge>'], type='http', auth="public", website=True, multilang=True)
515     def badge_users(self, forum, badge, **kwargs):
516         user_ids = [badge_user.user_id.id for badge_user in badge.owner_ids]
517         users = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, user_ids, context=request.context)
518
519         values = self._prepare_forum_values(forum=forum, searches={'badges': True})
520         values.update({
521             'badge': badge,
522             'users': users,
523         })
524         return request.website.render("website_forum.badge_user", values)
525
526     # Messaging
527     # --------------------------------------------------
528
529     @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)
530     def convert_comment_to_answer(self, forum, post, comment, **kwarg):
531         values = {
532             'content': comment.body,
533         }
534         request.registry['mail.message'].unlink(request.cr, request.uid, [comment.id], context=request.context)
535         question = post.parent_id if post.parent_id else post
536         return self.post_new(forum, question, **values)
537
538     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/convert_to_comment', type='http', auth="user", multilang=True, website=True)
539     def convert_answer_to_comment(self, forum, post, **kwarg):
540         values = {
541             'comment': html2plaintext(post.content),
542         }
543         question = post.parent_id
544         request.registry['forum.post'].unlink(request.cr, SUPERUSER_ID, [post.id], context=request.context)
545         return self.post_comment(forum, question, **values)
546
547     @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)
548     def delete_comment(self, forum, post, comment, **kwarg):
549         request.registry['mail.message'].unlink(request.cr, SUPERUSER_ID, [comment.id], context=request.context)
550         return True