[MERGE] new module: website_forum_doc + small fixes (close) in forum
[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             'main_object': tag or forum,
109             'question_ids': question_ids,
110             'pager': pager,
111             'tag': tag,
112             'filters': filters,
113             'sorting': sorting,
114             'search': search,
115         })
116         return request.website.render("website_forum.forum_index", values)
117
118     @http.route(['/forum/<model("forum.forum"):forum>/faq'], type='http', auth="public", website=True, multilang=True)
119     def forum_faq(self, forum, **post):
120         values = self._prepare_forum_values(forum=forum, searches=dict(), **post)
121         return request.website.render("website_forum.faq", values)
122
123     @http.route('/forum/get_tags', type='http', auth="public", multilang=True, methods=['GET'], website=True)
124     def tag_read(self, **post):
125         tags = request.registry['forum.tag'].search_read(request.cr, request.uid, [], ['name'], context=request.context)
126         data = [tag['name'] for tag in tags]
127         return simplejson.dumps(data)
128
129     @http.route(['/forum/<model("forum.forum"):forum>/tag'], type='http', auth="public", website=True, multilang=True)
130     def tags(self, forum, page=1, **post):
131         cr, uid, context = request.cr, request.uid, request.context
132         Tag = request.registry['forum.tag']
133         obj_ids = Tag.search(cr, uid, [('forum_id', '=', forum.id)], limit=None, context=context)
134         tags = Tag.browse(cr, uid, obj_ids, context=context)
135         values = self._prepare_forum_values(forum=forum, searches={'tags': True}, **post)
136         values.update({
137             'tags': tags,
138             'main_object': forum,
139         })
140         return request.website.render("website_forum.tag", values)
141
142     # Questions
143     # --------------------------------------------------
144
145     @http.route(['/forum/<model("forum.forum"):forum>/ask'], type='http', auth="public", website=True, multilang=True)
146     def question_ask(self, forum, **post):
147         if not request.session.uid:
148             return login_redirect()
149         values = self._prepare_forum_values(forum=forum, searches={},  header={'ask_hide': True})
150         return request.website.render("website_forum.ask_question", values)
151
152     @http.route('/forum/<model("forum.forum"):forum>/question/new', type='http', auth="user", multilang=True, methods=['POST'], website=True)
153     def question_create(self, forum, **post):
154         cr, uid, context = request.cr, request.uid, request.context
155         Tag = request.registry['forum.tag']
156         question_tag_ids = []
157         if post.get('question_tags').strip('[]'):
158             tags = post.get('question_tags').strip('[]').replace('"', '').split(",")
159             for tag in tags:
160                 tag_ids = Tag.search(cr, uid, [('name', '=', tag)], context=context)
161                 if tag_ids:
162                     question_tag_ids.append((4, tag_ids[0]))
163                 else:
164                     question_tag_ids.append((0, 0, {'name': tag, 'forum_id': forum.id}))
165
166         new_question_id = request.registry['forum.post'].create(
167             request.cr, request.uid, {
168                 'forum_id': forum.id,
169                 'name': post.get('question_name'),
170                 'content': post.get('content'),
171                 'tag_ids': question_tag_ids,
172             }, context=context)
173         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), new_question_id))
174
175     @http.route(['/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>'], type='http', auth="public", website=True, multilang=True)
176     def question(self, forum, question, **post):
177         cr, uid, context = request.cr, request.uid, request.context
178         # increment view counter
179         request.registry['forum.post'].set_viewed(cr, SUPERUSER_ID, [question.id], context=context)
180
181         filters = 'question'
182         values = self._prepare_forum_values(forum=forum, searches=post)
183         values.update({
184             'main_object': question,
185             'question': question,
186             'header': {'question_data': True},
187             'filters': filters,
188             'reversed': reversed,
189         })
190         return request.website.render("website_forum.post_description_full", values)
191
192     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/toggle_favourite', type='json', auth="user", multilang=True, methods=['POST'], website=True)
193     def question_toggle_favorite(self, forum, question, **post):
194         if not request.session.uid:
195             return {'error': 'anonymous_user'}
196         # TDE: add check for not public
197         favourite = False if question.user_favourite else True
198         if favourite:
199             favourite_ids = [(4, request.uid)]
200         else:
201             favourite_ids = [(3, request.uid)]
202         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'favourite_ids': favourite_ids}, context=request.context)
203         return favourite
204
205     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/ask_for_close', type='http', auth="user", multilang=True, website=True)
206     def question_ask_for_close(self, forum, question, **post):
207         cr, uid, context = request.cr, request.uid, request.context
208         Reason = request.registry['forum.post.reason']
209         reason_ids = Reason.search(cr, uid, [], context=context)
210         reasons = Reason.browse(cr, uid, reason_ids, context)
211
212         values = self._prepare_forum_values(**post)
213         values.update({
214             'post': question,
215             'question': question,
216             'forum': forum,
217             'reasons': reasons,
218         })
219         return request.website.render("website_forum.close_question", values)
220
221     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/edit_answer', type='http', auth="user", website=True, multilang=True)
222     def question_edit_answer(self, forum, question, **kwargs):
223         for record in question.child_ids:
224             if record.create_uid.id == request.uid:
225                 answer = record
226                 break
227         return werkzeug.utils.redirect("/forum/%s/post/%s/edit" % (slug(forum), slug(answer)))
228
229     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/close', type='http', auth="user", multilang=True, methods=['POST'], website=True)
230     def question_close(self, forum, question, **post):
231         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {
232             'state': 'close',
233             'closed_uid': request.uid,
234             'closed_date': datetime.today().strftime(tools.DEFAULT_SERVER_DATETIME_FORMAT),
235             'closed_reason_id': int(post.get('reason_id', False)),
236         }, context=request.context)
237         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
238
239     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/reopen', type='http', auth="user", multilang=True, website=True)
240     def question_reopen(self, forum, question, **kwarg):
241         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'state': 'active'}, context=request.context)
242         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
243
244     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/delete', type='http', auth="user", multilang=True, website=True)
245     def question_delete(self, forum, question, **kwarg):
246         #instead of unlink record just change 'active' to false so user can undelete it.
247         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': False}, context=request.context)
248         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
249
250     @http.route('/forum/<model("forum.forum"):forum>/question/<model("forum.post"):question>/undelete', type='http', auth="user", multilang=True, website=True)
251     def question_undelete(self, forum, question, **kwarg):
252         request.registry['forum.post'].write(request.cr, request.uid, [question.id], {'active': True}, context=request.context)
253         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
254
255     # Post
256     # --------------------------------------------------
257
258     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/new', type='http', auth="public", multilang=True, methods=['POST'], website=True)
259     def post_new(self, forum, post, **kwargs):
260         if not request.session.uid:
261             return login_redirect()
262         request.registry['forum.post'].create(
263             request.cr, request.uid, {
264                 'forum_id': forum.id,
265                 'parent_id': post.id,
266                 'content': kwargs.get('content'),
267             }, context=request.context)
268         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(post)))
269
270     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/comment', type='http', auth="public", methods=['POST'], website=True)
271     def post_comment(self, forum, post, **kwargs):
272         if not request.session.uid:
273             return login_redirect()
274         question = post.parent_id if post.parent_id else post
275         cr, uid, context = request.cr, request.uid, request.context
276         if kwargs.get('comment') and post.forum_id.id == forum.id:
277             # TDE FIXME: check that post_id is the question or one of its answers
278             if request.registry['res.users'].has_group(cr, uid, 'website_mail.group_comment'):
279                 request.registry['forum.post'].message_post(
280                     cr, uid, post.id,
281                     body=kwargs.get('comment'),
282                     type='comment',
283                     subtype='mt_comment',
284                     context=dict(context, mail_create_nosubcribe=True))
285         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
286
287     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/toggle_correct', type='json', auth="public", website=True)
288     def post_toggle_correct(self, forum, post, **kwargs):
289         cr, uid, context = request.cr, request.uid, request.context
290         if not request.session.uid:
291             return {'error': 'anonymous_user'}
292         # if user have not access to accept answer then reise warning
293         if post.parent_id is False or post.parent_id.create_uid.id != uid:
294             return {'error': 'own_post'}
295
296         # set all answers to False, only one can be accepted
297         request.registry['forum.post'].write(cr, uid, [c.id for c in post.parent_id.child_ids], {'is_correct': False}, context=context)
298         request.registry['forum.post'].write(cr, uid, [post.id, post.parent_id.id], {'is_correct': not post.is_correct}, context=context)
299         return not post.is_correct
300
301     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/delete', type='http', auth="user", multilang=True, website=True)
302     def post_delete(self, forum, post, **kwargs):
303         question = post.parent_id
304         request.registry['forum.post'].unlink(request.cr, request.uid, [post.id], context=request.context)
305         if question:
306             werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
307         return werkzeug.utils.redirect("/forum/%s" % slug(forum))
308
309     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/edit', type='http', auth="user", website=True, multilang=True)
310     def post_edit(self, forum, post, **kwargs):
311         tags = ""
312         for tag_name in post.tag_ids:
313             tags += tag_name.name + ","
314         values = self._prepare_forum_values(forum=forum)
315         values.update({
316             'tags': tags,
317             'post': post,
318             'is_answer': bool(post.parent_id),
319             'searches': kwargs
320         })
321         return request.website.render("website_forum.edit_post", values)
322
323     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/save', type='http', auth="user", multilang=True, methods=['POST'], website=True)
324     def post_save(self, forum, post, **kwargs):
325         cr, uid, context = request.cr, request.uid, request.context
326         question_tags = []
327         if kwargs.get('question_tag') and kwargs.get('question_tag').strip('[]'):
328             Tag = request.registry['forum.tag']
329             tags = kwargs.get('question_tag').strip('[]').replace('"', '').split(",")
330             for tag in tags:
331                 tag_ids = Tag.search(cr, uid, [('name', '=', tag)], context=context)
332                 if tag_ids:
333                     question_tags += tag_ids
334                 else:
335                     new_tag = Tag.create(cr, uid, {'name': tag, 'forum_id': forum.id}, context=context)
336                     question_tags.append(new_tag)
337         vals = {
338             'tag_ids': [(6, 0, question_tags)],
339             'name': kwargs.get('question_name'),
340             'content': kwargs.get('content'),
341         }
342         request.registry['forum.post'].write(cr, uid, [post.id], vals, context=context)
343         question = post.parent_id if post.parent_id else post
344         return werkzeug.utils.redirect("/forum/%s/question/%s" % (slug(forum), slug(question)))
345
346     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/upvote', type='json', auth="public", multilang=True, website=True)
347     def post_upvote(self, forum, post, **kwargs):
348         # check for karma and not self vote
349         if not request.session.uid:
350             return {'error': 'anonymous_user'}
351         if request.uid == post.create_uid.id:
352             return {'error': 'own_post'}
353         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
354         if user.karma <= 5:
355             return {'error': 'not_enough_karma', 'karma': 1}
356         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=True, context=request.context)
357
358     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/downvote', type='json', auth="public", multilang=True, website=True)
359     def post_downvote(self, forum, post, **kwargs):
360         if not request.session.uid:
361             return {'error': 'anonymous_user'}
362         if request.uid == post.create_uid.id:
363             return {'error': 'own_post'}
364         user = request.registry['res.users'].browse(request.cr, SUPERUSER_ID, request.uid, context=request.context)
365         if user.karma <= 50:
366             return {'error': 'not_enough_karma', 'karma': 50}
367         return request.registry['forum.post'].vote(request.cr, request.uid, [post.id], upvote=False, context=request.context)
368
369     # User
370     # --------------------------------------------------
371
372     @http.route('/forum/<model("forum.forum"):forum>/users', type='http', auth="public", website=True, multilang=True)
373     def users(self, forum, page=1, **searches):
374         cr, uid, context = request.cr, request.uid, request.context
375         User = request.registry['res.users']
376
377         step = 30
378         tag_count = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], count=True, context=context)
379         pager = request.website.pager(url="/forum/users", total=tag_count, page=page, step=step, scope=30)
380
381         obj_ids = User.search(cr, SUPERUSER_ID, [('karma', '>', 1)], limit=step, offset=pager['offset'], context=context)
382         users = User.browse(cr, SUPERUSER_ID, obj_ids, context=context)
383         searches['users'] = 'True'
384
385         values = self._prepare_forum_values(forum=forum, searches=searches)
386         values .update({
387             'users': users,
388             'main_object': forum,
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         values = self._prepare_forum_values(forum=forum, searches={'badges': True})
524         values.update({
525             'badge': badge,
526             'users': users,
527         })
528         return request.website.render("website_forum.badge_user", values)
529
530     # Messaging
531     # --------------------------------------------------
532
533     @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)
534     def convert_comment_to_answer(self, forum, post, comment, **kwarg):
535         values = {
536             'content': comment.body,
537         }
538         request.registry['mail.message'].unlink(request.cr, request.uid, [comment.id], context=request.context)
539         question = post.parent_id if post.parent_id else post
540         return self.post_new(forum, question, **values)
541
542     @http.route('/forum/<model("forum.forum"):forum>/post/<model("forum.post"):post>/convert_to_comment', type='http', auth="user", multilang=True, website=True)
543     def convert_answer_to_comment(self, forum, post, **kwarg):
544         values = {
545             'comment': html2plaintext(post.content),
546         }
547         question = post.parent_id
548         request.registry['forum.post'].unlink(request.cr, SUPERUSER_ID, [post.id], context=request.context)
549         return self.post_comment(forum, question, **values)
550
551     @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)
552     def delete_comment(self, forum, post, comment, **kwarg):
553         request.registry['mail.message'].unlink(request.cr, SUPERUSER_ID, [comment.id], context=request.context)
554         return True