+
+
+ {isAdminRole
+ ? intl.formatMessage(messages.roleAdmin)
+ : intl.formatMessage(messages.roleStaff)}
+ {currentUserEmail === email && (
+ {intl.formatMessage(messages.roleYou)}
+ )}
+
+ {userName}
+ {email}
+
+ {/* eslint-disable-next-line no-nested-ternary */}
+ {isAllowActions && (
+ !isHideActions ? (
+
+
+
+ ) : (
+
+ {intl.formatMessage(messages.hint)}
+
+ )
+ )}
+
+ );
+};
+
+CourseTeamMember.propTypes = {
+ userName: PropTypes.string.isRequired,
+ role: PropTypes.string.isRequired,
+ email: PropTypes.string.isRequired,
+ onChangeRole: PropTypes.func.isRequired,
+ onDelete: PropTypes.func.isRequired,
+ currentUserEmail: PropTypes.string.isRequired,
+ isHideActions: PropTypes.bool.isRequired,
+ isAllowActions: PropTypes.bool.isRequired,
+};
+
+export default CourseTeamMember;
diff --git a/src/course-team/course-team-member/CourseTeamMember.scss b/src/course-team/course-team-member/CourseTeamMember.scss
new file mode 100644
index 000000000..0285f13a9
--- /dev/null
+++ b/src/course-team/course-team-member/CourseTeamMember.scss
@@ -0,0 +1,63 @@
+.course-team-container {
+ margin-top: 3rem;
+}
+
+.course-team-member {
+ display: flex;
+ align-items: center;
+ justify-content: space-between;
+ position: relative;
+ padding: 1.563rem 1.875rem 1.25rem;
+ background-color: $white;
+ border: .0625rem solid $gray-200;
+ border-radius: .375rem;
+ box-shadow: 0 1px 1px $gray-200;
+
+ &:not(:last-child) {
+ margin-bottom: 1.25rem;
+ }
+
+ .member-info {
+ position: relative;
+ display: flex;
+ flex-direction: column;
+
+ .badge {
+ position: absolute;
+ top: -2.25rem;
+ left: -.25rem;
+ }
+
+ .badge-current-user {
+ color: $gray-100;
+ margin-left: .25rem;
+ }
+
+ .member-info-name {
+ font-size: 1.5rem;
+ margin-bottom: .25rem;
+ }
+ }
+
+ .member-hint {
+ width: 45%;
+ margin-right: 3.875rem;
+ color: $gray-300;
+ font-size: $font-size-sm;
+ }
+
+ .member-actions {
+ display: flex;
+ gap: $spacer;
+
+ .delete-button {
+ display: flex;
+ align-items: center;
+ justify-content: center;
+
+ & > span {
+ margin: 0;
+ }
+ }
+ }
+}
diff --git a/src/course-team/course-team-member/CourseTeamMember.test.jsx b/src/course-team/course-team-member/CourseTeamMember.test.jsx
new file mode 100644
index 000000000..be6719ac6
--- /dev/null
+++ b/src/course-team/course-team-member/CourseTeamMember.test.jsx
@@ -0,0 +1,91 @@
+import React from 'react';
+import { render, fireEvent } from '@testing-library/react';
+import { IntlProvider } from '@edx/frontend-platform/i18n';
+
+import { USER_ROLES } from '../../constants';
+import CourseTeamMember from './CourseTeamMember';
+import messages from './messages';
+
+const userNameMock = 'User';
+const emailMock = 'user@example.com';
+const currentUserEmailMock = 'user@example.com';
+const onChangeRoleMock = jest.fn();
+const onDeleteMock = jest.fn();
+
+const renderComponent = (props) => render(
+
+
+
+ {intl.formatMessage(messages.sidebarTitle)}
+
+
+ {intl.formatMessage(messages.sidebarAbout_1)}
+
+
+ {intl.formatMessage(messages.sidebarAbout_2)}
+
+
+ {intl.formatMessage(messages.sidebarAbout_3)}
+
+
+ {isOwnershipHint && (
+ <>
+
+
+
+ {intl.formatMessage(messages.ownershipTitle)}
+
+
+ {intl.formatMessage(
+ messages.ownershipDescription,
+ { strong: {intl.formatMessage(messages.addAdminAccess)} },
+ )}
+
+
+ >
+ )}
+
+ );
+};
+
+CourseTeamSideBar.defaultProps = {
+ isShowInitialSidebar: false,
+};
+
+CourseTeamSideBar.propTypes = {
+ courseId: PropTypes.string.isRequired,
+ isOwnershipHint: PropTypes.bool.isRequired,
+ isShowInitialSidebar: PropTypes.bool,
+};
+
+export default CourseTeamSideBar;
diff --git a/src/course-team/course-team-sidebar/CourseTeamSidebar.scss b/src/course-team/course-team-sidebar/CourseTeamSidebar.scss
new file mode 100644
index 000000000..42a8acf52
--- /dev/null
+++ b/src/course-team/course-team-sidebar/CourseTeamSidebar.scss
@@ -0,0 +1,11 @@
+.course-team-sidebar {
+ .help-sidebar {
+ &:not(:first-child) {
+ margin-top: 0;
+ }
+
+ hr {
+ display: none;
+ }
+ }
+}
diff --git a/src/course-team/course-team-sidebar/messages.js b/src/course-team/course-team-sidebar/messages.js
new file mode 100644
index 000000000..a1934c19b
--- /dev/null
+++ b/src/course-team/course-team-sidebar/messages.js
@@ -0,0 +1,34 @@
+import { defineMessages } from '@edx/frontend-platform/i18n';
+
+const messages = defineMessages({
+ sidebarTitle: {
+ id: 'course-authoring.course-team.sidebar.title',
+ defaultMessage: 'Course team roles',
+ },
+ sidebarAbout_1: {
+ id: 'course-authoring.course-team.sidebar.about-1',
+ defaultMessage: 'Course team members with the Staff role are course co-authors. They have full writing and editing privileges on all course content.',
+ },
+ sidebarAbout_2: {
+ id: 'course-authoring.course-team.sidebar.about-2',
+ defaultMessage: 'Admins are course team members who can add and remove other course team members.',
+ },
+ sidebarAbout_3: {
+ id: 'course-authoring.course-team.sidebar.about-3',
+ defaultMessage: 'All course team members can access content in Studio, the LMS, and Insights, but are not automatically enrolled in the course.',
+ },
+ ownershipTitle: {
+ id: 'course-authoring.course-team.sidebar.ownership.title',
+ defaultMessage: 'Transferring ownership',
+ },
+ ownershipDescription: {
+ id: 'course-authoring.course-team.sidebar.ownership.description',
+ defaultMessage: 'Every course must have an Admin. If you are the Admin and you want to transfer ownership of the course, click {strong} to make another user the Admin, then ask that user to remove you from the Course Team list.',
+ },
+ addAdminAccess: {
+ id: 'course-authoring.course-team.sidebar.ownership.addAdminAccess',
+ defaultMessage: 'Add admin access',
+ },
+});
+
+export default messages;
diff --git a/src/course-team/data/api.js b/src/course-team/data/api.js
new file mode 100644
index 000000000..321671600
--- /dev/null
+++ b/src/course-team/data/api.js
@@ -0,0 +1,54 @@
+import { camelCaseObject, getConfig } from '@edx/frontend-platform';
+import { getAuthenticatedHttpClient } from '@edx/frontend-platform/auth';
+
+import { USER_ROLES } from '../../constants';
+
+const getApiBaseUrl = () => getConfig().STUDIO_BASE_URL;
+export const getCourseTeamApiUrl = (courseId) => `${getApiBaseUrl()}/api/contentstore/v1/course_team/${courseId}`;
+export const updateCourseTeamUserApiUrl = (courseId, email) => `${getApiBaseUrl()}/course_team/${courseId}/${email}`;
+
+/**
+ * Get course team.
+ * @param {string} courseId
+ * @returns {Promise