Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
E
edx-platform
Overview
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
edx
edx-platform
Commits
2ac544ca
Commit
2ac544ca
authored
Nov 29, 2017
by
zubair-arbi
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Update account settings page for enterprise learners
ENT-762
parent
17c889b7
Expand all
Hide whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
476 additions
and
21 deletions
+476
-21
common/test/acceptance/tests/lms/test_account_settings.py
+2
-2
lms/djangoapps/student_account/test/test_views.py
+57
-1
lms/djangoapps/student_account/views.py
+17
-1
lms/envs/common.py
+6
-0
lms/static/js/spec/student_account/account_settings_factory_spec.js
+179
-0
lms/static/js/spec/student_account/account_settings_view_spec.js
+3
-0
lms/static/js/spec/student_account/helpers.js
+46
-1
lms/static/js/spec/views/fields_helpers.js
+36
-0
lms/static/js/student_account/views/account_section_view.js
+3
-1
lms/static/js/student_account/views/account_settings_factory.js
+0
-0
lms/static/sass/views/_account-settings.scss
+104
-12
lms/templates/student_account/account_settings.html
+12
-2
lms/templates/student_account/account_settings_section.underscore
+11
-1
No files found.
common/test/acceptance/tests/lms/test_account_settings.py
View file @
2ac544ca
...
...
@@ -163,7 +163,7 @@ class AccountSettingsPageTest(AccountSettingsTestMixin, AcceptanceTest):
'fields'
:
[
'Username'
,
'Full Name'
,
'Email Address'
,
'Email Address
(Sign In)
'
,
'Password'
,
'Language'
,
'Country or Region of Residence'
,
...
...
@@ -289,7 +289,7 @@ class AccountSettingsPageTest(AccountSettingsTestMixin, AcceptanceTest):
self
.
visit_account_settings_page
()
self
.
_test_text_field
(
u'email'
,
u'Email Address'
,
u'Email Address
(Sign In)
'
,
email
,
u'test@example.com'
+
XSS_INJECTION
,
[
u'me@here.com'
,
u'you@there.com'
],
...
...
lms/djangoapps/student_account/test/test_views.py
View file @
2ac544ca
...
...
@@ -35,6 +35,7 @@ from lms.djangoapps.commerce.tests import factories
from
lms.djangoapps.commerce.tests.mocks
import
mock_get_orders
from
openedx.core.djangoapps.oauth_dispatch.tests
import
factories
as
dot_factories
from
openedx.core.djangoapps.programs.tests.mixins
import
ProgramsApiConfigMixin
from
openedx.core.djangoapps.site_configuration.tests.factories
import
SiteFactory
from
openedx.core.djangoapps.site_configuration.tests.mixins
import
SiteMixin
from
openedx.core.djangoapps.theming.tests.test_util
import
with_comprehensive_theme_context
from
openedx.core.djangoapps.user_api.accounts.api
import
activate_account
,
create_account
...
...
@@ -727,8 +728,54 @@ class AccountSettingsViewTest(ThirdPartyAuthTestMixin, TestCase, ProgramsApiConf
MessageMiddleware
()
.
process_request
(
self
.
request
)
messages
.
error
(
self
.
request
,
'Facebook is already in use.'
,
extra_tags
=
'Auth facebook'
)
def
test_context
(
self
):
@mock.patch
(
'student_account.views.get_enterprise_learner_data'
)
def
test_context
(
self
,
mock_get_enterprise_learner_data
):
self
.
request
.
site
=
SiteFactory
.
create
()
mock_get_enterprise_learner_data
.
return_value
=
[]
context
=
account_settings_context
(
self
.
request
)
user_accounts_api_url
=
reverse
(
"accounts_api"
,
kwargs
=
{
'username'
:
self
.
user
.
username
})
self
.
assertEqual
(
context
[
'user_accounts_api_url'
],
user_accounts_api_url
)
user_preferences_api_url
=
reverse
(
'preferences_api'
,
kwargs
=
{
'username'
:
self
.
user
.
username
})
self
.
assertEqual
(
context
[
'user_preferences_api_url'
],
user_preferences_api_url
)
for
attribute
in
self
.
FIELDS
:
self
.
assertIn
(
attribute
,
context
[
'fields'
])
self
.
assertEqual
(
context
[
'user_accounts_api_url'
],
reverse
(
"accounts_api"
,
kwargs
=
{
'username'
:
self
.
user
.
username
})
)
self
.
assertEqual
(
context
[
'user_preferences_api_url'
],
reverse
(
'preferences_api'
,
kwargs
=
{
'username'
:
self
.
user
.
username
})
)
self
.
assertEqual
(
context
[
'duplicate_provider'
],
'facebook'
)
self
.
assertEqual
(
context
[
'auth'
][
'providers'
][
0
][
'name'
],
'Facebook'
)
self
.
assertEqual
(
context
[
'auth'
][
'providers'
][
1
][
'name'
],
'Google'
)
self
.
assertEqual
(
context
[
'sync_learner_profile_data'
],
False
)
self
.
assertEqual
(
context
[
'edx_support_url'
],
settings
.
SUPPORT_SITE_LINK
)
self
.
assertEqual
(
context
[
'enterprise_name'
],
None
)
self
.
assertEqual
(
context
[
'enterprise_readonly_account_fields'
],
{
'fields'
:
settings
.
ENTERPRISE_READONLY_ACCOUNT_FIELDS
}
)
@mock.patch
(
'student_account.views.get_enterprise_learner_data'
)
@mock.patch
(
'student_account.views.third_party_auth.provider.Registry.get'
)
def
test_context_for_enterprise_learner
(
self
,
mock_get_auth_provider
,
mock_get_enterprise_learner_data
):
dummy_enterprise_customer
=
{
'uuid'
:
'real-ent-uuid'
,
'name'
:
'Dummy Enterprise'
,
'identity_provider'
:
'saml-ubc'
}
mock_get_enterprise_learner_data
.
return_value
=
[
{
'enterprise_customer'
:
dummy_enterprise_customer
}
]
self
.
request
.
site
=
SiteFactory
.
create
()
mock_get_auth_provider
.
return_value
.
sync_learner_profile_data
=
True
context
=
account_settings_context
(
self
.
request
)
user_accounts_api_url
=
reverse
(
"accounts_api"
,
kwargs
=
{
'username'
:
self
.
user
.
username
})
...
...
@@ -751,6 +798,15 @@ class AccountSettingsViewTest(ThirdPartyAuthTestMixin, TestCase, ProgramsApiConf
self
.
assertEqual
(
context
[
'auth'
][
'providers'
][
0
][
'name'
],
'Facebook'
)
self
.
assertEqual
(
context
[
'auth'
][
'providers'
][
1
][
'name'
],
'Google'
)
self
.
assertEqual
(
context
[
'sync_learner_profile_data'
],
mock_get_auth_provider
.
return_value
.
sync_learner_profile_data
)
self
.
assertEqual
(
context
[
'edx_support_url'
],
settings
.
SUPPORT_SITE_LINK
)
self
.
assertEqual
(
context
[
'enterprise_name'
],
dummy_enterprise_customer
[
'name'
])
self
.
assertEqual
(
context
[
'enterprise_readonly_account_fields'
],
{
'fields'
:
settings
.
ENTERPRISE_READONLY_ACCOUNT_FIELDS
}
)
def
test_view
(
self
):
"""
Test that all fields are visible
...
...
lms/djangoapps/student_account/views.py
View file @
2ac544ca
...
...
@@ -40,7 +40,7 @@ from openedx.core.djangoapps.user_api.errors import (
)
from
openedx.core.lib.edx_api_utils
import
get_edx_api_data
from
openedx.core.lib.time_zone_utils
import
TIME_ZONE_CHOICES
from
openedx.features.enterprise_support.api
import
enterprise_customer_for_request
from
openedx.features.enterprise_support.api
import
enterprise_customer_for_request
,
get_enterprise_learner_data
from
student.helpers
import
destroy_oauth_tokens
,
get_next_url_for_login_page
from
student.models
import
UserProfile
from
student.views
import
register_user
as
old_register_view
...
...
@@ -567,6 +567,22 @@ def account_settings_context(request):
'order_history'
:
user_orders
}
enterprise_customer_name
=
None
sync_learner_profile_data
=
False
enterprise_learner_data
=
get_enterprise_learner_data
(
site
=
request
.
site
,
user
=
request
.
user
)
if
enterprise_learner_data
:
enterprise_customer_name
=
enterprise_learner_data
[
0
][
'enterprise_customer'
][
'name'
]
enterprise_idp
=
enterprise_learner_data
[
0
][
'enterprise_customer'
][
'identity_provider'
]
identity_provider
=
third_party_auth
.
provider
.
Registry
.
get
(
provider_id
=
enterprise_idp
)
sync_learner_profile_data
=
identity_provider
.
sync_learner_profile_data
if
identity_provider
else
False
context
[
'sync_learner_profile_data'
]
=
sync_learner_profile_data
context
[
'edx_support_url'
]
=
configuration_helpers
.
get_value
(
'SUPPORT_SITE_LINK'
,
settings
.
SUPPORT_SITE_LINK
)
context
[
'enterprise_name'
]
=
enterprise_customer_name
context
[
'enterprise_readonly_account_fields'
]
=
{
'fields'
:
settings
.
ENTERPRISE_READONLY_ACCOUNT_FIELDS
}
if
third_party_auth
.
is_enabled
():
# If the account on the third party provider is already connected with another edX account,
# we display a message to the user.
...
...
lms/envs/common.py
View file @
2ac544ca
...
...
@@ -3395,6 +3395,12 @@ ENTERPRISE_EXCLUDED_REGISTRATION_FIELDS = {
'year_of_birth'
,
'mailing_address'
,
}
ENTERPRISE_READONLY_ACCOUNT_FIELDS
=
[
'username'
,
'name'
,
'email'
,
'country'
,
]
ENTERPRISE_CUSTOMER_COOKIE_NAME
=
'enterprise_customer_uuid'
BASE_COOKIE_DOMAIN
=
'localhost'
...
...
lms/static/js/spec/student_account/account_settings_factory_spec.js
View file @
2ac544ca
...
...
@@ -148,4 +148,183 @@ define(['backbone',
});
});
});
describe
(
'edx.user.AccountSettingsFactory'
,
function
()
{
var
createEnterpriseLearnerAccountSettingsPage
=
function
()
{
var
context
=
AccountSettingsPage
(
Helpers
.
FIELDS_DATA
,
[],
Helpers
.
AUTH_DATA
,
Helpers
.
PASSWORD_RESET_SUPPORT_LINK
,
Helpers
.
USER_ACCOUNTS_API_URL
,
Helpers
.
USER_PREFERENCES_API_URL
,
1
,
Helpers
.
PLATFORM_NAME
,
Helpers
.
CONTACT_EMAIL
,
true
,
''
,
Helpers
.
SYNC_LEARNER_PROFILE_DATA
,
Helpers
.
ENTERPRISE_NAME
,
Helpers
.
ENTERPRISE_READ_ONLY_ACCOUNT_FIELDS
,
Helpers
.
EDX_SUPPORT_URL
);
return
context
.
accountSettingsView
;
};
var
requests
;
var
accountInfoTab
=
{
BASIC_ACCOUNT_INFORMATION
:
0
,
ADDITIONAL_INFORMATION
:
1
};
var
basicAccountInfoFields
=
{
USERNAME
:
0
,
FULL_NAME
:
1
,
EMAIL_ADDRESS
:
2
,
PASSWORD
:
3
,
LANGUAGE
:
4
,
COUNTRY
:
5
,
TIMEZONE
:
6
};
var
additionalInfoFields
=
{
EDUCATION
:
0
,
GENDER
:
1
,
YEAR_OF_BIRTH
:
2
,
PREFERRED_LANGUAGE
:
3
};
beforeEach
(
function
()
{
setFixtures
(
'<div class="wrapper-account-settings"></div>'
);
});
it
(
'shows loading error when UserAccountModel fails to load for enterprise learners'
,
function
()
{
var
accountSettingsView
,
request
;
requests
=
AjaxHelpers
.
requests
(
this
);
accountSettingsView
=
createEnterpriseLearnerAccountSettingsPage
();
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
false
);
request
=
requests
[
0
];
expect
(
request
.
method
).
toBe
(
'GET'
);
expect
(
request
.
url
).
toBe
(
Helpers
.
USER_ACCOUNTS_API_URL
);
AjaxHelpers
.
respondWithError
(
requests
,
500
);
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
true
);
});
it
(
'shows loading error when UserPreferencesModel fails to load for enterprise learners'
,
function
()
{
var
accountSettingsView
,
request
;
requests
=
AjaxHelpers
.
requests
(
this
);
accountSettingsView
=
createEnterpriseLearnerAccountSettingsPage
();
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
false
);
request
=
requests
[
0
];
expect
(
request
.
method
).
toBe
(
'GET'
);
expect
(
request
.
url
).
toBe
(
Helpers
.
USER_ACCOUNTS_API_URL
);
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
createAccountSettingsData
());
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
false
);
request
=
requests
[
1
];
expect
(
request
.
method
).
toBe
(
'GET'
);
expect
(
request
.
url
).
toBe
(
'/user_api/v1/preferences/time_zones/?country_code=1'
);
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
TIME_ZONE_RESPONSE
);
request
=
requests
[
2
];
expect
(
request
.
method
).
toBe
(
'GET'
);
expect
(
request
.
url
).
toBe
(
Helpers
.
USER_PREFERENCES_API_URL
);
AjaxHelpers
.
respondWithError
(
requests
,
500
);
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
true
);
});
it
(
'renders fields after the models are successfully fetched for enterprise learners'
,
function
()
{
var
accountSettingsView
;
requests
=
AjaxHelpers
.
requests
(
this
);
accountSettingsView
=
createEnterpriseLearnerAccountSettingsPage
();
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
false
);
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
createAccountSettingsData
());
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
TIME_ZONE_RESPONSE
);
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
createUserPreferencesData
());
accountSettingsView
.
render
();
Helpers
.
expectLoadingErrorIsVisible
(
accountSettingsView
,
false
);
Helpers
.
expectSettingsSectionsAndFieldsToBeRenderedWithMessage
(
accountSettingsView
);
});
it
(
'expects all fields to behave correctly for enterprise learners'
,
function
()
{
var
accountSettingsView
,
i
,
view
,
sectionsData
,
textFields
,
dropdownFields
;
requests
=
AjaxHelpers
.
requests
(
this
);
accountSettingsView
=
createEnterpriseLearnerAccountSettingsPage
();
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
createAccountSettingsData
());
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
TIME_ZONE_RESPONSE
);
AjaxHelpers
.
respondWithJson
(
requests
,
Helpers
.
createUserPreferencesData
());
AjaxHelpers
.
respondWithJson
(
requests
,
{});
// Page viewed analytics event
sectionsData
=
accountSettingsView
.
options
.
tabSections
.
aboutTabSections
;
expect
(
sectionsData
[
accountInfoTab
.
BASIC_ACCOUNT_INFORMATION
].
fields
.
length
).
toBe
(
7
);
// Verify that username, name and email fields are readonly
textFields
=
[
sectionsData
[
accountInfoTab
.
BASIC_ACCOUNT_INFORMATION
].
fields
[
basicAccountInfoFields
.
USERNAME
],
sectionsData
[
accountInfoTab
.
BASIC_ACCOUNT_INFORMATION
].
fields
[
basicAccountInfoFields
.
FULL_NAME
],
sectionsData
[
accountInfoTab
.
BASIC_ACCOUNT_INFORMATION
].
fields
[
basicAccountInfoFields
.
EMAIL_ADDRESS
]
];
for
(
i
=
0
;
i
<
textFields
.
length
;
i
++
)
{
view
=
textFields
[
i
].
view
;
FieldViewsSpecHelpers
.
verifyReadonlyTextField
(
view
,
{
title
:
view
.
options
.
title
,
valueAttribute
:
view
.
options
.
valueAttribute
,
helpMessage
:
view
.
options
.
helpMessage
,
validValue
:
'My Name'
,
defaultValue
:
''
},
requests
);
}
// Verify un-editable country dropdown field
view
=
sectionsData
[
accountInfoTab
.
BASIC_ACCOUNT_INFORMATION
].
fields
[
basicAccountInfoFields
.
COUNTRY
].
view
;
FieldViewsSpecHelpers
.
verifyReadonlyDropDownField
(
view
,
{
title
:
view
.
options
.
title
,
valueAttribute
:
view
.
options
.
valueAttribute
,
helpMessage
:
''
,
validValue
:
Helpers
.
FIELD_OPTIONS
[
1
][
0
],
editable
:
'never'
,
defaultValue
:
null
});
expect
(
sectionsData
[
accountInfoTab
.
ADDITIONAL_INFORMATION
].
fields
.
length
).
toBe
(
4
);
dropdownFields
=
[
sectionsData
[
accountInfoTab
.
ADDITIONAL_INFORMATION
].
fields
[
additionalInfoFields
.
EDUCATION
],
sectionsData
[
accountInfoTab
.
ADDITIONAL_INFORMATION
].
fields
[
additionalInfoFields
.
GENDER
],
sectionsData
[
accountInfoTab
.
ADDITIONAL_INFORMATION
].
fields
[
additionalInfoFields
.
YEAR_OF_BIRTH
]
];
_
.
each
(
dropdownFields
,
function
(
field
)
{
view
=
field
.
view
;
FieldViewsSpecHelpers
.
verifyDropDownField
(
view
,
{
title
:
view
.
options
.
title
,
valueAttribute
:
view
.
options
.
valueAttribute
,
helpMessage
:
''
,
validValue
:
Helpers
.
FIELD_OPTIONS
[
1
][
0
],
// dummy option for dropdown field
invalidValue1
:
Helpers
.
FIELD_OPTIONS
[
2
][
0
],
// dummy option for dropdown field
invalidValue2
:
Helpers
.
FIELD_OPTIONS
[
3
][
0
],
// dummy option for dropdown field
validationError
:
'Nope, this will not do!'
,
defaultValue
:
null
},
requests
);
});
});
});
});
lms/static/js/spec/student_account/account_settings_view_spec.js
View file @
2ac544ca
...
...
@@ -20,6 +20,9 @@ define(['backbone',
var
aboutSectionsData
=
[
{
title
:
'Basic Account Information'
,
messageType
:
'info'
,
message
:
'Your profile settings are managed by Test Enterprise. '
+
'Contact your administrator or <a href="https://support.edx.org/">edX Support</a> for help.'
,
fields
:
[
{
view
:
new
FieldViews
.
ReadonlyFieldView
({
...
...
lms/static/js/spec/student_account/helpers.js
View file @
2ac544ca
...
...
@@ -10,6 +10,14 @@ define(['underscore'], function(_) {
var
PASSWORD_RESET_SUPPORT_LINK
=
'https://support.edx.org/hc/en-us/articles/206212088-What-if-I-did-not-receive-a-password-reset-message-'
;
// eslint-disable-line max-len
var
PLATFORM_NAME
=
'edX'
;
var
CONTACT_EMAIL
=
'info@example.com'
;
var
SYNC_LEARNER_PROFILE_DATA
=
true
;
var
ENTERPRISE_NAME
=
'Test Enterprise'
;
var
ENTERPRISE_READ_ONLY_ACCOUNT_FIELDS
=
{
fields
:
[
'username'
,
'name'
,
'email'
,
'country'
]
};
var
EDX_SUPPORT_URL
=
'https://support.edx.org/'
;
var
PROFILE_IMAGE
=
{
image_url_large
:
'/media/profile-images/image.jpg'
,
has_image
:
true
...
...
@@ -167,6 +175,36 @@ define(['underscore'], function(_) {
});
};
var
expectSettingsSectionsAndFieldsToBeRenderedWithMessage
=
function
(
accountSettingsView
,
fieldsAreRendered
)
{
var
sectionFieldElements
;
var
sectionsData
=
accountSettingsView
.
options
.
tabSections
.
aboutTabSections
;
var
sectionElements
=
accountSettingsView
.
$
(
'#aboutTabSections-tabpanel .section'
);
expect
(
sectionElements
.
length
).
toBe
(
sectionsData
.
length
);
_
.
each
(
sectionElements
,
function
(
sectionElement
,
sectionIndex
)
{
expect
(
$
(
sectionElement
).
find
(
'.section-header'
).
text
()
.
trim
()).
toBe
(
sectionsData
[
sectionIndex
].
title
);
if
(
!
_
.
isUndefined
(
sectionsData
[
sectionIndex
].
message
))
{
expect
(
$
(
sectionElement
).
find
(
'.account-settings-section-message span'
).
html
()
.
trim
()).
toBe
(
String
(
sectionsData
[
sectionIndex
].
message
));
}
sectionFieldElements
=
$
(
sectionElement
).
find
(
'.u-field'
);
if
(
fieldsAreRendered
===
false
)
{
expect
(
sectionFieldElements
.
length
).
toBe
(
0
);
}
else
{
expect
(
sectionFieldElements
.
length
).
toBe
(
sectionsData
[
sectionIndex
].
fields
.
length
);
_
.
each
(
sectionFieldElements
,
function
(
sectionFieldElement
,
fieldIndex
)
{
expectElementContainsField
(
sectionFieldElement
,
sectionsData
[
sectionIndex
].
fields
[
fieldIndex
]);
});
}
});
};
var
expectSettingsSectionsButNotFieldsToBeRendered
=
function
(
accountSettingsView
)
{
expectSettingsSectionsAndFieldsToBeRendered
(
accountSettingsView
,
false
);
};
...
...
@@ -181,6 +219,12 @@ define(['underscore'], function(_) {
PASSWORD_RESET_SUPPORT_LINK
:
PASSWORD_RESET_SUPPORT_LINK
,
PLATFORM_NAME
:
PLATFORM_NAME
,
CONTACT_EMAIL
:
CONTACT_EMAIL
,
SYNC_LEARNER_PROFILE_DATA
:
SYNC_LEARNER_PROFILE_DATA
,
ENTERPRISE_NAME
:
ENTERPRISE_NAME
,
ENTERPRISE_READ_ONLY_ACCOUNT_FIELDS
:
ENTERPRISE_READ_ONLY_ACCOUNT_FIELDS
,
EDX_SUPPORT_URL
:
EDX_SUPPORT_URL
,
PROFILE_IMAGE
:
PROFILE_IMAGE
,
FIELD_OPTIONS
:
FIELD_OPTIONS
,
TIME_ZONE_RESPONSE
:
TIME_ZONE_RESPONSE
,
...
...
@@ -195,6 +239,7 @@ define(['underscore'], function(_) {
expectLoadingErrorIsVisible
:
expectLoadingErrorIsVisible
,
expectElementContainsField
:
expectElementContainsField
,
expectSettingsSectionsButNotFieldsToBeRendered
:
expectSettingsSectionsButNotFieldsToBeRendered
,
expectSettingsSectionsAndFieldsToBeRendered
:
expectSettingsSectionsAndFieldsToBeRendered
expectSettingsSectionsAndFieldsToBeRendered
:
expectSettingsSectionsAndFieldsToBeRendered
,
expectSettingsSectionsAndFieldsToBeRenderedWithMessage
:
expectSettingsSectionsAndFieldsToBeRenderedWithMessage
};
});
lms/static/js/spec/views/fields_helpers.js
View file @
2ac544ca
...
...
@@ -244,6 +244,28 @@ define(['backbone',
}
};
var
verifyReadonlyField
=
function
(
view
,
data
)
{
if
(
data
.
editable
===
'toggle'
)
{
expect
(
view
.
el
).
toHaveClass
(
'mode-placeholder'
);
expectTitleToContain
(
view
,
data
.
title
);
expectMessageContains
(
view
,
view
.
indicators
.
canEdit
);
view
.
$el
.
click
();
}
else
{
expectTitleAndMessageToContain
(
view
,
data
.
title
,
data
.
helpMessage
);
}
expect
(
view
.
el
).
toHaveClass
(
'u-field-readonly'
);
if
(
view
.
fieldValue
()
!==
null
)
{
expect
(
view
.
fieldValue
()).
not
.
toContain
(
data
.
validValue
);
}
};
var
verifyUneditableDropdownField
=
function
(
view
,
data
)
{
expectTitleAndMessageToContain
(
view
,
data
.
title
,
data
.
helpMessage
);
expect
(
view
.
el
).
toHaveClass
(
'u-field-dropdown'
);
expect
(
view
.
el
).
toHaveClass
(
'editable-never'
);
};
var
verifyTextField
=
function
(
view
,
data
,
requests
)
{
verifyEditableField
(
view
,
_
.
extend
({
valueSelector
:
'.u-field-value'
,
...
...
@@ -252,6 +274,12 @@ define(['backbone',
requests
);
};
var
verifyReadonlyTextField
=
function
(
view
,
data
)
{
verifyReadonlyField
(
view
,
_
.
extend
({
valueSelector
:
'.u-field-value'
},
data
));
};
var
verifyDropDownField
=
function
(
view
,
data
,
requests
)
{
verifyEditableField
(
view
,
_
.
extend
({
valueSelector
:
'.u-field-value'
,
...
...
@@ -260,6 +288,12 @@ define(['backbone',
),
requests
);
};
var
verifyReadonlyDropDownField
=
function
(
view
,
data
)
{
verifyUneditableDropdownField
(
view
,
_
.
extend
({
valueSelector
:
'.editable-never'
},
data
));
};
return
{
SELECT_OPTIONS
:
SELECT_OPTIONS
,
UserAccountModel
:
UserAccountModel
,
...
...
@@ -274,7 +308,9 @@ define(['backbone',
verifySuccessMessageReset
:
verifySuccessMessageReset
,
verifyEditableField
:
verifyEditableField
,
verifyTextField
:
verifyTextField
,
verifyReadonlyTextField
:
verifyReadonlyTextField
,
verifyDropDownField
:
verifyDropDownField
,
verifyReadonlyDropDownField
:
verifyReadonlyDropDownField
,
verifyPersistence
:
verifyPersistence
};
});
lms/static/js/student_account/views/account_section_view.js
View file @
2ac544ca
...
...
@@ -5,8 +5,9 @@
'jquery'
,
'underscore'
,
'backbone'
,
'edx-ui-toolkit/js/utils/html-utils'
,
'text!templates/student_account/account_settings_section.underscore'
],
function
(
gettext
,
$
,
_
,
Backbone
,
sectionTemplate
)
{
],
function
(
gettext
,
$
,
_
,
Backbone
,
HtmlUtils
,
sectionTemplate
)
{
var
AccountSectionView
=
Backbone
.
View
.
extend
({
initialize
:
function
(
options
)
{
...
...
@@ -16,6 +17,7 @@
render
:
function
()
{
this
.
$el
.
html
(
_
.
template
(
sectionTemplate
)({
HtmlUtils
:
HtmlUtils
,
sections
:
this
.
options
.
sections
,
tabName
:
this
.
options
.
tabName
,
tabLabel
:
this
.
options
.
tabLabel
...
...
lms/static/js/student_account/views/account_settings_factory.js
View file @
2ac544ca
This diff is collapsed.
Click to expand it.
lms/static/sass/views/_account-settings.scss
View file @
2ac544ca
...
...
@@ -78,7 +78,7 @@
@include
appearance
(
none
);
display
:block
;
display
:
block
;
padding
:
(
$baseline
/
4
);
&
:hover
,
...
...
@@ -101,7 +101,7 @@
border-bottom
:
none
;
.account-nav-link
{
border-bottom
:
4px
solid
theme-color
(
"light"
);
border-bottom
:
4px
solid
theme-color
(
"light"
);
}
}
}
...
...
@@ -130,6 +130,59 @@
padding-bottom
:
10px
;
}
.account-settings-section-message
{
font-size
:
16px
;
line-height
:
22px
;
margin-top
:
15px
;
margin-bottom
:
30px
;
.alert-message
{
color
:
#292b2c
;
font-family
:
$font-family-sans-serif
;
position
:
relative
;
padding
:
10px
10px
10px
35px
;
border
:
1px
solid
transparent
;
border-radius
:
0
;
box-shadow
:
none
;
margin-bottom
:
8px
;
&
>
.fa
{
position
:
absolute
;
left
:
11px
;
top
:
13px
;
font-size
:
16px
;
}
span
{
display
:
block
;
a
{
text-decoration
:
underline
;
}
}
}
.success
{
background-color
:
#ecfaec
;
border-color
:
#b9edb9
;
}
.info
{
background-color
:
#d8edf8
;
border-color
:
#bbdff2
;
}
.warning
{
background-color
:
#fcf8e3
;
border-color
:
#faebcc
;
}
.error
{
background-color
:
#f2dede
;
border-color
:
#ebccd1
;
}
}
.account-settings-section-body
{
.u-field
{
border-bottom
:
2px
solid
$m-gray-l4
;
...
...
@@ -176,6 +229,7 @@
font-size
:
1rem
;
line-height
:
1
;
color
:
$dark-gray
;
white-space
:
nowrap
;
}
.field-input
{
...
...
@@ -219,8 +273,7 @@
padding-top
:
$baseline
;
padding-bottom
:
$baseline
;
line-height
:
normal
;
flex-direction
:
row
;
flex-wrap
:
wrap
;
flex-flow
:
row
wrap
;
span
{
padding
:
$baseline
;
...
...
@@ -301,7 +354,9 @@
font-weight
:
$font-semibold
;
padding
:
0
;
&
:focus
,
&
:hover
,
&
:active
{
&
:focus
,
&
:hover
,
&
:active
{
background-color
:
transparent
;
color
:
$m-blue-d3
;
border
:
none
;
...
...
@@ -372,10 +427,10 @@
display
:
flex
;
flex-wrap
:
nowrap
;
u-field-order-number
,
u-field-order-date
,
u-field-order-price
,
u-field-order-link
,
{
.
u-field-order-number
,
.
u-field-order-date
,
.
u-field-order-price
,
.u-field-order-link
{
width
:
auto
;
float
:
none
;
flex-grow
:
1
;
...
...
@@ -388,8 +443,43 @@
}
}
.u-field-readonly
.u-field-value
{
.u-field
{
&
.u-field-dropdown
,
&
.editable-never
&
.mode-display
{
.u-field-value
{
margin-bottom
:
(
$baseline
);
.u-field-title
{
font-size
:
16px
;
line-height
:
22px
;
margin-bottom
:
18px
;
}
.u-field-value-readonly
{
font-size
:
22px
;
color
:
#636c72
;
line-height
:
30px
;
white-space
:
nowrap
;
}
}
}
}
.u-field-readonly
.u-field-title
{
font-size
:
16px
;
color
:
#636c72
;
line-height
:
22px
;
padding-top
:
(
$baseline
/
2
);
padding-bottom
:
0
;
margin-bottom
:
8px
!
important
;
}
.u-field-readonly
.u-field-value
{
font-size
:
22px
;
color
:
#636c72
;
line-height
:
30px
;
padding-top
:
8px
;
padding-bottom
:
(
$baseline
);
white-space
:
nowrap
;
}
.u-field-orderHistory
{
...
...
@@ -402,7 +492,8 @@
border-bottom
:
1px
solid
$m-gray-l4
;
}
&
:hover
,
&
:focus
{
&
:hover
,
&
:focus
{
background-color
:
$light-gray4
;
}
}
...
...
@@ -413,7 +504,8 @@
margin-bottom
:
0
;
padding-bottom
:
0
;
&
:hover
,
&
:focus
{
&
:hover
,
&
:focus
{
background-color
:
transparent
;
}
...
...
lms/templates/student_account/account_settings.html
View file @
2ac544ca
...
...
@@ -38,7 +38,12 @@ from openedx.core.djangolib.js_utils import dump_js_escaped_json, js_escaped_str
platformName = '${ static.get_platform_name() | n, js_escaped_string }',
contactEmail = '${ static.get_contact_email_address() | n, js_escaped_string }',
allowEmailChange = ${ bool(settings.FEATURES['ALLOW_EMAIL_ADDRESS_CHANGE']) | n, dump_js_escaped_json },
socialPlatforms = ${ settings.SOCIAL_PLATFORMS | n, dump_js_escaped_json };
socialPlatforms = ${ settings.SOCIAL_PLATFORMS | n, dump_js_escaped_json },
syncLearnerProfileData = ${ bool(sync_learner_profile_data) | n, dump_js_escaped_json },
enterpriseName = '${ enterprise_name | n, js_escaped_string }',
enterpriseReadonlyAccountFields = ${ enterprise_readonly_account_fields | n, dump_js_escaped_json },
edxSupportUrl = '${ edx_support_url | n, js_escaped_string }';
AccountSettingsFactory(
fieldsData,
...
...
@@ -51,7 +56,12 @@ from openedx.core.djangolib.js_utils import dump_js_escaped_json, js_escaped_str
platformName,
contactEmail,
allowEmailChange,
socialPlatforms
socialPlatforms,
syncLearnerProfileData,
enterpriseName,
enterpriseReadonlyAccountFields,
edxSupportUrl
);
</
%
static:require
_module
>
</
%
block>
lms/templates/student_account/account_settings_section.underscore
View file @
2ac544ca
...
...
@@ -4,9 +4,19 @@
<% _.each(sections, function(section) { %>
<div class="section">
<h3 class="section-header"><%- gettext(section.title) %></h3>
<% if (section.subtitle) { %>
<% if (section.subtitle
&& _.isUndefined(section.message)
) { %>
<p class="account-settings-header-subtitle"><%- section.subtitle %></p>
<% } %>
<% if (section.message) { %>
<div class="account-settings-section-message">
<div class="alert-message <%- section.messageType%>" aria-live="polite">
<i class="fa fa-info-circle message-icon <%- section.messageType %>" aria-hidden="true"></i>
<span><%= HtmlUtils.ensureHtml(section.message) %></span>
</div>
</div>
<% } %>
<div class="account-settings-section-body <%- tabName %>-section-body">
<div class="ui-loading-error is-hidden">
<span class="fa fa-exclamation-triangle message-error" aria-hidden="true"></span>
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment