Commit c9f9b728 by Saleem Latif Committed by GitHub

Merge pull request #14874 from edx/saleem-latif/ENT-306

ENT-306: Change Account Activation UI on Dashboard
parents 70e13407 0361b502
...@@ -3,10 +3,12 @@ from mock import patch ...@@ -3,10 +3,12 @@ from mock import patch
import unittest import unittest
from django.conf import settings from django.conf import settings
from django.contrib.auth.models import User
from django.test import TestCase, override_settings from django.test import TestCase, override_settings
from django.core.urlresolvers import reverse
from edxmako.shortcuts import render_to_string
from student.models import Registration from student.models import Registration
from student.tests.factories import UserFactory
@unittest.skipUnless(settings.ROOT_URLCONF == 'lms.urls', 'Test only valid in lms') @unittest.skipUnless(settings.ROOT_URLCONF == 'lms.urls', 'Test only valid in lms')
...@@ -17,13 +19,38 @@ class TestActivateAccount(TestCase): ...@@ -17,13 +19,38 @@ class TestActivateAccount(TestCase):
super(TestActivateAccount, self).setUp() super(TestActivateAccount, self).setUp()
self.username = "jack" self.username = "jack"
self.email = "jack@fake.edx.org" self.email = "jack@fake.edx.org"
self.user = User.objects.create(username=self.username, email=self.email, is_active=False) self.password = "test-password"
self.user = UserFactory.create(
username=self.username, email=self.email, password=self.password, is_active=False,
)
# Set Up Registration # Set Up Registration
self.registration = Registration() self.registration = Registration()
self.registration.register(self.user) self.registration.register(self.user)
self.registration.save() self.registration.save()
def login(self):
"""
Login with test user.
Since, only active users can login, so we must activate the user before login.
This method does the following tasks in order,
1. Stores user's active/in-active status in a variable.
2. Makes sure user account is active.
3. Authenticated user with the client.
4. Reverts user's original active/in-active status.
"""
is_active = self.user.is_active
# Make sure user is active before login
self.user.is_active = True
self.user.save()
self.client.login(username=self.username, password=self.password)
# Revert user activation status
self.user.is_active = is_active
self.user.save()
def assert_no_tracking(self, mock_segment_identify): def assert_no_tracking(self, mock_segment_identify):
""" Assert that activate sets the flag but does not call segment. """ """ Assert that activate sets the flag but does not call segment. """
# Ensure that the user starts inactive # Ensure that the user starts inactive
...@@ -76,3 +103,59 @@ class TestActivateAccount(TestCase): ...@@ -76,3 +103,59 @@ class TestActivateAccount(TestCase):
@patch('student.models.analytics.identify') @patch('student.models.analytics.identify')
def test_activation_without_keys(self, mock_segment_identify): def test_activation_without_keys(self, mock_segment_identify):
self.assert_no_tracking(mock_segment_identify) self.assert_no_tracking(mock_segment_identify)
@override_settings(FEATURES=dict(settings.FEATURES, DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR=True))
def test_account_activation_message(self):
"""
Verify that account correct activation message is displayed.
If logged in user has not activated his/her account, make sure that an
account activation message is displayed on dashboard sidebar.
"""
# Log in with test user.
self.login()
expected_message = render_to_string(
'registration/account_activation_sidebar_notice.html',
{'email': self.user.email}
)
response = self.client.get(reverse('dashboard'))
self.assertContains(response, expected_message, html=True)
# Now make sure account activation message goes away when user activated the account
self.user.is_active = True
self.user.save()
self.login()
expected_message = render_to_string(
'registration/account_activation_sidebar_notice.html',
{'email': self.user.email}
)
response = self.client.get(reverse('dashboard'))
self.assertNotContains(response, expected_message, html=True)
@override_settings(FEATURES=dict(settings.FEATURES, DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR=False))
def test_account_activation_message_disabled(self):
"""
Verify that old account activation message is displayed when
DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR is disabled.
"""
# Log in with test user.
self.login()
expected_message = render_to_string(
'registration/activate_account_notice.html',
{'email': self.user.email}
)
response = self.client.get(reverse('dashboard'))
self.assertContains(response, expected_message, html=True)
# Now make sure account activation message goes away when user activated the account
self.user.is_active = True
self.user.save()
self.login()
expected_message = render_to_string(
'registration/activate_account_notice.html',
{'email': self.user.email}
)
response = self.client.get(reverse('dashboard'))
self.assertNotContains(response, expected_message, html=True)
...@@ -685,9 +685,22 @@ def dashboard(request): ...@@ -685,9 +685,22 @@ def dashboard(request):
course_optouts = Optout.objects.filter(user=user).values_list('course_id', flat=True) course_optouts = Optout.objects.filter(user=user).values_list('course_id', flat=True)
message = "" sidebar_account_activation_message = ''
if not user.is_active: banner_account_activation_message = ''
message = render_to_string( display_account_activation_message_on_sidebar = configuration_helpers.get_value(
'DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR',
settings.FEATURES.get('DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR', False)
)
# Display activation message in sidebar if DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR
# flag is active. Otherwise display existing message at the top.
if display_account_activation_message_on_sidebar and not user.is_active:
sidebar_account_activation_message = render_to_string(
'registration/account_activation_sidebar_notice.html',
{'email': user.email}
)
elif not user.is_active:
banner_account_activation_message = render_to_string(
'registration/activate_account_notice.html', 'registration/activate_account_notice.html',
{'email': user.email, 'platform_name': platform_name} {'email': user.email, 'platform_name': platform_name}
) )
...@@ -819,7 +832,8 @@ def dashboard(request): ...@@ -819,7 +832,8 @@ def dashboard(request):
'redirect_message': redirect_message, 'redirect_message': redirect_message,
'course_enrollments': course_enrollments, 'course_enrollments': course_enrollments,
'course_optouts': course_optouts, 'course_optouts': course_optouts,
'message': message, 'banner_account_activation_message': banner_account_activation_message,
'sidebar_account_activation_message': sidebar_account_activation_message,
'staff_access': staff_access, 'staff_access': staff_access,
'errored_courses': errored_courses, 'errored_courses': errored_courses,
'show_courseware_links_for': show_courseware_links_for, 'show_courseware_links_for': show_courseware_links_for,
......
...@@ -377,6 +377,9 @@ FEATURES = { ...@@ -377,6 +377,9 @@ FEATURES = {
# Enable one click program purchase # Enable one click program purchase
# See LEARNER-493 # See LEARNER-493
'ENABLE_ONE_CLICK_PROGRAM_PURCHASE': False, 'ENABLE_ONE_CLICK_PROGRAM_PURCHASE': False,
# Whether to display account activation notification on dashboard.
'DISPLAY_ACCOUNT_ACTIVATION_MESSAGE_ON_SIDEBAR': False,
} }
# Ignore static asset files on import which match this pattern # Ignore static asset files on import which match this pattern
......
...@@ -1227,6 +1227,35 @@ ...@@ -1227,6 +1227,35 @@
border-bottom-color: $m-gray-l4; border-bottom-color: $m-gray-l4;
} }
} }
// Warning for status verification
&.warning {
border-top: 3px solid #ffc01f !important;
.status-title {
font-weight: 400 !important;
.fa {
color: #ffc01f;
}
}
.btn {
font-size: 16px;
line-height: 25.16px;
padding: 10px 10px;
border: 1px solid #0079bc;
color: #0079bc;
text-decoration: none;
display: block;
}
.btn:hover {
cursor: pointer;
color: #fff;
background-color: #0079bc;
}
}
} }
// status - verification // status - verification
......
...@@ -62,9 +62,10 @@ from openedx.core.djangolib.markup import HTML, Text ...@@ -62,9 +62,10 @@ from openedx.core.djangolib.markup import HTML, Text
</%block> </%block>
<div class="dashboard-notifications" tabindex="-1"> <div class="dashboard-notifications" tabindex="-1">
%if message:
%if banner_account_activation_message:
<div class="dashboard-banner"> <div class="dashboard-banner">
${message | n, decode.utf8} ${banner_account_activation_message | n, decode.utf8}
</div> </div>
%endif %endif
...@@ -137,6 +138,12 @@ from openedx.core.djangolib.markup import HTML, Text ...@@ -137,6 +138,12 @@ from openedx.core.djangolib.markup import HTML, Text
% endif % endif
</div> </div>
%if sidebar_account_activation_message:
<div class="sidebar-notification">
${sidebar_account_activation_message | n, decode.utf8}
</div>
%endif
% if settings.FEATURES.get('ENABLE_DASHBOARD_SEARCH'): % if settings.FEATURES.get('ENABLE_DASHBOARD_SEARCH'):
<div id="dashboard-search-bar" class="search-bar dashboard-search-bar" role="search" aria-label="Dashboard"> <div id="dashboard-search-bar" class="search-bar dashboard-search-bar" role="search" aria-label="Dashboard">
<form> <form>
......
<%page expression_filter="h"/>
<%!
from django.utils.translation import ugettext as _
from openedx.core.djangolib.markup import HTML, Text
%>
<div class="profile-sidebar" role="region" aria-label="Account Activation Info">
<header class="profile">
<h2 class="account-activation-title sr">${_("Account Activation Info")}: </h2>
</header>
<div class="user-info">
<ul>
<li class="status status-verification warning" role="alert" aria-labelledby="title status-title" tabindex="1">
<span class="title status-title">
<i class="fa fa-exclamation-circle" aria-hidden="true"></i>
${_("Activate your account!")}
</span>
<p class="status-note">${Text(_(
"Check your {email_start}{email}{email_end} inbox for an account activation link from edX. "
"If you need help, contact {link_start}edX Support{link_end}."
)).format(
email_start=HTML("<strong>"),
email_end=HTML("</strong>"),
email=email,
link_start=HTML("<a target='_blank' href='https://support.edx.org/hc/en-us/articles/227340127-Why-haven-t-I-received-my-activation-email-'>"),
link_end=HTML("</a>"),
)}
</p>
## TODO: Add resend activation email functionality.
## TODO: Implementation of this is part of ENT-353.
## <a class="btn btn-neutral"><i class="fa fa-envelope-o"></i> Resend Activation Email </a>
</li>
</ul>
</div>
</div>
...@@ -4,25 +4,24 @@ from django.utils.translation import ugettext as _ ...@@ -4,25 +4,24 @@ from django.utils.translation import ugettext as _
from openedx.core.djangolib.markup import HTML, Text from openedx.core.djangolib.markup import HTML, Text
%> %>
<div class="wrapper-msg urgency-high"> <div class="wrapper-msg urgency-high">
<div class="msg"> <div class="msg">
<div class="msg-content"> <div class="msg-content">
<h2 class="title">${_("You're almost there!")}</h2> <h2 class="title">${_("You're almost there!")}</h2>
<div class="copy"> <div class="copy">
<p class='activation-message'>${Text(_( <p class='activation-message'>${Text(_(
"There's just one more step: Before you " "There's just one more step: Before you "
"enroll in a course, you need to activate " "enroll in a course, you need to activate "
"your account. We've sent an email message to " "your account. We've sent an email message to "
"{email_start}{email}{email_end} with " "{email_start}{email}{email_end} with "
"instructions for activating your account. If " "instructions for activating your account. If "
"you don't receive this message, check your " "you don't receive this message, check your "
"spam folder." "spam folder."
)).format(email_start=HTML("<strong>"), )).format(email_start=HTML("<strong>"),
email_end=HTML("</strong>"), email_end=HTML("</strong>"),
email=email, email=email,
platform_name=platform_name )}
)} </p>
</p> </div>
</div>
</div>
</div> </div>
</div> </div>
</div>
\ No newline at end of file
...@@ -64,10 +64,11 @@ from openedx.core.djangoapps.theming import helpers as theming_helpers ...@@ -64,10 +64,11 @@ from openedx.core.djangoapps.theming import helpers as theming_helpers
</%block> </%block>
<div class="dashboard-notifications" tabindex="-1"> <div class="dashboard-notifications" tabindex="-1">
%if message:
<section class="dashboard-banner"> %if banner_account_activation_message:
${message | n, decode.utf8} <div class="dashboard-banner">
</section> ${banner_account_activation_message | n, decode.utf8}
</div>
%endif %endif
%if enrollment_message: %if enrollment_message:
...@@ -142,6 +143,12 @@ from openedx.core.djangoapps.theming import helpers as theming_helpers ...@@ -142,6 +143,12 @@ from openedx.core.djangoapps.theming import helpers as theming_helpers
</section> </section>
</main> </main>
%if sidebar_account_activation_message:
<div class="sidebar-notification">
${sidebar_account_activation_message | n, decode.utf8}
</div>
%endif
% if settings.FEATURES.get('ENABLE_DASHBOARD_SEARCH'): % if settings.FEATURES.get('ENABLE_DASHBOARD_SEARCH'):
<div id="dashboard-search-bar" class="search-bar dashboard-search-bar" role="search" aria-label="Dashboard"> <div id="dashboard-search-bar" class="search-bar dashboard-search-bar" role="search" aria-label="Dashboard">
<form> <form>
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment