-
Notifications
You must be signed in to change notification settings - Fork 354
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat(wizard): adds the pf wizard and modal components
- Loading branch information
Showing
28 changed files
with
2,105 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,3 +1,31 @@ | ||
/** | ||
Patternfly React Specific Extensions | ||
*/ | ||
|
||
.wizard-pf-header { | ||
background-color: #f5f5f5; | ||
border-bottom: none; | ||
padding: 10px 18px; | ||
} | ||
.wizard-pf-title { | ||
font-size: 13px; | ||
font-weight: 700; | ||
} | ||
.wizard-pf-footer { | ||
padding: 14px 15px 17px; | ||
text-align: right; | ||
& .btn + .btn { | ||
margin-left: 5px; | ||
margin-bottom: 0; | ||
} | ||
& > .btn { | ||
padding-left: 10px; | ||
padding-right: 10px; | ||
& > .fa-angle-left { | ||
margin-right: 5px; | ||
} | ||
& > .fa-angle-right { | ||
margin-left: 5px; | ||
} | ||
} | ||
} |
Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.
Oops, something went wrong.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import WizardHeader from './WizardHeader'; | ||
|
||
/** | ||
* Wizard - main Wizard component. | ||
*/ | ||
const Wizard = ({ children, className, embedded, ...rest }) => { | ||
const renderChildren = () => { | ||
return React.Children.map(children, child => { | ||
if (child && child.type === WizardHeader) { | ||
return React.cloneElement(child, { | ||
embedded: embedded, | ||
}); | ||
} else { | ||
return child; | ||
} | ||
}); | ||
}; | ||
|
||
return ( | ||
<div className={className} {...rest}> | ||
{renderChildren()} | ||
</div> | ||
); | ||
}; | ||
Wizard.propTypes = { | ||
/** Children nodes */ | ||
children: PropTypes.node, | ||
/** Additional css classes */ | ||
className: PropTypes.string, | ||
/** Embedded wizard */ | ||
embedded: PropTypes.bool, | ||
}; | ||
export default Wizard; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,93 @@ | ||
import React from 'react'; | ||
import { storiesOf } from '@storybook/react'; | ||
import { withInfo } from '@storybook/addon-info'; | ||
import { Row, Col } from 'react-bootstrap'; | ||
import { withKnobs } from '@storybook/addon-knobs'; | ||
import { defaultTemplate } from '../../../storybook/decorators/storyTemplates'; | ||
|
||
import { mockWizardItems } from './__mocks__/mockWizardItems'; | ||
|
||
import { | ||
MockLoadingWizardManager, | ||
mockLoadingWizardSource, | ||
} from './__mocks__/mockLoadingWizardManager'; | ||
|
||
import { | ||
MockModalWizardManager, | ||
mockModalWizardSource, | ||
} from './__mocks__/mockModalWizardManager'; | ||
|
||
import { | ||
MockEmbeddedWizardManager, | ||
mockEmbeddedWizardSource, | ||
} from './__mocks__/mockEmbeddedWizardManager'; | ||
|
||
const stories = storiesOf('Wizard', module); | ||
stories.addDecorator(withKnobs); | ||
stories.addDecorator( | ||
defaultTemplate({ | ||
title: 'Wizard', | ||
documentationLink: | ||
'http://www.patternfly.org/pattern-library/communication/wizard/#/overview', | ||
}), | ||
); | ||
|
||
stories.add( | ||
'Loading wizard', | ||
withInfo({ | ||
source: false, | ||
propTablesExclude: [Row, Col, MockLoadingWizardManager], | ||
text: ( | ||
<div> | ||
<h1>Story Source</h1> | ||
<pre>{mockLoadingWizardSource}</pre> | ||
</div> | ||
), | ||
})(() => ( | ||
<Row> | ||
<Col sm={12}> | ||
<MockLoadingWizardManager steps={mockWizardItems} /> | ||
</Col> | ||
</Row> | ||
)), | ||
); | ||
|
||
stories.add( | ||
'Modal wizard', | ||
withInfo({ | ||
source: false, | ||
propTablesExclude: [Row, Col, MockModalWizardManager], | ||
text: ( | ||
<div> | ||
<h1>Story Source</h1> | ||
<pre>{mockModalWizardSource}</pre> | ||
</div> | ||
), | ||
})(() => ( | ||
<Row> | ||
<Col sm={12}> | ||
<MockModalWizardManager steps={mockWizardItems} /> | ||
</Col> | ||
</Row> | ||
)), | ||
); | ||
|
||
stories.add( | ||
'Embedded in page', | ||
withInfo({ | ||
source: false, | ||
propTablesExclude: [Row, Col, MockEmbeddedWizardManager], | ||
text: ( | ||
<div> | ||
<h1>Story Source</h1> | ||
<pre>{mockEmbeddedWizardSource}</pre> | ||
</div> | ||
), | ||
})(() => ( | ||
<Row> | ||
<Col sm={12}> | ||
<MockEmbeddedWizardManager steps={mockWizardItems} /> | ||
</Col> | ||
</Row> | ||
)), | ||
); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,99 @@ | ||
import React from 'react'; | ||
import renderer from 'react-test-renderer'; | ||
import { Row, Col } from 'react-bootstrap'; | ||
import { Button } from '../Button'; | ||
import { Wizard } from './index'; | ||
|
||
import { | ||
mockWizardItems, | ||
mockLoadingContents, | ||
} from './__mocks__/mockWizardItems'; | ||
|
||
import { | ||
renderWizardSteps, | ||
renderSidebarItems, | ||
renderWizardContents, | ||
} from './__mocks__/mockWizardRenderers'; | ||
|
||
test('Wizard loading renders properly', () => { | ||
const component = renderer.create( | ||
<Row> | ||
<Col sm={12}> | ||
<Wizard> | ||
<Wizard.Header title="Wizard Title" /> | ||
<Wizard.Body> | ||
<Wizard.Row> | ||
<Wizard.Main>{mockLoadingContents()}</Wizard.Main> | ||
</Wizard.Row> | ||
</Wizard.Body> | ||
<Wizard.Footer> | ||
<Button bsStyle="default" className="btn-cancel"> | ||
Cancel | ||
</Button> | ||
<Button bsStyle="default" disabled> | ||
<span className="i fa fa-angle-left" />Back | ||
</Button> | ||
<Button bsStyle="primary" disabled> | ||
Next<span className="i fa fa-angle-right" /> | ||
</Button> | ||
</Wizard.Footer> | ||
</Wizard> | ||
</Col> | ||
</Row>, | ||
); | ||
|
||
const tree = component.toJSON(); | ||
expect(tree).toMatchSnapshot(); | ||
}); | ||
|
||
test('Wizard embedded renders properly', () => { | ||
const onStepClick = jest.fn(); | ||
const onSidebarItemClick = jest.fn(); | ||
const activeStepIndex = 0; | ||
const activeSubStepIndex = 0; | ||
const onNextButtonClick = jest.fn(); | ||
const onBackButtonClick = jest.fn(); | ||
|
||
const component = renderer.create( | ||
<Wizard embedded> | ||
<Wizard.Header title="Wizard Title" /> | ||
<Wizard.Body> | ||
<Wizard.Steps | ||
steps={renderWizardSteps(mockWizardItems, 0, 0, onStepClick)} | ||
/> | ||
<Wizard.Row> | ||
<Wizard.Sidebar | ||
items={renderSidebarItems( | ||
mockWizardItems, | ||
activeStepIndex, | ||
activeSubStepIndex, | ||
onSidebarItemClick, | ||
)} | ||
/> | ||
<Wizard.Main> | ||
{renderWizardContents( | ||
mockWizardItems, | ||
activeStepIndex, | ||
activeSubStepIndex, | ||
)} | ||
</Wizard.Main> | ||
</Wizard.Row> | ||
</Wizard.Body> | ||
<Wizard.Footer> | ||
<Button bsStyle="default" className="btn-cancel"> | ||
Cancel | ||
</Button> | ||
<Button bsStyle="default" onClick={onBackButtonClick}> | ||
<span className="i fa fa-angle-left" />Back | ||
</Button> | ||
|
||
<Button bsStyle="primary" onClick={onNextButtonClick}> | ||
Next<span className="i fa fa-angle-right" /> | ||
</Button> | ||
</Wizard.Footer> | ||
</Wizard>, | ||
); | ||
|
||
const tree = component.toJSON(); | ||
expect(tree).toMatchSnapshot(); | ||
}); |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import cx from 'classnames'; | ||
|
||
/** | ||
* WizardBody component for Patternfly React | ||
*/ | ||
const WizardBody = ({ children, className, ...rest }) => { | ||
const classes = cx('wizard-pf-body', className); | ||
return ( | ||
<div className={classes} {...rest}> | ||
{children} | ||
</div> | ||
); | ||
}; | ||
WizardBody.propTypes = { | ||
/** Children nodes */ | ||
children: PropTypes.node, | ||
/** Additional css classes */ | ||
className: PropTypes.string, | ||
}; | ||
export default WizardBody; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,48 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import cx from 'classnames'; | ||
|
||
/** | ||
* WizardContents component for Patternfly React | ||
*/ | ||
const WizardContents = ({ | ||
children, | ||
className, | ||
stepIndex, | ||
subStepIndex, | ||
activeStepIndex, | ||
activeSubStepIndex, | ||
...rest | ||
}) => { | ||
const classes = cx( | ||
'wizard-pf-contents', | ||
{ | ||
// hide contents if the step is not active | ||
// OR if we have sub steps and this sub step is not active | ||
hidden: | ||
activeStepIndex !== stepIndex || | ||
(activeSubStepIndex !== null && activeSubStepIndex !== subStepIndex), | ||
}, | ||
className, | ||
); | ||
return ( | ||
<div className={classes} {...rest}> | ||
{children} | ||
</div> | ||
); | ||
}; | ||
WizardContents.propTypes = { | ||
/** WizardStep nodes */ | ||
children: PropTypes.node, | ||
/** Additional css classes */ | ||
className: PropTypes.string, | ||
/** The wizard step index for these contents */ | ||
stepIndex: PropTypes.number, | ||
/** The wizard sub step index for these contents */ | ||
subStepIndex: PropTypes.number, | ||
/** The active wizard step index */ | ||
activeStepIndex: PropTypes.number, | ||
/** The active wizard sub step index */ | ||
activeSubStepIndex: PropTypes.number, | ||
}; | ||
export default WizardContents; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import cx from 'classnames'; | ||
|
||
/** | ||
* WizardFooter component for Patternfly React | ||
*/ | ||
const WizardFooter = ({ children, className, ...rest }) => { | ||
const classes = cx('wizard-pf-footer', className); | ||
return ( | ||
<div className={classes} {...rest}> | ||
{children} | ||
</div> | ||
); | ||
}; | ||
WizardFooter.propTypes = { | ||
/** Children nodes */ | ||
children: PropTypes.node, | ||
/** Additional css classes */ | ||
className: PropTypes.string, | ||
}; | ||
export default WizardFooter; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,35 @@ | ||
import React from 'react'; | ||
import PropTypes from 'prop-types'; | ||
import cx from 'classnames'; | ||
|
||
/** | ||
* WizardHeader component for Patternfly React | ||
*/ | ||
const WizardHeader = ({ children, className, embedded, title, ...rest }) => { | ||
const classes = cx({ 'wizard-pf-header': !embedded }, className); | ||
|
||
if (embedded) { | ||
return ( | ||
<h2 className={classes} {...rest}> | ||
{title} | ||
</h2> | ||
); | ||
} else { | ||
return ( | ||
<div className={classes} {...rest}> | ||
<h4 className="wizard-pf-title">{title}</h4> | ||
</div> | ||
); | ||
} | ||
}; | ||
WizardHeader.propTypes = { | ||
/** Children nodes */ | ||
children: PropTypes.node, | ||
/** Additional css classes */ | ||
className: PropTypes.string, | ||
/** Embedded wizard */ | ||
embedded: PropTypes.bool, | ||
/** The wizard title */ | ||
title: PropTypes.string, | ||
}; | ||
export default WizardHeader; |
Oops, something went wrong.