Compare commits
26 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7de6ba4381 | ||
|
|
14fe2d9bc6 | ||
|
|
78573e30f1 | ||
|
|
ae014d2f24 | ||
|
|
726aff9f8d | ||
|
|
bb7c5cb39f | ||
|
|
90df9feb87 | ||
|
|
a86cde04bd | ||
|
|
afc03b2253 | ||
|
|
01cee41441 | ||
|
|
307669cf9f | ||
|
|
265f3be635 | ||
|
|
7742de1609 | ||
|
|
1778b78f0a | ||
|
|
36ced5252f | ||
|
|
00ca9197d5 | ||
|
|
adf6f71c05 | ||
|
|
fee5d11aec | ||
|
|
5772de187f | ||
|
|
4c033a655a | ||
|
|
e75043dd2b | ||
|
|
97a15f5059 | ||
|
|
0b882173eb | ||
|
|
ebaa9ea32e | ||
|
|
07faa1290a | ||
|
|
61b68fdd80 |
@@ -13,3 +13,7 @@ REFRESH_ACCESS_TOKEN_ENDPOINT=http://localhost:18000/login_refresh
|
|||||||
SEGMENT_KEY=null
|
SEGMENT_KEY=null
|
||||||
SITE_NAME=Open edX
|
SITE_NAME=Open edX
|
||||||
USER_INFO_COOKIE_NAME=edx-user-info
|
USER_INFO_COOKIE_NAME=edx-user-info
|
||||||
|
LOGO_URL=https://edx-cdn.org/v3/default/logo.svg
|
||||||
|
LOGO_TRADEMARK_URL=https://edx-cdn.org/v3/default/logo-trademark.svg
|
||||||
|
LOGO_WHITE_URL=https://edx-cdn.org/v3/default/logo-white.svg
|
||||||
|
FAVICON_URL=https://edx-cdn.org/v3/default/favicon.ico
|
||||||
|
|||||||
26
.github/workflows/ci.yml
vendored
Normal file
26
.github/workflows/ci.yml
vendored
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
name: Default CI
|
||||||
|
on: [push, pull_request]
|
||||||
|
jobs:
|
||||||
|
build:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- name: Setup Nodejs
|
||||||
|
uses: actions/setup-node@v1
|
||||||
|
with:
|
||||||
|
node-version: 12
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
- name: Validate package-lock.json changes
|
||||||
|
run: make validate-no-uncommitted-package-lock-changes
|
||||||
|
- name: Lint
|
||||||
|
run: npm run lint
|
||||||
|
- name: Test
|
||||||
|
run: npm run test
|
||||||
|
- name: i18n_extract
|
||||||
|
run: npm run i18n_extract
|
||||||
|
- name: Coverage
|
||||||
|
uses: codecov/codecov-action@v1
|
||||||
37
.github/workflows/release.yml
vendored
Normal file
37
.github/workflows/release.yml
vendored
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
name: Release CI
|
||||||
|
on:
|
||||||
|
push:
|
||||||
|
branches:
|
||||||
|
- master
|
||||||
|
jobs:
|
||||||
|
release:
|
||||||
|
name: Release
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v2
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
- name: Setup Node.js
|
||||||
|
uses: actions/setup-node@v1
|
||||||
|
with:
|
||||||
|
node-version: 12
|
||||||
|
- name: Install dependencies
|
||||||
|
run: npm ci
|
||||||
|
- name: Validate package-lock.json changes
|
||||||
|
run: make validate-no-uncommitted-package-lock-changes
|
||||||
|
- name: Lint
|
||||||
|
run: npm run lint
|
||||||
|
- name: Test
|
||||||
|
run: npm run test
|
||||||
|
- name: i18n_extract
|
||||||
|
run: npm run i18n_extract
|
||||||
|
- name: Coverage
|
||||||
|
uses: codecov/codecov-action@v1
|
||||||
|
- name: Build
|
||||||
|
run: npm run build
|
||||||
|
- name: Release
|
||||||
|
env:
|
||||||
|
GITHUB_TOKEN: ${{ secrets.SEMANTIC_RELEASE_GITHUB_TOKEN }}
|
||||||
|
NPM_TOKEN: ${{ secrets.SEMANTIC_RELEASE_NPM_TOKEN }}
|
||||||
|
run: npx semantic-release
|
||||||
1
.gitignore
vendored
1
.gitignore
vendored
@@ -5,3 +5,4 @@ dist
|
|||||||
node_modules
|
node_modules
|
||||||
temp
|
temp
|
||||||
src/i18n/transifex_input.json
|
src/i18n/transifex_input.json
|
||||||
|
module.config.js
|
||||||
|
|||||||
17
.travis.yml
17
.travis.yml
@@ -1,17 +0,0 @@
|
|||||||
language: node_js
|
|
||||||
node_js: 12
|
|
||||||
install:
|
|
||||||
- npm install
|
|
||||||
script:
|
|
||||||
- npm run lint
|
|
||||||
- npm run test
|
|
||||||
- npm run build
|
|
||||||
after_success:
|
|
||||||
- npx semantic-release
|
|
||||||
- codecov
|
|
||||||
env:
|
|
||||||
global:
|
|
||||||
# GH_TOKEN
|
|
||||||
- secure: CFN/uOByWC+7S+AAXECLQ0mNgoiyCDl1ZB5AT5+/qP+xEVs0ysFKDWrD9W8KeQqHqCMPUKNt23nrgxwvFCkSx+n1MAnxZmOdVJWbzGbUB9TsqrwVUALkqof1MrRB8UFVEzIRaO60iRN/L3zVXML+4GsycYX6rHyptbAypxplpljDyKrY8tc/mM7AGZ9eVFGSq+7CXXmdvkhP9kLkH1tIYvR7wjTKvZHbHf6YRjIVCiyzxM4S/E9l8JRnbERp02XosRD62PUJXXk6EJVn6Qoub6CaPnpew5crW0iRF1UJs54U29zWd/S+LuW66WkLfJu7rDq6AFJNMtMNusJxwVkOv4X+p0oJDWZEhojW+/Wm10UAu8/g5oAqeePZEGiSbT3Hp1VqOc4FY/kmOLiM+L6oq/AA2XX6iiE8lA64IH5R8ApQamF4GTUYTvKHeLPgXnGJH7A95Xy9/+jmX9I9wJREMrHrkyPjoX/NTRdG+RrebB1+An9Bt9vAbG862gbOZfgTcuWHDOlG5gcA964Fr7RqR5a62yr45Gw+Q0lTrLj5mGAjjSpMRIAQzi9e7oXmoMZnvenu/WJAe5M+u+/gv82HeXcMwLvNNGSvz+0i1xNUOoX1zHG046oGKiX0Zu+l0JfNwihJTO7vJlaITmjhfOyufwpk74xEyrhf8nLF9e5Frec=
|
|
||||||
# NPM_TOKEN
|
|
||||||
- secure: fuV04Ctf0mgbw6nTJhsTzGZ6dyafZtGVj30ZkvSWsB9hUU8KDtl7wWVW9EayCQsSyyFgPY9RVav5olgC/zljAjDGg0nfF7n8uKIABA0TXdP683WZd06bVOmDXfL26B3yM83aW2xgHZN6VCCvCE8bLP1V6eV8nsr38gDgxVbHa0YasDMmvtYrog+IjxwjcJx7fD0RbYyi7iJC++pdw9kcFqOad28Us7L/jCn+rC3CmUT4kOwPjP5g1v5sB2FA7ouN5s1hUUTKuttV32VJgRP7wbZzoHeHX5BRGSqijdXNSaK5UwzqRnM1sGZkuNDZhJbSB3q90SQrPRgV+fRizwN8zs8Htb+Kk8+wGY6zNhmi9C+lUIv7UpDYbstMWYIf39+P/24Oj+vJBjMY30M9NWB8gt1OQ0dJUoK53v1+BMVmDB0doL6I53xwzUjQetvqOF0Wm3E3OrqJP00OJdzIcAeh2DzcIRW1SrBhI4HAsl7QJZNpRw11QzJ3K2iQSiWNd8qIuX+XJjzQdn1v09gCstvbP33Vn8tP1x0XTSi8wIhTDqE0bII/Sc80Jh76nu0ItQ8pmX4lGsER4C0N3Dp7Zz51yW7E70AWWLsUrMNdF0oHoP437ZRGPhYs5OI6x5AM2jlU8fJ5aUroEsYFCwkH0OO37THohpAQpApe9zL10miTEbw=
|
|
||||||
@@ -4,8 +4,9 @@ import React from 'react';
|
|||||||
import ReactDOM from 'react-dom';
|
import ReactDOM from 'react-dom';
|
||||||
import { initialize, getConfig, subscribe, APP_READY } from '@edx/frontend-platform';
|
import { initialize, getConfig, subscribe, APP_READY } from '@edx/frontend-platform';
|
||||||
import { AppContext, AppProvider } from '@edx/frontend-platform/react';
|
import { AppContext, AppProvider } from '@edx/frontend-platform/react';
|
||||||
|
import Header from '@edx/frontend-component-header';
|
||||||
|
|
||||||
import './index.scss';
|
import './index.scss';
|
||||||
import Header from '../src/';
|
|
||||||
|
|
||||||
subscribe(APP_READY, () => {
|
subscribe(APP_READY, () => {
|
||||||
ReactDOM.render(
|
ReactDOM.render(
|
||||||
|
|||||||
@@ -1,2 +1,6 @@
|
|||||||
@import "~@edx/paragon/scss/core/core.scss";
|
@import "@edx/brand/paragon/fonts";
|
||||||
@import '../src/index.scss';
|
@import "@edx/brand/paragon/variables";
|
||||||
|
@import "@edx/paragon/scss/core/core";
|
||||||
|
@import "@edx/brand/paragon/overrides";
|
||||||
|
|
||||||
|
@import "@edx/frontend-component-header/index";
|
||||||
|
|||||||
14684
package-lock.json
generated
14684
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
23
package.json
23
package.json
@@ -38,30 +38,31 @@
|
|||||||
"@commitlint/config-angular": "8.2.0",
|
"@commitlint/config-angular": "8.2.0",
|
||||||
"@commitlint/prompt": "8.2.0",
|
"@commitlint/prompt": "8.2.0",
|
||||||
"@commitlint/prompt-cli": "8.2.0",
|
"@commitlint/prompt-cli": "8.2.0",
|
||||||
"@edx/frontend-build": "2.0.3",
|
"@edx/brand": "npm:@edx/brand-openedx@1.1.0",
|
||||||
"@edx/frontend-platform": "1.1.4",
|
"@edx/frontend-build": "5.4.0",
|
||||||
"@edx/paragon": "7.1.4",
|
"@edx/frontend-platform": "1.8.0",
|
||||||
"codecov": "3.6.1",
|
"@edx/paragon": "12.0.5",
|
||||||
|
"codecov": "3.7.2",
|
||||||
"enzyme": "3.10.0",
|
"enzyme": "3.10.0",
|
||||||
"enzyme-adapter-react-16": "1.14.0",
|
"enzyme-adapter-react-16": "1.14.0",
|
||||||
"husky": "3.0.8",
|
"husky": "3.0.9",
|
||||||
"prop-types": "15.7.2",
|
"prop-types": "15.7.2",
|
||||||
"react": "16.9.0",
|
"react": "16.9.0",
|
||||||
"react-dom": "16.9.0",
|
"react-dom": "16.9.0",
|
||||||
"react-redux": "^7.1.1",
|
"react-redux": "7.1.1",
|
||||||
"react-router-dom": "^5.1.2",
|
"react-router-dom": "5.1.2",
|
||||||
"react-test-renderer": "16.9.0",
|
"react-test-renderer": "16.9.0",
|
||||||
"reactifex": "1.1.1",
|
"reactifex": "1.1.1",
|
||||||
"redux": "^4.0.4",
|
"redux": "4.0.4",
|
||||||
"redux-saga": "^1.1.1"
|
"redux-saga": "1.1.1"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"babel-polyfill": "6.26.0",
|
"babel-polyfill": "6.26.0",
|
||||||
"react-responsive": "8.0.1",
|
"react-responsive": "8.0.3",
|
||||||
"react-transition-group": "4.3.0"
|
"react-transition-group": "4.3.0"
|
||||||
},
|
},
|
||||||
"peerDependencies": {
|
"peerDependencies": {
|
||||||
"@edx/frontend-platform": "^1.1.4",
|
"@edx/frontend-platform": "^1.8.0",
|
||||||
"@edx/paragon": "^7.0.0",
|
"@edx/paragon": "^7.0.0",
|
||||||
"prop-types": "^15.5.10",
|
"prop-types": "^15.5.10",
|
||||||
"react": "^16.9.0",
|
"react": "^16.9.0",
|
||||||
|
|||||||
@@ -1,5 +1,9 @@
|
|||||||
{
|
{
|
||||||
"extends": [
|
"extends": [
|
||||||
"config:base"
|
"config:base"
|
||||||
]
|
],
|
||||||
|
"patch": {
|
||||||
|
"automerge": true
|
||||||
|
},
|
||||||
|
"rebaseStalePrs": true
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ function Avatar({
|
|||||||
const avatar = src ? (
|
const avatar = src ? (
|
||||||
<img className="d-block w-100 h-100" src={src} alt={alt} />
|
<img className="d-block w-100 h-100" src={src} alt={alt} />
|
||||||
) : (
|
) : (
|
||||||
<AvatarIcon className="text-muted" style={{ width: size, height: size }} role="img" aria-hidden focusable="false" />
|
<AvatarIcon style={{ width: size, height: size }} role="img" aria-hidden focusable="false" />
|
||||||
);
|
);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
|||||||
@@ -22,7 +22,9 @@ class DesktopHeader extends React.Component {
|
|||||||
const { mainMenu } = this.props;
|
const { mainMenu } = this.props;
|
||||||
|
|
||||||
// Nodes are accepted as a prop
|
// Nodes are accepted as a prop
|
||||||
if (!Array.isArray(mainMenu)) return mainMenu;
|
if (!Array.isArray(mainMenu)) {
|
||||||
|
return mainMenu;
|
||||||
|
}
|
||||||
|
|
||||||
return mainMenu.map((menuItem) => {
|
return mainMenu.map((menuItem) => {
|
||||||
const {
|
const {
|
||||||
@@ -64,7 +66,7 @@ class DesktopHeader extends React.Component {
|
|||||||
<MenuTrigger
|
<MenuTrigger
|
||||||
tag="button"
|
tag="button"
|
||||||
aria-label={intl.formatMessage(messages['header.label.account.menu.for'], { username })}
|
aria-label={intl.formatMessage(messages['header.label.account.menu.for'], { username })}
|
||||||
className="btn btn-light d-inline-flex align-items-center pl-2 pr-3"
|
className="btn btn-outline-primary d-inline-flex align-items-center pl-2 pr-3"
|
||||||
>
|
>
|
||||||
<Avatar size="1.5em" src={avatar} alt="" className="mr-2" />
|
<Avatar size="1.5em" src={avatar} alt="" className="mr-2" />
|
||||||
{username} <CaretIcon role="img" aria-hidden focusable="false" />
|
{username} <CaretIcon role="img" aria-hidden focusable="false" />
|
||||||
@@ -106,7 +108,7 @@ class DesktopHeader extends React.Component {
|
|||||||
<header className="site-header-desktop">
|
<header className="site-header-desktop">
|
||||||
<div className="container-fluid">
|
<div className="container-fluid">
|
||||||
<div className="nav-container position-relative d-flex align-items-center">
|
<div className="nav-container position-relative d-flex align-items-center">
|
||||||
{ logoDestination === null ? <Logo className="logo" src={logo} alt={logoAltText} /> : <LinkedLogo className="logo" {...logoProps} />}
|
{logoDestination === null ? <Logo className="logo" src={logo} alt={logoAltText} /> : <LinkedLogo className="logo" {...logoProps} />}
|
||||||
<nav
|
<nav
|
||||||
aria-label={intl.formatMessage(messages['header.label.main.nav'])}
|
aria-label={intl.formatMessage(messages['header.label.main.nav'])}
|
||||||
className="nav main-nav"
|
className="nav main-nav"
|
||||||
|
|||||||
@@ -2,13 +2,17 @@ import React, { useContext } from 'react';
|
|||||||
import Responsive from 'react-responsive';
|
import Responsive from 'react-responsive';
|
||||||
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
import { injectIntl, intlShape } from '@edx/frontend-platform/i18n';
|
||||||
import { AppContext } from '@edx/frontend-platform/react';
|
import { AppContext } from '@edx/frontend-platform/react';
|
||||||
import { ensureConfig } from '@edx/frontend-platform/config';
|
import {
|
||||||
|
APP_CONFIG_INITIALIZED,
|
||||||
|
ensureConfig,
|
||||||
|
mergeConfig,
|
||||||
|
getConfig,
|
||||||
|
subscribe,
|
||||||
|
} from '@edx/frontend-platform';
|
||||||
|
|
||||||
import DesktopHeader from './DesktopHeader';
|
import DesktopHeader from './DesktopHeader';
|
||||||
import MobileHeader from './MobileHeader';
|
import MobileHeader from './MobileHeader';
|
||||||
|
|
||||||
import LogoSVG from './logo.svg';
|
|
||||||
|
|
||||||
import messages from './Header.messages';
|
import messages from './Header.messages';
|
||||||
|
|
||||||
ensureConfig([
|
ensureConfig([
|
||||||
@@ -16,8 +20,15 @@ ensureConfig([
|
|||||||
'LOGOUT_URL',
|
'LOGOUT_URL',
|
||||||
'LOGIN_URL',
|
'LOGIN_URL',
|
||||||
'SITE_NAME',
|
'SITE_NAME',
|
||||||
|
'LOGO_URL',
|
||||||
], 'Header component');
|
], 'Header component');
|
||||||
|
|
||||||
|
subscribe(APP_CONFIG_INITIALIZED, () => {
|
||||||
|
mergeConfig({
|
||||||
|
LOGISTRATION_MINIMAL_HEADER: !!process.env.LOGISTRATION_MINIMAL_HEADER,
|
||||||
|
}, 'Header additional config');
|
||||||
|
});
|
||||||
|
|
||||||
function Header({ intl }) {
|
function Header({ intl }) {
|
||||||
const { authenticatedUser, config } = useContext(AppContext);
|
const { authenticatedUser, config } = useContext(AppContext);
|
||||||
|
|
||||||
@@ -66,27 +77,27 @@ function Header({ intl }) {
|
|||||||
];
|
];
|
||||||
|
|
||||||
const props = {
|
const props = {
|
||||||
logo: LogoSVG,
|
logo: config.LOGO_URL,
|
||||||
logoAltText: config.SITE_NAME,
|
logoAltText: config.SITE_NAME,
|
||||||
siteName: config.SITE_NAME,
|
siteName: config.SITE_NAME,
|
||||||
logoDestination: `${config.LMS_BASE_URL}/dashboard`,
|
logoDestination: `${config.LMS_BASE_URL}/dashboard`,
|
||||||
loggedIn: authenticatedUser !== null,
|
loggedIn: authenticatedUser !== null,
|
||||||
username: authenticatedUser !== null ? authenticatedUser.username : null,
|
username: authenticatedUser !== null ? authenticatedUser.username : null,
|
||||||
avatar: authenticatedUser !== null ? authenticatedUser.avatar : null,
|
avatar: authenticatedUser !== null ? authenticatedUser.avatar : null,
|
||||||
mainMenu,
|
mainMenu: getConfig().LOGISTRATION_MINIMAL_HEADER ? [] : mainMenu,
|
||||||
userMenu,
|
userMenu: getConfig().LOGISTRATION_MINIMAL_HEADER ? [] : userMenu,
|
||||||
loggedOutItems,
|
loggedOutItems: getConfig().LOGISTRATION_MINIMAL_HEADER ? [] : loggedOutItems,
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<React.Fragment>
|
<>
|
||||||
<Responsive maxWidth={768}>
|
<Responsive maxWidth={768}>
|
||||||
<MobileHeader {...props} />
|
<MobileHeader {...props} />
|
||||||
</Responsive>
|
</Responsive>
|
||||||
<Responsive minWidth={769}>
|
<Responsive minWidth={769}>
|
||||||
<DesktopHeader {...props} />
|
<DesktopHeader {...props} />
|
||||||
</Responsive>
|
</Responsive>
|
||||||
</React.Fragment>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ const messages = defineMessages({
|
|||||||
description: 'The aria label for the account menu trigger',
|
description: 'The aria label for the account menu trigger',
|
||||||
},
|
},
|
||||||
'header.label.account.menu.for': {
|
'header.label.account.menu.for': {
|
||||||
id: 'header.label.account.menu',
|
id: 'header.label.account.menu.for',
|
||||||
defaultMessage: 'Account menu for {username}',
|
defaultMessage: 'Account menu for {username}',
|
||||||
description: 'The aria label for the account menu trigger when the username is displayed in it',
|
description: 'The aria label for the account menu trigger when the username is displayed in it',
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -11,14 +11,16 @@ describe('<Header />', () => {
|
|||||||
const component = (
|
const component = (
|
||||||
<ResponsiveContext.Provider value={{ width: 1280 }}>
|
<ResponsiveContext.Provider value={{ width: 1280 }}>
|
||||||
<IntlProvider locale="en" messages={{}}>
|
<IntlProvider locale="en" messages={{}}>
|
||||||
<AppContext.Provider value={{
|
<AppContext.Provider
|
||||||
authenticatedUser: null,
|
value={{
|
||||||
config: {
|
authenticatedUser: null,
|
||||||
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
config: {
|
||||||
SITE_NAME: process.env.SITE_NAME,
|
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
||||||
LOGIN_URL: process.env.LOGIN_URL,
|
SITE_NAME: process.env.SITE_NAME,
|
||||||
LOGOUT_URL: process.env.LOGOUT_URL,
|
LOGIN_URL: process.env.LOGIN_URL,
|
||||||
},
|
LOGOUT_URL: process.env.LOGOUT_URL,
|
||||||
|
LOGO_URL: process.env.LOGO_URL,
|
||||||
|
},
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Header />
|
<Header />
|
||||||
@@ -36,19 +38,21 @@ describe('<Header />', () => {
|
|||||||
const component = (
|
const component = (
|
||||||
<ResponsiveContext.Provider value={{ width: 1280 }}>
|
<ResponsiveContext.Provider value={{ width: 1280 }}>
|
||||||
<IntlProvider locale="en" messages={{}}>
|
<IntlProvider locale="en" messages={{}}>
|
||||||
<AppContext.Provider value={{
|
<AppContext.Provider
|
||||||
authenticatedUser: {
|
value={{
|
||||||
userId: 'abc123',
|
authenticatedUser: {
|
||||||
username: 'edX',
|
userId: 'abc123',
|
||||||
roles: [],
|
username: 'edX',
|
||||||
administrator: false,
|
roles: [],
|
||||||
},
|
administrator: false,
|
||||||
config: {
|
},
|
||||||
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
config: {
|
||||||
SITE_NAME: process.env.SITE_NAME,
|
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
||||||
LOGIN_URL: process.env.LOGIN_URL,
|
SITE_NAME: process.env.SITE_NAME,
|
||||||
LOGOUT_URL: process.env.LOGOUT_URL,
|
LOGIN_URL: process.env.LOGIN_URL,
|
||||||
},
|
LOGOUT_URL: process.env.LOGOUT_URL,
|
||||||
|
LOGO_URL: process.env.LOGO_URL,
|
||||||
|
},
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Header />
|
<Header />
|
||||||
@@ -66,14 +70,16 @@ describe('<Header />', () => {
|
|||||||
const component = (
|
const component = (
|
||||||
<ResponsiveContext.Provider value={{ width: 500 }}>
|
<ResponsiveContext.Provider value={{ width: 500 }}>
|
||||||
<IntlProvider locale="en" messages={{}}>
|
<IntlProvider locale="en" messages={{}}>
|
||||||
<AppContext.Provider value={{
|
<AppContext.Provider
|
||||||
authenticatedUser: null,
|
value={{
|
||||||
config: {
|
authenticatedUser: null,
|
||||||
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
config: {
|
||||||
SITE_NAME: process.env.SITE_NAME,
|
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
||||||
LOGIN_URL: process.env.LOGIN_URL,
|
SITE_NAME: process.env.SITE_NAME,
|
||||||
LOGOUT_URL: process.env.LOGOUT_URL,
|
LOGIN_URL: process.env.LOGIN_URL,
|
||||||
},
|
LOGOUT_URL: process.env.LOGOUT_URL,
|
||||||
|
LOGO_URL: process.env.LOGO_URL,
|
||||||
|
},
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Header />
|
<Header />
|
||||||
@@ -91,19 +97,21 @@ describe('<Header />', () => {
|
|||||||
const component = (
|
const component = (
|
||||||
<ResponsiveContext.Provider value={{ width: 500 }}>
|
<ResponsiveContext.Provider value={{ width: 500 }}>
|
||||||
<IntlProvider locale="en" messages={{}}>
|
<IntlProvider locale="en" messages={{}}>
|
||||||
<AppContext.Provider value={{
|
<AppContext.Provider
|
||||||
authenticatedUser: {
|
value={{
|
||||||
userId: 'abc123',
|
authenticatedUser: {
|
||||||
username: 'edX',
|
userId: 'abc123',
|
||||||
roles: [],
|
username: 'edX',
|
||||||
administrator: false,
|
roles: [],
|
||||||
},
|
administrator: false,
|
||||||
config: {
|
},
|
||||||
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
config: {
|
||||||
SITE_NAME: process.env.SITE_NAME,
|
LMS_BASE_URL: process.env.LMS_BASE_URL,
|
||||||
LOGIN_URL: process.env.LOGIN_URL,
|
SITE_NAME: process.env.SITE_NAME,
|
||||||
LOGOUT_URL: process.env.LOGOUT_URL,
|
LOGIN_URL: process.env.LOGIN_URL,
|
||||||
},
|
LOGOUT_URL: process.env.LOGOUT_URL,
|
||||||
|
LOGO_URL: process.env.LOGO_URL,
|
||||||
|
},
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
<Header />
|
<Header />
|
||||||
|
|||||||
@@ -1,7 +1,6 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
|
||||||
|
|
||||||
function Logo({ src, alt, ...attributes }) {
|
function Logo({ src, alt, ...attributes }) {
|
||||||
return (
|
return (
|
||||||
<img src={src} alt={alt} {...attributes} />
|
<img src={src} alt={alt} {...attributes} />
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ import React from 'react';
|
|||||||
import { CSSTransition } from 'react-transition-group';
|
import { CSSTransition } from 'react-transition-group';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
|
|
||||||
|
|
||||||
function MenuTrigger({ tag, className, ...attributes }) {
|
function MenuTrigger({ tag, className, ...attributes }) {
|
||||||
return React.createElement(tag, {
|
return React.createElement(tag, {
|
||||||
className: `menu-trigger ${className}`,
|
className: `menu-trigger ${className}`,
|
||||||
@@ -19,7 +18,6 @@ MenuTrigger.defaultProps = {
|
|||||||
};
|
};
|
||||||
const MenuTriggerType = <MenuTrigger />.type;
|
const MenuTriggerType = <MenuTrigger />.type;
|
||||||
|
|
||||||
|
|
||||||
function MenuContent({ tag, className, ...attributes }) {
|
function MenuContent({ tag, className, ...attributes }) {
|
||||||
return React.createElement(tag, {
|
return React.createElement(tag, {
|
||||||
className: ['menu-content', className].join(' '),
|
className: ['menu-content', className].join(' '),
|
||||||
@@ -35,6 +33,17 @@ MenuContent.defaultProps = {
|
|||||||
className: null,
|
className: null,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const menuPropTypes = {
|
||||||
|
tag: PropTypes.string,
|
||||||
|
onClose: PropTypes.func,
|
||||||
|
onOpen: PropTypes.func,
|
||||||
|
closeOnDocumentClick: PropTypes.bool,
|
||||||
|
respondToPointerEvents: PropTypes.bool,
|
||||||
|
className: PropTypes.string,
|
||||||
|
transitionTimeout: PropTypes.number,
|
||||||
|
transitionClassName: PropTypes.string,
|
||||||
|
children: PropTypes.arrayOf(PropTypes.node).isRequired,
|
||||||
|
};
|
||||||
|
|
||||||
class Menu extends React.Component {
|
class Menu extends React.Component {
|
||||||
constructor(props) {
|
constructor(props) {
|
||||||
@@ -66,10 +75,14 @@ class Menu extends React.Component {
|
|||||||
|
|
||||||
// Event handlers
|
// Event handlers
|
||||||
onDocumentClick(e) {
|
onDocumentClick(e) {
|
||||||
if (!this.props.closeOnDocumentClick) return;
|
if (!this.props.closeOnDocumentClick) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const clickIsInMenu = this.menu.current === e.target || this.menu.current.contains(e.target);
|
const clickIsInMenu = this.menu.current === e.target || this.menu.current.contains(e.target);
|
||||||
if (clickIsInMenu) return;
|
if (clickIsInMenu) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
this.close();
|
this.close();
|
||||||
}
|
}
|
||||||
@@ -77,7 +90,9 @@ class Menu extends React.Component {
|
|||||||
onTriggerClick(e) {
|
onTriggerClick(e) {
|
||||||
// Let the browser follow the link of the trigger if the menu
|
// Let the browser follow the link of the trigger if the menu
|
||||||
// is already expanded and the trigger has an href attribute
|
// is already expanded and the trigger has an href attribute
|
||||||
if (this.state.expanded && e.target.getAttribute('href')) return;
|
if (this.state.expanded && e.target.getAttribute('href')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
this.toggle();
|
this.toggle();
|
||||||
@@ -89,7 +104,9 @@ class Menu extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onKeyDown(e) {
|
onKeyDown(e) {
|
||||||
if (!this.state.expanded) return;
|
if (!this.state.expanded) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
switch (e.key) {
|
switch (e.key) {
|
||||||
case 'Escape': {
|
case 'Escape': {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@@ -131,16 +148,19 @@ class Menu extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
onMouseEnter() {
|
onMouseEnter() {
|
||||||
if (!this.props.respondToPointerEvents) return;
|
if (!this.props.respondToPointerEvents) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.open();
|
this.open();
|
||||||
}
|
}
|
||||||
|
|
||||||
onMouseLeave() {
|
onMouseLeave() {
|
||||||
if (!this.props.respondToPointerEvents) return;
|
if (!this.props.respondToPointerEvents) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
this.close();
|
this.close();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Internal functions
|
// Internal functions
|
||||||
|
|
||||||
getFocusableElements() {
|
getFocusableElements() {
|
||||||
@@ -151,7 +171,7 @@ class Menu extends React.Component {
|
|||||||
// Any extra props are attributes for the menu
|
// Any extra props are attributes for the menu
|
||||||
const attributes = {};
|
const attributes = {};
|
||||||
Object.keys(this.props)
|
Object.keys(this.props)
|
||||||
.filter(property => Menu.propTypes[property] === undefined)
|
.filter(property => menuPropTypes[property] === undefined)
|
||||||
.forEach((property) => {
|
.forEach((property) => {
|
||||||
attributes[property] = this.props[property];
|
attributes[property] = this.props[property];
|
||||||
});
|
});
|
||||||
@@ -173,7 +193,9 @@ class Menu extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
open() {
|
open() {
|
||||||
if (this.props.onOpen) this.props.onOpen();
|
if (this.props.onOpen) {
|
||||||
|
this.props.onOpen();
|
||||||
|
}
|
||||||
this.setState({ expanded: true });
|
this.setState({ expanded: true });
|
||||||
// Listen to touchend and click events to ensure the menu
|
// Listen to touchend and click events to ensure the menu
|
||||||
// can be closed on mobile, pointer, and mixed input devices
|
// can be closed on mobile, pointer, and mixed input devices
|
||||||
@@ -182,7 +204,9 @@ class Menu extends React.Component {
|
|||||||
}
|
}
|
||||||
|
|
||||||
close() {
|
close() {
|
||||||
if (this.props.onClose) this.props.onClose();
|
if (this.props.onClose) {
|
||||||
|
this.props.onClose();
|
||||||
|
}
|
||||||
this.setState({ expanded: false });
|
this.setState({ expanded: false });
|
||||||
document.removeEventListener('touchend', this.onDocumentClick, true);
|
document.removeEventListener('touchend', this.onDocumentClick, true);
|
||||||
document.removeEventListener('click', this.onDocumentClick, true);
|
document.removeEventListener('click', this.onDocumentClick, true);
|
||||||
@@ -240,18 +264,7 @@ class Menu extends React.Component {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Menu.propTypes = menuPropTypes;
|
||||||
Menu.propTypes = {
|
|
||||||
tag: PropTypes.string,
|
|
||||||
onClose: PropTypes.func,
|
|
||||||
onOpen: PropTypes.func,
|
|
||||||
closeOnDocumentClick: PropTypes.bool,
|
|
||||||
respondToPointerEvents: PropTypes.bool,
|
|
||||||
className: PropTypes.string,
|
|
||||||
transitionTimeout: PropTypes.number,
|
|
||||||
transitionClassName: PropTypes.string,
|
|
||||||
children: PropTypes.arrayOf(PropTypes.node).isRequired,
|
|
||||||
};
|
|
||||||
Menu.defaultProps = {
|
Menu.defaultProps = {
|
||||||
tag: 'div',
|
tag: 'div',
|
||||||
className: null,
|
className: null,
|
||||||
@@ -263,5 +276,4 @@ Menu.defaultProps = {
|
|||||||
transitionClassName: 'menu-content',
|
transitionClassName: 'menu-content',
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
export { Menu, MenuTrigger, MenuContent };
|
export { Menu, MenuTrigger, MenuContent };
|
||||||
|
|||||||
@@ -22,7 +22,9 @@ class MobileHeader extends React.Component {
|
|||||||
const { mainMenu } = this.props;
|
const { mainMenu } = this.props;
|
||||||
|
|
||||||
// Nodes are accepted as a prop
|
// Nodes are accepted as a prop
|
||||||
if (!Array.isArray(mainMenu)) return mainMenu;
|
if (!Array.isArray(mainMenu)) {
|
||||||
|
return mainMenu;
|
||||||
|
}
|
||||||
|
|
||||||
return mainMenu.map((menuItem) => {
|
return mainMenu.map((menuItem) => {
|
||||||
const {
|
const {
|
||||||
@@ -89,6 +91,8 @@ class MobileHeader extends React.Component {
|
|||||||
stickyOnMobile,
|
stickyOnMobile,
|
||||||
intl,
|
intl,
|
||||||
mainMenu,
|
mainMenu,
|
||||||
|
userMenu,
|
||||||
|
loggedOutItems,
|
||||||
} = this.props;
|
} = this.props;
|
||||||
const logoProps = { src: logo, alt: logoAltText, href: logoDestination };
|
const logoProps = { src: logo, alt: logoAltText, href: logoDestination };
|
||||||
const stickyClassName = stickyOnMobile ? 'sticky-top' : '';
|
const stickyClassName = stickyOnMobile ? 'sticky-top' : '';
|
||||||
@@ -99,7 +103,7 @@ class MobileHeader extends React.Component {
|
|||||||
className={`site-header-mobile d-flex justify-content-between align-items-center shadow ${stickyClassName}`}
|
className={`site-header-mobile d-flex justify-content-between align-items-center shadow ${stickyClassName}`}
|
||||||
>
|
>
|
||||||
<div className="w-100 d-flex justify-content-start">
|
<div className="w-100 d-flex justify-content-start">
|
||||||
{mainMenu.length > 0 ?
|
{mainMenu.length > 0 ? (
|
||||||
<Menu className="position-static">
|
<Menu className="position-static">
|
||||||
<MenuTrigger
|
<MenuTrigger
|
||||||
tag="button"
|
tag="button"
|
||||||
@@ -116,25 +120,28 @@ class MobileHeader extends React.Component {
|
|||||||
>
|
>
|
||||||
{this.renderMainMenu()}
|
{this.renderMainMenu()}
|
||||||
</MenuContent>
|
</MenuContent>
|
||||||
</Menu> : null }
|
</Menu>
|
||||||
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
<div className="w-100 d-flex justify-content-center">
|
<div className="w-100 d-flex justify-content-center">
|
||||||
{ logoDestination === null ? <Logo className="logo" src={logo} alt={logoAltText} /> : <LinkedLogo className="logo" {...logoProps} itemType="http://schema.org/Organization" />}
|
{ logoDestination === null ? <Logo className="logo" src={logo} alt={logoAltText} /> : <LinkedLogo className="logo" {...logoProps} itemType="http://schema.org/Organization" />}
|
||||||
</div>
|
</div>
|
||||||
<div className="w-100 d-flex justify-content-end align-items-center">
|
<div className="w-100 d-flex justify-content-end align-items-center">
|
||||||
<Menu tag="nav" aria-label={intl.formatMessage(messages['header.label.secondary.nav'])} className="position-static">
|
{userMenu.length > 0 || loggedOutItems.length > 0 ? (
|
||||||
<MenuTrigger
|
<Menu tag="nav" aria-label={intl.formatMessage(messages['header.label.secondary.nav'])} className="position-static">
|
||||||
tag="button"
|
<MenuTrigger
|
||||||
className="icon-button"
|
tag="button"
|
||||||
aria-label={intl.formatMessage(messages['header.label.account.menu'])}
|
className="icon-button"
|
||||||
title={intl.formatMessage(messages['header.label.account.menu'])}
|
aria-label={intl.formatMessage(messages['header.label.account.menu'])}
|
||||||
>
|
title={intl.formatMessage(messages['header.label.account.menu'])}
|
||||||
<Avatar size="1.5rem" src={avatar} alt={username} />
|
>
|
||||||
</MenuTrigger>
|
<Avatar size="1.5rem" src={avatar} alt={username} />
|
||||||
<MenuContent tag="ul" className="nav flex-column pin-left pin-right border-top shadow py-2">
|
</MenuTrigger>
|
||||||
{loggedIn ? this.renderUserMenuItems() : this.renderLoggedOutItems()}
|
<MenuContent tag="ul" className="nav flex-column pin-left pin-right border-top shadow py-2">
|
||||||
</MenuContent>
|
{loggedIn ? this.renderUserMenuItems() : this.renderLoggedOutItems()}
|
||||||
</Menu>
|
</MenuContent>
|
||||||
|
</Menu>
|
||||||
|
) : null}
|
||||||
</div>
|
</div>
|
||||||
</header>
|
</header>
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -17,7 +17,7 @@ exports[`<Header /> renders correctly for anonymous desktop 1`] = `
|
|||||||
<img
|
<img
|
||||||
alt="edX"
|
alt="edX"
|
||||||
className="d-block"
|
className="d-block"
|
||||||
src="icon/mock/path"
|
src="https://edx-cdn.org/v3/default/logo.svg"
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
<nav
|
<nav
|
||||||
@@ -126,7 +126,7 @@ exports[`<Header /> renders correctly for anonymous mobile 1`] = `
|
|||||||
<img
|
<img
|
||||||
alt="edX"
|
alt="edX"
|
||||||
className="d-block"
|
className="d-block"
|
||||||
src="icon/mock/path"
|
src="https://edx-cdn.org/v3/default/logo.svg"
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
@@ -159,7 +159,6 @@ exports[`<Header /> renders correctly for anonymous mobile 1`] = `
|
|||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
aria-hidden={true}
|
aria-hidden={true}
|
||||||
className="text-muted"
|
|
||||||
focusable="false"
|
focusable="false"
|
||||||
height="24px"
|
height="24px"
|
||||||
role="img"
|
role="img"
|
||||||
@@ -202,7 +201,7 @@ exports[`<Header /> renders correctly for authenticated desktop 1`] = `
|
|||||||
<img
|
<img
|
||||||
alt="edX"
|
alt="edX"
|
||||||
className="d-block"
|
className="d-block"
|
||||||
src="icon/mock/path"
|
src="https://edx-cdn.org/v3/default/logo.svg"
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
<nav
|
<nav
|
||||||
@@ -230,7 +229,7 @@ exports[`<Header /> renders correctly for authenticated desktop 1`] = `
|
|||||||
aria-expanded={false}
|
aria-expanded={false}
|
||||||
aria-haspopup="menu"
|
aria-haspopup="menu"
|
||||||
aria-label="Account menu for edX"
|
aria-label="Account menu for edX"
|
||||||
className="menu-trigger btn btn-light d-inline-flex align-items-center pl-2 pr-3"
|
className="menu-trigger btn btn-outline-primary d-inline-flex align-items-center pl-2 pr-3"
|
||||||
onClick={[Function]}
|
onClick={[Function]}
|
||||||
>
|
>
|
||||||
<span
|
<span
|
||||||
@@ -244,7 +243,6 @@ exports[`<Header /> renders correctly for authenticated desktop 1`] = `
|
|||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
aria-hidden={true}
|
aria-hidden={true}
|
||||||
className="text-muted"
|
|
||||||
focusable="false"
|
focusable="false"
|
||||||
height="24px"
|
height="24px"
|
||||||
role="img"
|
role="img"
|
||||||
@@ -362,7 +360,7 @@ exports[`<Header /> renders correctly for authenticated mobile 1`] = `
|
|||||||
<img
|
<img
|
||||||
alt="edX"
|
alt="edX"
|
||||||
className="d-block"
|
className="d-block"
|
||||||
src="icon/mock/path"
|
src="https://edx-cdn.org/v3/default/logo.svg"
|
||||||
/>
|
/>
|
||||||
</a>
|
</a>
|
||||||
</div>
|
</div>
|
||||||
@@ -395,7 +393,6 @@ exports[`<Header /> renders correctly for authenticated mobile 1`] = `
|
|||||||
>
|
>
|
||||||
<svg
|
<svg
|
||||||
aria-hidden={true}
|
aria-hidden={true}
|
||||||
className="text-muted"
|
|
||||||
focusable="false"
|
focusable="false"
|
||||||
height="24px"
|
height="24px"
|
||||||
role="img"
|
role="img"
|
||||||
|
|||||||
@@ -32,6 +32,8 @@ $white: #fff;
|
|||||||
}
|
}
|
||||||
|
|
||||||
.site-header-mobile {
|
.site-header-mobile {
|
||||||
|
height: 3rem;
|
||||||
|
|
||||||
.nav-link {
|
.nav-link {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
@@ -43,7 +45,6 @@ $white: #fff;
|
|||||||
|
|
||||||
|
|
||||||
.site-header-desktop {
|
.site-header-desktop {
|
||||||
height: 3.75rem;
|
|
||||||
box-shadow: 0 1px 0 0 rgba(0,0,0,.1);
|
box-shadow: 0 1px 0 0 rgba(0,0,0,.1);
|
||||||
background: $white;
|
background: $white;
|
||||||
.nav-link {
|
.nav-link {
|
||||||
|
|||||||
File diff suppressed because one or more lines are too long
|
Before Width: | Height: | Size: 17 KiB |
@@ -23,3 +23,7 @@ process.env.REFRESH_ACCESS_TOKEN_ENDPOINT = 'http://localhost:18000/login_refres
|
|||||||
process.env.SEGMENT_KEY = 'segment_whoa';
|
process.env.SEGMENT_KEY = 'segment_whoa';
|
||||||
process.env.SITE_NAME = 'edX';
|
process.env.SITE_NAME = 'edX';
|
||||||
process.env.USER_INFO_COOKIE_NAME = 'edx-user-info';
|
process.env.USER_INFO_COOKIE_NAME = 'edx-user-info';
|
||||||
|
process.env.LOGO_URL = 'https://edx-cdn.org/v3/default/logo.svg';
|
||||||
|
process.env.LOGO_TRADEMARK_URL = 'https://edx-cdn.org/v3/default/logo-trademark.svg';
|
||||||
|
process.env.LOGO_WHITE_URL = 'https://edx-cdn.org/v3/default/logo-white.svg';
|
||||||
|
process.env.FAVICON_URL = 'https://edx-cdn.org/v3/default/favicon.ico';
|
||||||
|
|||||||
@@ -7,4 +7,9 @@ module.exports = createConfig('webpack-dev', {
|
|||||||
path: path.resolve(__dirname, 'example/dist'),
|
path: path.resolve(__dirname, 'example/dist'),
|
||||||
publicPath: '/',
|
publicPath: '/',
|
||||||
},
|
},
|
||||||
|
resolve: {
|
||||||
|
alias: {
|
||||||
|
'@edx/frontend-component-header': path.resolve(__dirname, 'src'),
|
||||||
|
},
|
||||||
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
Reference in New Issue
Block a user