Files
website-v2/users/views.py
2023-08-08 12:19:46 -07:00

177 lines
6.3 KiB
Python

from django.contrib import messages
from django.contrib.auth.mixins import LoginRequiredMixin
from django.contrib.messages.views import SuccessMessageMixin
from django.http import HttpResponseRedirect
from django.urls import reverse_lazy
from django.views.generic import DetailView
from django.views.generic.base import TemplateView
from allauth.account.forms import ChangePasswordForm
from allauth.socialaccount.models import SocialAccount
from rest_framework import generics
from rest_framework import viewsets
from rest_framework.permissions import IsAuthenticated
from .forms import PreferencesForm, UserProfileForm, UserProfilePhotoForm
from .models import User
from .permissions import CustomUserPermissions
from .serializers import UserSerializer, FullUserSerializer, CurrentUserSerializer
from . import tasks
class UserViewSet(viewsets.ModelViewSet):
"""
Main User API ViewSet
"""
queryset = User.objects.all()
permission_classes = [CustomUserPermissions]
def get_serializer_class(self):
"""Pick the right serializer based on the user"""
if self.request.user.is_staff or self.request.user.is_superuser:
return FullUserSerializer
else:
return UserSerializer
class CurrentUserAPIView(generics.RetrieveUpdateAPIView):
"""
This gives the current user a convenient way to retrieve or
update slightly more detailed information about themselves.
Typically set to a route of `/api/v1/users/me`
"""
serializer_class = CurrentUserSerializer
permission_classes = [IsAuthenticated]
def get_object(self):
return self.request.user
class ProfileView(DetailView):
"""
ViewSet to show statistics about a user to include
stats, badges, reviews, etc.
"""
model = User
queryset = User.objects.all()
template_name = "users/profile.html"
context_object_name = "user"
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
user = self.get_object()
context["authored"] = user.authors.all()
context["maintained"] = user.maintainers.all().distinct()
return context
class CurrentUserProfileView(LoginRequiredMixin, SuccessMessageMixin, TemplateView):
template_name = "users/profile.html"
success_message = "Your profile was successfully updated."
success_url = reverse_lazy("profile-account")
def get_context_data(self, **kwargs):
context = super().get_context_data(**kwargs)
context["change_password_form"] = ChangePasswordForm(user=self.request.user)
context["profile_form"] = UserProfileForm(instance=self.request.user)
context["profile_photo_form"] = UserProfilePhotoForm(instance=self.request.user)
context["profile_preferences_form"] = PreferencesForm(
instance=self.request.user.preferences
)
context["social_accounts"] = self.get_social_accounts()
return context
def get_social_accounts(self):
account_data = []
for account in SocialAccount.objects.filter(user=self.request.user):
provider_account = account.get_provider_account()
account_data.append(
{
"id": account.pk,
"provider": account.provider,
"name": provider_account.to_str(),
}
)
return account_data
def post(self, request, *args, **kwargs):
"""
Process each form submission individually if present
"""
if "change_password" in request.POST:
change_password_form = ChangePasswordForm(
data=request.POST, user=self.request.user
)
self.change_password(change_password_form, request)
if "update_profile" in request.POST:
profile_form = UserProfileForm(
self.request.POST, instance=self.request.user
)
self.update_profile(profile_form, request)
if "update_photo" in request.POST:
profile_photo_form = UserProfilePhotoForm(
self.request.POST, self.request.FILES, instance=self.request.user
)
self.update_photo(profile_photo_form, request)
if "update_github_photo" in request.POST:
self.update_github_photo(request)
if "update_preferences" in request.POST:
profile_preferences_form = PreferencesForm(
self.request.POST, instance=request.user.preferences
)
self.update_preferences(profile_preferences_form, request)
return HttpResponseRedirect(self.success_url)
def change_password(self, form, request):
"""Change the password of the user."""
if form.is_valid():
self.object = request.user
self.object.set_password(form.cleaned_data["password1"])
self.object.save()
messages.success(request, "Your password was successfully updated.")
else:
for error in form.errors.values():
messages.error(request, f"{error}")
def update_photo(self, form, request):
"""Update the profile photo of the user."""
if form.is_valid():
form.save()
messages.success(request, "Your profile photo was successfully updated.")
else:
for error in form.errors.values():
messages.error(request, f"{error}")
def update_github_photo(self, request):
"""Update the GitHub photo of the user."""
tasks.update_user_github_photo(str(request.user.pk))
messages.success(request, "Your GitHub photo has been retrieved.")
def update_preferences(self, form, request):
"""Update the preferences of the user."""
if form.is_valid():
form.save()
messages.success(request, "Your preferences were successfully updated.")
else:
for error in form.errors.values():
messages.error(request, f"{error}")
def update_profile(self, form, request):
"""Update the profile of the user."""
if form.is_valid():
form.save()
messages.success(request, "Your profile was successfully updated.")
else:
for error in form.errors.values():
messages.error(request, f"{error}")