profile.py 9.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308
  1. from django.contrib import messages
  2. from django.contrib.auth import get_user_model
  3. from django.core.urlresolvers import reverse
  4. from django.db.transaction import atomic
  5. from django.http import Http404, JsonResponse
  6. from django.shortcuts import redirect, render as django_render
  7. from django.utils.translation import ugettext as _
  8. from misago.acl import add_acl
  9. from misago.core.decorators import require_POST
  10. from misago.core.shortcuts import get_object_or_404, paginate, validate_slug
  11. from misago.core.utils import clean_return_path
  12. from misago.notifications import notify_user, read_user_notification
  13. from misago.users.bans import get_user_ban
  14. from misago.users.decorators import deny_guests
  15. from misago.users.online.utils import get_user_state
  16. from misago.users.permissions.profiles import (allow_follow_user,
  17. allow_block_user)
  18. from misago.users.sites import user_profile
  19. from misago.users.warnings import (get_warning_levels, get_user_warning_level,
  20. get_user_warning_obj)
  21. def profile_view(f):
  22. def decorator(request, *args, **kwargs):
  23. User = get_user_model()
  24. relations = ('rank', 'online_tracker', 'ban_cache')
  25. queryset = User.objects.select_related(*relations)
  26. profile = get_object_or_404(queryset, id=kwargs.pop('user_id'))
  27. validate_slug(profile, kwargs.pop('user_slug'))
  28. kwargs['profile'] = profile
  29. add_acl(request.user, profile)
  30. if profile.acl_['can_follow']:
  31. profile.is_followed = request.user.is_following(profile)
  32. else:
  33. profile.is_followed = False
  34. if profile.acl_['can_block'] and request.user.is_authenticated():
  35. profile.is_blocked = request.user.is_blocking(profile)
  36. else:
  37. profile.is_blocked = False
  38. if request.user.is_authenticated and request.method == "GET":
  39. read_user_notification(request.user, "profile_%s" % profile.pk)
  40. return f(request, *args, **kwargs)
  41. return decorator
  42. def profile_view_restricted_visibility(f):
  43. @profile_view
  44. def decorator(request, *args, **kwargs):
  45. pages = user_profile.get_pages(request, kwargs['profile'])
  46. for page in pages:
  47. if page['is_active']:
  48. return f(request, *args, **kwargs)
  49. else:
  50. # we are trying to display page thats not in nav
  51. raise Http404()
  52. return decorator
  53. def notification_view(trigger):
  54. def function_decorator(f):
  55. def decorator(*args, **kwargs):
  56. request = args[0]
  57. user = request.user
  58. profile = kwargs.get('profile')
  59. if user.is_authenticated and request.method == "GET":
  60. read_user_notification(user, trigger % profile.pk)
  61. return f(*args, **kwargs)
  62. return decorator
  63. return function_decorator
  64. def render(request, template, context):
  65. context['pages'] = user_profile.get_pages(request, context['profile'])
  66. for page in context['pages']:
  67. if page['is_active']:
  68. context['active_page'] = page
  69. break
  70. if request.user.is_authenticated():
  71. is_authenticated_user = context['profile'].pk == request.user.pk
  72. else:
  73. is_authenticated_user = False
  74. context['is_authenticated_user'] = is_authenticated_user
  75. user_acl = request.user.acl
  76. if request.user.is_authenticated():
  77. if is_authenticated_user:
  78. context['show_email'] = True
  79. else:
  80. context['show_email'] = user_acl['can_see_users_emails']
  81. else:
  82. context['show_email'] = False
  83. context['state'] = get_user_state(context['profile'], user_acl)
  84. return django_render(request, template, context)
  85. @profile_view
  86. def posts(request, profile, page=0):
  87. return render(request, 'misago/profile/posts.html', {'profile': profile})
  88. @profile_view
  89. def threads(request, profile, page=0):
  90. return render(request, 'misago/profile/threads.html', {'profile': profile})
  91. @profile_view
  92. def followers(request, profile, page=0):
  93. followers_qs = profile.followed_by.order_by('slug')
  94. followers = paginate(followers_qs, page, 6 * 4, 6)
  95. items_left = followers.paginator.count - followers.end_index()
  96. if followers.paginator.count != profile.followers:
  97. profile.followers = followers.paginator.count
  98. profile.save(update_fields=['followers'])
  99. return render(request, 'misago/profile/followers.html', {
  100. 'profile': profile,
  101. 'followers': followers,
  102. 'items_left': items_left,
  103. })
  104. @profile_view
  105. def follows(request, profile, page=0):
  106. followers_qs = profile.follows.order_by('slug')
  107. followers = paginate(followers_qs, page, 6 * 4, 6)
  108. items_left = followers.paginator.count - followers.end_index()
  109. if followers.paginator.count != profile.following:
  110. profile.following = followers.paginator.count
  111. profile.save(update_fields=['following'])
  112. return render(request, 'misago/profile/follows.html', {
  113. 'profile': profile,
  114. 'followers': followers,
  115. 'items_left': items_left,
  116. })
  117. @profile_view_restricted_visibility
  118. @notification_view("warnings_%s")
  119. def warnings(request, profile, page=0):
  120. warnings_qs = profile.warnings.order_by('-id')
  121. warnings = paginate(warnings_qs, page, 5, 2)
  122. items_left = warnings.paginator.count - warnings.end_index()
  123. add_acl(request.user, warnings.object_list)
  124. warning_level = get_user_warning_level(profile)
  125. warning_level_obj = get_user_warning_obj(profile)
  126. active_warnings = warning_level - warnings.start_index() + 1
  127. for warning in warnings.object_list:
  128. if warning.is_canceled:
  129. warning.is_active = False
  130. else:
  131. warning.is_active = active_warnings > 0
  132. active_warnings -= 1
  133. levels_total = len(get_warning_levels()) - 1
  134. if levels_total and warning_level:
  135. warning_progress = 100 - warning_level * 100 / levels_total
  136. else:
  137. warning_progress = 100
  138. if warning_level:
  139. warning_level_obj.level = warning_level
  140. return render(request, 'misago/profile/warnings.html', {
  141. 'profile': profile,
  142. 'warnings': warnings,
  143. 'warning_level': warning_level_obj,
  144. 'warning_progress': warning_progress,
  145. 'page_number': warnings.number,
  146. 'items_left': items_left
  147. })
  148. @profile_view_restricted_visibility
  149. @notification_view("name_history_%s")
  150. def name_history(request, profile, page=0):
  151. name_changes_qs = profile.namechanges.all().order_by('-id')
  152. name_changes = paginate(name_changes_qs, page, 12, 4)
  153. items_left = name_changes.paginator.count - name_changes.end_index()
  154. return render(request, 'misago/profile/name_history.html', {
  155. 'profile': profile,
  156. 'name_changes': name_changes,
  157. 'page_number': name_changes.number,
  158. 'items_left': items_left
  159. })
  160. @profile_view_restricted_visibility
  161. def user_ban(request, profile):
  162. ban = get_user_ban(profile)
  163. if not ban:
  164. raise Http404()
  165. return render(request, 'misago/profile/ban_details.html', {
  166. 'profile': profile,
  167. 'ban': ban
  168. })
  169. """
  170. Profile actions
  171. """
  172. def action_view(f):
  173. @deny_guests
  174. @require_POST
  175. @profile_view
  176. @atomic
  177. def decorator(request, profile):
  178. response = f(request, profile.lock())
  179. if request.is_ajax():
  180. response['is_error'] = False
  181. return JsonResponse(response)
  182. else:
  183. messages.success(request, response['message'])
  184. return_path = clean_return_path(request)
  185. if return_path:
  186. return redirect(return_path)
  187. else:
  188. return redirect(user_profile.get_default_link(),
  189. user_slug=profile.slug, user_id=profile.id)
  190. return decorator
  191. @action_view
  192. def follow_user(request, profile):
  193. user_locked = request.user.lock()
  194. profile.lock()
  195. if request.user.is_following(profile):
  196. request.user.follows.remove(profile)
  197. followed = False
  198. else:
  199. allow_follow_user(request.user, profile)
  200. followed = True
  201. request.user.follows.add(profile)
  202. if followed:
  203. message = _("You are now following %(user)s.")
  204. notify_user(profile,
  205. _("%(user)s is now following you."),
  206. reverse(user_profile.get_default_link(), kwargs={
  207. 'user_slug': user_locked.slug, 'user_id': user_locked.id
  208. }),
  209. "profile_%s" % user_locked.pk,
  210. formats={'user': user_locked.username},
  211. sender=user_locked,
  212. update_user=False)
  213. else:
  214. message = _("You have stopped following %(user)s.")
  215. message = message % {'user': profile.username}
  216. profile.followers = profile.followed_by.count()
  217. if followed:
  218. profile.save(update_fields=['followers', 'new_notifications'])
  219. else:
  220. profile.save(update_fields=['followers'])
  221. user_locked.following = user_locked.follows.count()
  222. user_locked.save(update_fields=['following'])
  223. if request.is_ajax:
  224. return {'is_following': followed, 'message': message}
  225. else:
  226. messages.success(request, message)
  227. @action_view
  228. def block_user(request, profile):
  229. request.user.lock()
  230. if request.user.is_blocking(profile):
  231. request.user.blocks.remove(profile)
  232. blocked = False
  233. else:
  234. allow_block_user(request.user, profile)
  235. blocked = True
  236. request.user.blocks.add(profile)
  237. if blocked:
  238. message = _("You are now blocking %(user)s.")
  239. else:
  240. message = _("You have stopped blocking %(user)s.")
  241. message = message % {'user': profile.username}
  242. if request.is_ajax:
  243. return {'is_blocking': blocked, 'message': message}
  244. else:
  245. messages.success(request, message)