Merge pull request #198 from jlmitch5/translationUpdate

utilize i18n correctly
This commit is contained in:
John Mitchell
2019-05-16 13:54:05 -04:00
committed by GitHub
98 changed files with 5818 additions and 4917 deletions

View File

@@ -20,6 +20,7 @@ Have questions about this document or anything not covered here? Feel free to re
* [Typechecking with PropTypes](#typechecking-with-proptypes) * [Typechecking with PropTypes](#typechecking-with-proptypes)
* [Naming Functions](#naming-functions) * [Naming Functions](#naming-functions)
* [Default State Initialization](#default-state-initialization) * [Default State Initialization](#default-state-initialization)
* [Internationalization](#internationalization)
## Things to know prior to submitting code ## Things to know prior to submitting code
@@ -214,3 +215,30 @@ mountWithContexts(<Organization />< {
} }
}); });
``` ```
## Internationalization
Internationalization leans on the [lingui](https://github.com/lingui/js-lingui) project. [Official documentation here](https://lingui.js.org/). We use this libary to mark our strings for translation. If you want to see this in action you'll need to take the following steps:
### Marking strings for translation and replacement in the UI
The lingui library provides various React helpers for dealing with both marking strings for translation, and replacing strings that have been traslated. For consistency and ease of use, we have consolidated on one pattern for the codebase. To set strings to be translated in the UI:
- import the withI18n function and wrap the export of your component in it (i.e. `export default withI18n()(Foo)`)
- doing the above gives you access to the i18n object on props. Make sure to put it in the scope of the function that contains strings needed to be translated (i.e. `const { i18n } = this.props;`)
- import the t template tag function from the @lingui/macro package.
- wrap your string using the following format: ```i18n._(t`String to be translated`)```
**Note:** Variables that are put inside the t-marked template tag will not be translated. If you have a variable string with text that needs translating, you must wrap it in ```i18n._(t``)``` where it is defined.
**Note:** We do not use the `I18n` consumer, `i18nMark` function, or `<Trans>` component lingui gives us access to in this repo. i18nMark does not actually replace the string in the UI (leading to the potential for untranslated bugs), and the other helpers are redundant. Settling on a consistent, single pattern helps us ease the mental overhead of the need to understand the ins and outs of the lingui API.
You can learn more about the ways lingui and its React helpers at [this link](https://lingui.js.org/tutorials/react-patterns.html).
### Setting up .po files to give to translation team
1) `npm run add-locale` to add the language that you want to translate to (we should only have to do this once and the commit to repo afaik). Example: `npm run add-locale en es fr` # Add English, Spanish and French locale
2) `npm run extract-strings` to create .po files for each language specified. The .po files will be placed in src/locales but this is configurable.
3) Open up the .po file for the language you want to test and add some translations. In production we would pass this .po file off to the translation team.
4) Once you've edited your .po file (or we've gotten a .po file back from the translation team) run `npm run compile-strings`. This command takes the .po files and turns them into a minified JSON object and can be seen in the `messages.js` file in each locale directory. These files get loaded at the App root level (see: App.jsx).
5) Change the language in your browser and reload the page. You should see your specified translations in place of English strings.

View File

@@ -27,13 +27,3 @@ To run a single test (in this case the login page test):
* `npm test -- __tests__/pages/Login.jsx` * `npm test -- __tests__/pages/Login.jsx`
**note:** Once the test watcher is up and running you can hit `a` to run all the tests **note:** Once the test watcher is up and running you can hit `a` to run all the tests
## Internationalization
Internationalization leans on the [lingui](https://github.com/lingui/js-lingui) project. [Official documentation here](https://lingui.js.org/). We use this libary to mark our strings for translation. For common React use cases see [this link](https://lingui.js.org/tutorials/react-patterns.html). If you want to see this in action you'll need to take the following steps:
1) `npm run add-locale` to add the language that you want to translate to (we should only have to do this once and the commit to repo afaik). Example: `npm run add-locale en es fr` # Add English, Spanish and French locale
2) `npm run extract-strings` to create .po files for each language specified. The .po files will be placed in src/locales but this is configurable.
3) Open up the .po file for the language you want to test and add some translations. In production we would pass this .po file off to the translation team.
4) Once you've edited your .po file (or we've gotten a .po file back from the translation team) run `npm run compile-strings`. This command takes the .po files and turns them into a minified JSON object and can be seen in the `messages.js` file in each locale directory. These files get loaded at the App root level (see: App.jsx).
5) Change the language in your browser and reload the page. You should see your specified translations in place of English strings.

View File

@@ -21,29 +21,28 @@ exports[`mountWithContexts injected ConfigProvider should mount and render with
exports[`mountWithContexts injected I18nProvider should mount and render 1`] = ` exports[`mountWithContexts injected I18nProvider should mount and render 1`] = `
<div> <div>
<I18n <span>
update={true} Text content
withHash={true} </span>
>
<span>
Text content
</span>
</I18n>
</div> </div>
`; `;
exports[`mountWithContexts injected I18nProvider should mount and render deeply nested consumer 1`] = ` exports[`mountWithContexts injected I18nProvider should mount and render deeply nested consumer 1`] = `
<Parent> <Parent>
<Child> <WithI18n>
<I18n <I18n
update={true} update={true}
withHash={true} withHash={true}
> >
<div> <Component
Text content i18n={"/i18n/"}
</div> >
<div>
Text content
</div>
</Component>
</I18n> </I18n>
</Child> </WithI18n>
</Parent> </Parent>
`; `;

View File

@@ -1,9 +1,9 @@
import React from 'react'; import React from 'react';
import { shallow } from 'enzyme'; import { shallow } from 'enzyme';
import { mountWithContexts } from '../enzymeHelpers'; import { mountWithContexts } from '../enzymeHelpers';
import AddResourceRole from '../../src/components/AddRole/AddResourceRole'; import AddResourceRole, { _AddResourceRole } from '../../src/components/AddRole/AddResourceRole';
describe('<AddResourceRole />', () => { describe('<_AddResourceRole />', () => {
const readUsers = jest.fn().mockResolvedValue({ const readUsers = jest.fn().mockResolvedValue({
data: { data: {
count: 2, count: 2,
@@ -31,21 +31,23 @@ describe('<AddResourceRole />', () => {
}; };
test('initially renders without crashing', () => { test('initially renders without crashing', () => {
shallow( shallow(
<AddResourceRole <_AddResourceRole
api={api} api={api}
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
roles={roles} roles={roles}
i18n={{ _: val => val.toString() }}
/> />
); );
}); });
test('handleRoleCheckboxClick properly updates state', () => { test('handleRoleCheckboxClick properly updates state', () => {
const wrapper = shallow( const wrapper = shallow(
<AddResourceRole <_AddResourceRole
api={api} api={api}
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
roles={roles} roles={roles}
i18n={{ _: val => val.toString() }}
/> />
); );
wrapper.setState({ wrapper.setState({
@@ -76,11 +78,12 @@ describe('<AddResourceRole />', () => {
}); });
test('handleResourceCheckboxClick properly updates state', () => { test('handleResourceCheckboxClick properly updates state', () => {
const wrapper = shallow( const wrapper = shallow(
<AddResourceRole <_AddResourceRole
api={api} api={api}
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
roles={roles} roles={roles}
i18n={{ _: val => val.toString() }}
/> />
); );
wrapper.setState({ wrapper.setState({
@@ -106,14 +109,13 @@ describe('<AddResourceRole />', () => {
}]); }]);
}); });
test('clicking user/team cards updates state', () => { test('clicking user/team cards updates state', () => {
const spy = jest.spyOn(AddResourceRole.prototype, 'handleResourceSelect'); const spy = jest.spyOn(_AddResourceRole.prototype, 'handleResourceSelect');
const wrapper = mountWithContexts( const wrapper = mountWithContexts(
<AddResourceRole <AddResourceRole
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
api={api}
roles={roles} roles={roles}
/> />, { context: { network: { api, handleHttpError: () => {} } } }
).find('AddResourceRole'); ).find('AddResourceRole');
const selectableCardWrapper = wrapper.find('SelectableCard'); const selectableCardWrapper = wrapper.find('SelectableCard');
expect(selectableCardWrapper.length).toBe(2); expect(selectableCardWrapper.length).toBe(2);
@@ -126,11 +128,12 @@ describe('<AddResourceRole />', () => {
}); });
test('readUsers and readTeams call out to corresponding api functions', () => { test('readUsers and readTeams call out to corresponding api functions', () => {
const wrapper = shallow( const wrapper = shallow(
<AddResourceRole <_AddResourceRole
api={api} api={api}
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
roles={roles} roles={roles}
i18n={{ _: val => val.toString() }}
/> />
); );
wrapper.instance().readUsers({ wrapper.instance().readUsers({
@@ -150,11 +153,12 @@ describe('<AddResourceRole />', () => {
test('handleResourceSelect clears out selected lists and sets selectedResource', () => { test('handleResourceSelect clears out selected lists and sets selectedResource', () => {
const wrapper = shallow( const wrapper = shallow(
<AddResourceRole <_AddResourceRole
api={api} api={api}
onClose={() => {}} onClose={() => {}}
onSave={() => {}} onSave={() => {}}
roles={roles} roles={roles}
i18n={{ _: val => val.toString() }}
/> />
); );
wrapper.setState({ wrapper.setState({
@@ -193,11 +197,10 @@ describe('<AddResourceRole />', () => {
const handleSave = jest.fn(); const handleSave = jest.fn();
const wrapper = mountWithContexts( const wrapper = mountWithContexts(
<AddResourceRole <AddResourceRole
api={api}
onClose={() => {}} onClose={() => {}}
onSave={handleSave} onSave={handleSave}
roles={roles} roles={roles}
/> />, { context: { network: { api, handleHttpError: () => {} } } }
).find('AddResourceRole'); ).find('AddResourceRole');
wrapper.setState({ wrapper.setState({
selectedResource: 'users', selectedResource: 'users',

View File

@@ -1,6 +1,7 @@
import React from 'react'; import React from 'react';
import { mountWithContexts } from '../enzymeHelpers'; import { mountWithContexts } from '../enzymeHelpers';
import AnsibleSelect from '../../src/components/AnsibleSelect'; import AnsibleSelect from '../../src/components/AnsibleSelect';
import { _AnsibleSelect } from '../../src/components/AnsibleSelect/AnsibleSelect';
const label = 'test select'; const label = 'test select';
const mockData = ['/venv/baz/', '/venv/ansible/']; const mockData = ['/venv/baz/', '/venv/ansible/'];
@@ -18,7 +19,7 @@ describe('<AnsibleSelect />', () => {
}); });
test('calls "onSelectChange" on dropdown select change', () => { test('calls "onSelectChange" on dropdown select change', () => {
const spy = jest.spyOn(AnsibleSelect.prototype, 'onSelectChange'); const spy = jest.spyOn(_AnsibleSelect.prototype, 'onSelectChange');
const wrapper = mountWithContexts( const wrapper = mountWithContexts(
<AnsibleSelect <AnsibleSelect
value="foo" value="foo"

View File

@@ -216,6 +216,7 @@ describe('<Lookup />', () => {
getItems={getItems} getItems={getItems}
handleHttpError={() => {}} handleHttpError={() => {}}
location={{ history }} location={{ history }}
i18n={{ _: val => val.toString() }}
/> />
); );

View File

@@ -11,6 +11,7 @@ describe('<NotifyAndRedirect />', () => {
to="foo" to="foo"
setRootDialogMessage={setRootDialogMessage} setRootDialogMessage={setRootDialogMessage}
location={{ pathname: 'foo' }} location={{ pathname: 'foo' }}
i18n={{ _: val => val.toString() }}
/> />
); );
expect(setRootDialogMessage).toHaveBeenCalled(); expect(setRootDialogMessage).toHaveBeenCalled();

View File

@@ -2,202 +2,202 @@
exports[`<ToolbarDeleteButton /> should render button 1`] = ` exports[`<ToolbarDeleteButton /> should render button 1`] = `
<ToolbarDeleteButton <ToolbarDeleteButton
i18n={"/i18n/"}
itemName="item" itemName="item"
itemsToDelete={Array []} itemsToDelete={Array []}
onDelete={[Function]} onDelete={[Function]}
> >
<I18n <Tooltip
update={true} appendTo={[Function]}
withHash={true} className={null}
content="Select a row to delete"
enableFlip={true}
entryDelay={500}
exitDelay={500}
maxWidth="18.75rem"
position="left"
trigger="mouseenter focus"
zIndex={9999}
> >
<Tooltip <Tippy
animateFill={false}
appendTo={[Function]} appendTo={[Function]}
className={null} content={
content="Select a row to delete" <div
enableFlip={true} className="pf-c-tooltip"
entryDelay={500} role="tooltip"
exitDelay={500} >
<TooltipArrow
className={null}
/>
<TooltipContent
className={null}
>
Select a row to delete
</TooltipContent>
</div>
}
delay={
Array [
500,
500,
]
}
distance={15}
flip={true}
lazy={true}
maxWidth="18.75rem" maxWidth="18.75rem"
position="left" onCreate={[Function]}
performance={true}
placement="left"
popperOptions={
Object {
"modifiers": Object {
"hide": Object {
"enabled": true,
},
"preventOverflow": Object {
"enabled": true,
},
},
}
}
theme="pf-tippy"
trigger="mouseenter focus" trigger="mouseenter focus"
zIndex={9999} zIndex={9999}
> >
<Tippy <ToolbarDeleteButton__Button
animateFill={false} aria-label="Delete"
appendTo={[Function]} className="awx-ToolBarBtn"
content={ isDisabled={true}
<div onClick={[Function]}
className="pf-c-tooltip" variant="plain"
role="tooltip"
>
<TooltipArrow
className={null}
/>
<TooltipContent
className={null}
>
Select a row to delete
</TooltipContent>
</div>
}
delay={
Array [
500,
500,
]
}
distance={15}
flip={true}
lazy={true}
maxWidth="18.75rem"
onCreate={[Function]}
performance={true}
placement="left"
popperOptions={
Object {
"modifiers": Object {
"hide": Object {
"enabled": true,
},
"preventOverflow": Object {
"enabled": true,
},
},
}
}
theme="pf-tippy"
trigger="mouseenter focus"
zIndex={9999}
> >
<ToolbarDeleteButton__Button <StyledComponent
aria-label="Delete" aria-label="Delete"
className="awx-ToolBarBtn"
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "ToolbarDeleteButton__Button-sc-1e3r0eg-0",
"isStatic": true,
"lastClassName": "iyjqWq",
"rules": Array [
"width:30px;height:30px;display:flex;justify-content:center;margin-right:20px;border-radius:3px;padding:0;&:disabled{cursor:not-allowed;&:hover{background-color:white;> svg{color:#d2d2d2;}}}&:hover{background-color:#d9534f;> svg{color:white;}}",
],
},
"displayName": "ToolbarDeleteButton__Button",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "ToolbarDeleteButton__Button-sc-1e3r0eg-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
isDisabled={true} isDisabled={true}
onClick={[Function]} onClick={[Function]}
variant="plain" variant="plain"
> >
<StyledComponent <Button
aria-label="Delete" aria-label="Delete"
forwardedComponent={ className="awx-ToolBarBtn ToolbarDeleteButton__Button-sc-1e3r0eg-0 iyjqWq"
Object { component="button"
"$$typeof": Symbol(react.forward_ref), isActive={false}
"attrs": Array [], isBlock={false}
"componentStyle": ComponentStyle {
"componentId": "ToolbarDeleteButton__Button-sc-1e3r0eg-0",
"isStatic": true,
"lastClassName": "iyjqWq",
"rules": Array [
"width:30px;height:30px;display:flex;justify-content:center;margin-right:20px;border-radius:3px;padding:0;&:disabled{cursor:not-allowed;&:hover{background-color:white;> svg{color:#d2d2d2;}}}&:hover{background-color:#d9534f;> svg{color:white;}}",
],
},
"displayName": "ToolbarDeleteButton__Button",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "ToolbarDeleteButton__Button-sc-1e3r0eg-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
isDisabled={true} isDisabled={true}
isFocus={false}
isHover={false}
onClick={[Function]} onClick={[Function]}
type="button"
variant="plain" variant="plain"
> >
<Button <button
aria-disabled={null}
aria-label="Delete" aria-label="Delete"
className="ToolbarDeleteButton__Button-sc-1e3r0eg-0 iyjqWq" className="pf-c-button pf-m-plain pf-m-disabled awx-ToolBarBtn ToolbarDeleteButton__Button-sc-1e3r0eg-0 iyjqWq"
component="button" disabled={true}
isActive={false}
isBlock={false}
isDisabled={true}
isFocus={false}
isHover={false}
onClick={[Function]} onClick={[Function]}
tabIndex={null}
type="button" type="button"
variant="plain"
> >
<button <TrashAltIcon
aria-disabled={null} className="awx-ToolBarTrashCanIcon"
aria-label="Delete" color="currentColor"
className="pf-c-button pf-m-plain pf-m-disabled ToolbarDeleteButton__Button-sc-1e3r0eg-0 iyjqWq" size="sm"
disabled={true} title={null}
onClick={[Function]}
tabIndex={null}
type="button"
> >
<TrashAltIcon <svg
color="currentColor" aria-hidden={true}
size="sm" aria-labelledby={null}
title={null} className="awx-ToolBarTrashCanIcon"
> fill="currentColor"
<svg height="1em"
aria-hidden={true} role="img"
aria-labelledby={null} style={
fill="currentColor" Object {
height="1em" "verticalAlign": "-0.125em",
role="img"
style={
Object {
"verticalAlign": "-0.125em",
}
} }
viewBox="0 0 448 512" }
width="1em" viewBox="0 0 448 512"
> width="1em"
<path
d="M32 464a48 48 0 0 0 48 48h288a48 48 0 0 0 48-48V128H32zm272-256a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zm-96 0a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zm-96 0a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zM432 32H312l-9.4-18.7A24 24 0 0 0 281.1 0H166.8a23.72 23.72 0 0 0-21.4 13.3L136 32H16A16 16 0 0 0 0 48v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16V48a16 16 0 0 0-16-16z"
transform=""
/>
</svg>
</TrashAltIcon>
</button>
</Button>
</StyledComponent>
</ToolbarDeleteButton__Button>
<Portal
containerInfo={
<div>
<div
class="pf-c-tooltip"
role="tooltip"
>
<div
class="pf-c-tooltip__arrow"
/>
<div
class="pf-c-tooltip__content"
> >
Select a row to delete <path
</div> d="M32 464a48 48 0 0 0 48 48h288a48 48 0 0 0 48-48V128H32zm272-256a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zm-96 0a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zm-96 0a16 16 0 0 1 32 0v224a16 16 0 0 1-32 0zM432 32H312l-9.4-18.7A24 24 0 0 0 281.1 0H166.8a23.72 23.72 0 0 0-21.4 13.3L136 32H16A16 16 0 0 0 0 48v32a16 16 0 0 0 16 16h416a16 16 0 0 0 16-16V48a16 16 0 0 0-16-16z"
</div> transform=""
</div> />
} </svg>
> </TrashAltIcon>
<div </button>
className="pf-c-tooltip" </Button>
role="tooltip" </StyledComponent>
> </ToolbarDeleteButton__Button>
<TooltipArrow <Portal
className={null} containerInfo={
<div>
<div
class="pf-c-tooltip"
role="tooltip"
> >
<div <div
className="pf-c-tooltip__arrow" class="pf-c-tooltip__arrow"
/> />
</TooltipArrow>
<TooltipContent
className={null}
>
<div <div
className="pf-c-tooltip__content" class="pf-c-tooltip__content"
> >
Select a row to delete Select a row to delete
</div> </div>
</TooltipContent> </div>
</div> </div>
</Portal> }
</Tippy> >
</Tooltip> <div
</I18n> className="pf-c-tooltip"
role="tooltip"
>
<TooltipArrow
className={null}
>
<div
className="pf-c-tooltip__arrow"
/>
</TooltipArrow>
<TooltipContent
className={null}
>
<div
className="pf-c-tooltip__content"
>
Select a row to delete
</div>
</TooltipContent>
</div>
</Portal>
</Tippy>
</Tooltip>
</ToolbarDeleteButton> </ToolbarDeleteButton>
`; `;

View File

@@ -3,7 +3,7 @@ import { createMemoryHistory } from 'history';
import { shallow } from 'enzyme'; import { shallow } from 'enzyme';
import { mountWithContexts } from '../enzymeHelpers'; import { mountWithContexts } from '../enzymeHelpers';
import { sleep } from '../testUtils'; import { sleep } from '../testUtils';
import SelectResourceStep, { _SelectResourceStep } from '../../src/components/AddRole/SelectResourceStep'; import SelectResourceStep from '../../src/components/AddRole/SelectResourceStep';
describe('<SelectResourceStep />', () => { describe('<SelectResourceStep />', () => {
const columns = [ const columns = [
@@ -67,15 +67,14 @@ describe('<SelectResourceStep />', () => {
initialEntries: ['/organizations/1/access?resource.page=1&resource.order_by=-username'], initialEntries: ['/organizations/1/access?resource.page=1&resource.order_by=-username'],
}); });
const wrapper = await mountWithContexts( const wrapper = await mountWithContexts(
<_SelectResourceStep <SelectResourceStep
columns={columns} columns={columns}
displayKey="username" displayKey="username"
onRowClick={() => {}} onRowClick={() => {}}
onSearch={handleSearch} onSearch={handleSearch}
selectedResourceRows={selectedResourceRows} selectedResourceRows={selectedResourceRows}
sortedColumnKey="username" sortedColumnKey="username"
location={history.location} />, { context: { router: { history, route: { location: history.location } } } }
/>
).find('SelectResourceStep'); ).find('SelectResourceStep');
await wrapper.instance().readResourceList(); await wrapper.instance().readResourceList();
expect(handleSearch).toHaveBeenCalledWith({ expect(handleSearch).toHaveBeenCalledWith({

View File

@@ -1,5 +1,6 @@
import React from 'react'; import React from 'react';
import { mount, shallow } from 'enzyme'; import { shallow } from 'enzyme';
import { mountWithContexts } from '../enzymeHelpers';
import SelectRoleStep from '../../src/components/AddRole/SelectRoleStep'; import SelectRoleStep from '../../src/components/AddRole/SelectRoleStep';
describe('<SelectRoleStep />', () => { describe('<SelectRoleStep />', () => {
@@ -42,7 +43,7 @@ describe('<SelectRoleStep />', () => {
}); });
test('clicking role fires onRolesClick callback', () => { test('clicking role fires onRolesClick callback', () => {
const onRolesClick = jest.fn(); const onRolesClick = jest.fn();
wrapper = mount( wrapper = mountWithContexts(
<SelectRoleStep <SelectRoleStep
onRolesClick={onRolesClick} onRolesClick={onRolesClick}
roles={roles} roles={roles}

View File

@@ -5,6 +5,7 @@ exports[`<NotificationListItem canToggleNotifications /> initially renders succe
canToggleNotifications={true} canToggleNotifications={true}
detailUrl="/foo" detailUrl="/foo"
errorTurnedOn={false} errorTurnedOn={false}
i18n={"/i18n/"}
notification={ notification={
Object { Object {
"id": 9000, "id": 9000,
@@ -15,446 +16,526 @@ exports[`<NotificationListItem canToggleNotifications /> initially renders succe
successTurnedOn={false} successTurnedOn={false}
toggleNotification={[MockFunction]} toggleNotification={[MockFunction]}
> >
<I18n <DataListItem
update={true} aria-labelledby="items-list-item-9000"
withHash={true} className=""
isExpanded={false}
key="9000"
> >
<DataListItem <li
aria-labelledby="items-list-item-9000" aria-labelledby="items-list-item-9000"
className="" className="pf-c-data-list__item"
isExpanded={false}
key="9000"
> >
<li <DataListItemRow
aria-labelledby="items-list-item-9000" className=""
className="pf-c-data-list__item" key=".0"
rowid="items-list-item-9000"
> >
<DataListItemRow <div
className="" className="pf-c-data-list__item-row"
key=".0"
rowid="items-list-item-9000"
> >
<div <DataListItemCells
className="pf-c-data-list__item-row" className=""
> dataListCells={
<DataListItemCells Array [
className="" <ForwardRef>
dataListCells={
Array [
<ForwardRef>
<ForwardRef
to={
Object {
"pathname": "/foo",
}
}
>
<b
id="items-list-item-9000"
>
Foo
</b>
</ForwardRef>
<ForwardRef
isRead={true}
>
slack
</ForwardRef>
</ForwardRef>,
<ForwardRef <ForwardRef
righthalf="true" to={
>
<ForwardRef
aria-label="Toggle notification success"
id="notification-9000-success-toggle"
isChecked={false}
isDisabled={false}
label="Successful"
onChange={[Function]}
/>
<ForwardRef
aria-label="Toggle notification failure"
id="notification-9000-error-toggle"
isChecked={false}
isDisabled={false}
label="Failure"
onChange={[Function]}
/>
</ForwardRef>,
]
}
key=".0"
rowid="items-list-item-9000"
>
<div
className="pf-c-data-list__item-content"
>
<NotificationListItem__DataListCell
key="name"
>
<StyledComponent
forwardedComponent={
Object { Object {
"$$typeof": Symbol(react.forward_ref), "pathname": "/foo",
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__DataListCell-j7c411-0",
"isStatic": false,
"lastClassName": "hoXOpW",
"rules": Array [
"display:flex;justify-content:",
[Function],
";padding-bottom:",
[Function],
";@media screen and (min-width:768px){justify-content:",
[Function],
";padding-bottom:0;}",
],
},
"displayName": "NotificationListItem__DataListCell",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__DataListCell-j7c411-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
} }
} }
forwardedRef={null}
> >
<DataListCell <b
alignRight={false} id="items-list-item-9000"
className="NotificationListItem__DataListCell-j7c411-0 kIdLtz"
isFilled={true}
isIcon={false}
width={1}
> >
<div Foo
className="pf-c-data-list__cell NotificationListItem__DataListCell-j7c411-0 kIdLtz" </b>
</ForwardRef>
<ForwardRef
isRead={true}
>
slack
</ForwardRef>
</ForwardRef>,
<ForwardRef
righthalf="true"
>
<ForwardRef
aria-label="Toggle notification success"
id="notification-9000-success-toggle"
isChecked={false}
isDisabled={false}
label="Successful"
onChange={[Function]}
/>
<ForwardRef
aria-label="Toggle notification failure"
id="notification-9000-error-toggle"
isChecked={false}
isDisabled={false}
label="Failure"
onChange={[Function]}
/>
</ForwardRef>,
]
}
key=".0"
rowid="items-list-item-9000"
>
<div
className="pf-c-data-list__item-content"
>
<NotificationListItem__DataListCell
key="name"
>
<StyledComponent
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__DataListCell-j7c411-0",
"isStatic": false,
"lastClassName": "hoXOpW",
"rules": Array [
"display:flex;justify-content:",
[Function],
";padding-bottom:",
[Function],
";@media screen and (min-width:768px){justify-content:",
[Function],
";padding-bottom:0;}",
],
},
"displayName": "NotificationListItem__DataListCell",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__DataListCell-j7c411-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
>
<DataListCell
alignRight={false}
className="NotificationListItem__DataListCell-j7c411-0 kIdLtz"
isFilled={true}
isIcon={false}
width={1}
>
<div
className="pf-c-data-list__cell NotificationListItem__DataListCell-j7c411-0 kIdLtz"
>
<Styled(Link)
to={
Object {
"pathname": "/foo",
}
}
> >
<Styled(Link) <StyledComponent
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "sc-bdVaJa",
"isStatic": true,
"lastClassName": "eBseNd",
"rules": Array [
"margin-right: 1.5em;",
],
},
"displayName": "Styled(Link)",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "sc-bdVaJa",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
to={ to={
Object { Object {
"pathname": "/foo", "pathname": "/foo",
} }
} }
> >
<StyledComponent <Link
forwardedComponent={ className="sc-bdVaJa eBseNd"
Object { replace={false}
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "sc-bdVaJa",
"isStatic": true,
"lastClassName": "eBseNd",
"rules": Array [
"margin-right: 1.5em;",
],
},
"displayName": "Styled(Link)",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "sc-bdVaJa",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
to={ to={
Object { Object {
"pathname": "/foo", "pathname": "/foo",
} }
} }
> >
<Link <a
className="sc-bdVaJa eBseNd" className="sc-bdVaJa eBseNd"
replace={false} onClick={[Function]}
to={
Object {
"pathname": "/foo",
}
}
> >
<a <b
className="sc-bdVaJa eBseNd" id="items-list-item-9000"
onClick={[Function]}
> >
<b Foo
id="items-list-item-9000" </b>
> </a>
Foo </Link>
</b> </StyledComponent>
</a> </Styled(Link)>
</Link> <Styled(Badge)
</StyledComponent> isRead={true}
</Styled(Link)> >
<Styled(Badge) <StyledComponent
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "sc-bwzfXH",
"isStatic": true,
"lastClassName": "chTbOZ",
"rules": Array [
"text-transform: capitalize;",
],
},
"displayName": "Styled(Badge)",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "sc-bwzfXH",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
isRead={true} isRead={true}
> >
<StyledComponent <Badge
forwardedComponent={ className="sc-bwzfXH chTbOZ"
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "sc-bwzfXH",
"isStatic": true,
"lastClassName": "chTbOZ",
"rules": Array [
"text-transform: capitalize;",
],
},
"displayName": "Styled(Badge)",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "sc-bwzfXH",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
isRead={true} isRead={true}
> >
<Badge <span
className="sc-bwzfXH chTbOZ" className="pf-c-badge pf-m-read sc-bwzfXH chTbOZ"
isRead={true}
> >
<span slack
className="pf-c-badge pf-m-read sc-bwzfXH chTbOZ" </span>
> </Badge>
slack </StyledComponent>
</span> </Styled(Badge)>
</Badge> </div>
</StyledComponent> </DataListCell>
</Styled(Badge)> </StyledComponent>
</div> </NotificationListItem__DataListCell>
</DataListCell> <NotificationListItem__DataListCell
</StyledComponent> key="toggles"
</NotificationListItem__DataListCell> righthalf="true"
<NotificationListItem__DataListCell >
key="toggles" <StyledComponent
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__DataListCell-j7c411-0",
"isStatic": false,
"lastClassName": "hoXOpW",
"rules": Array [
"display:flex;justify-content:",
[Function],
";padding-bottom:",
[Function],
";@media screen and (min-width:768px){justify-content:",
[Function],
";padding-bottom:0;}",
],
},
"displayName": "NotificationListItem__DataListCell",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__DataListCell-j7c411-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
righthalf="true" righthalf="true"
> >
<StyledComponent <DataListCell
forwardedComponent={ alignRight={false}
Object { className="NotificationListItem__DataListCell-j7c411-0 hoXOpW"
"$$typeof": Symbol(react.forward_ref), isFilled={true}
"attrs": Array [], isIcon={false}
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__DataListCell-j7c411-0",
"isStatic": false,
"lastClassName": "hoXOpW",
"rules": Array [
"display:flex;justify-content:",
[Function],
";padding-bottom:",
[Function],
";@media screen and (min-width:768px){justify-content:",
[Function],
";padding-bottom:0;}",
],
},
"displayName": "NotificationListItem__DataListCell",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__DataListCell-j7c411-0",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
righthalf="true" righthalf="true"
width={1}
> >
<DataListCell <div
alignRight={false} className="pf-c-data-list__cell NotificationListItem__DataListCell-j7c411-0 hoXOpW"
className="NotificationListItem__DataListCell-j7c411-0 hoXOpW"
isFilled={true}
isIcon={false}
righthalf="true" righthalf="true"
width={1}
> >
<div <NotificationListItem__Switch
className="pf-c-data-list__cell NotificationListItem__DataListCell-j7c411-0 hoXOpW" aria-label="Toggle notification success"
righthalf="true" id="notification-9000-success-toggle"
isChecked={false}
isDisabled={false}
label="Successful"
onChange={[Function]}
> >
<NotificationListItem__Switch <StyledComponent
aria-label="Toggle notification success" aria-label="Toggle notification success"
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__Switch-j7c411-1",
"isStatic": true,
"lastClassName": "ceuHGn",
"rules": Array [
"display:flex;flex-wrap:no-wrap;",
],
},
"displayName": "NotificationListItem__Switch",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__Switch-j7c411-1",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
id="notification-9000-success-toggle" id="notification-9000-success-toggle"
isChecked={false} isChecked={false}
isDisabled={false} isDisabled={false}
label="Successful" label="Successful"
onChange={[Function]} onChange={[Function]}
> >
<StyledComponent <Switch
aria-label="Toggle notification success" aria-label="Toggle notification success"
forwardedComponent={ className="NotificationListItem__Switch-j7c411-1 ceuHGn"
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__Switch-j7c411-1",
"isStatic": true,
"lastClassName": "ceuHGn",
"rules": Array [
"display:flex;flex-wrap:no-wrap;",
],
},
"displayName": "NotificationListItem__Switch",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__Switch-j7c411-1",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
id="notification-9000-success-toggle" id="notification-9000-success-toggle"
isChecked={false} isChecked={false}
isDisabled={false} isDisabled={false}
label="Successful" label="Successful"
onChange={[Function]} onChange={[Function]}
> >
<Switch <label
aria-label="Toggle notification success" className="pf-c-switch NotificationListItem__Switch-j7c411-1 ceuHGn"
className="NotificationListItem__Switch-j7c411-1 ceuHGn" htmlFor="notification-9000-success-toggle"
id="notification-9000-success-toggle"
isChecked={false}
isDisabled={false}
label="Successful"
onChange={[Function]}
> >
<label <input
className="pf-c-switch NotificationListItem__Switch-j7c411-1 ceuHGn" aria-label="Toggle notification success"
htmlFor="notification-9000-success-toggle" checked={false}
className="pf-c-switch__input"
disabled={false}
id="notification-9000-success-toggle"
onChange={[Function]}
type="checkbox"
/>
<span
className="pf-c-switch__toggle"
/>
<span
aria-hidden="true"
className="pf-c-switch__label pf-m-on"
> >
<input Successful
aria-label="Toggle notification success" </span>
checked={false} <span
className="pf-c-switch__input" aria-hidden="true"
disabled={false} className="pf-c-switch__label pf-m-off"
id="notification-9000-success-toggle" >
onChange={[Function]} Successful
type="checkbox" </span>
/> </label>
<span </Switch>
className="pf-c-switch__toggle" </StyledComponent>
/> </NotificationListItem__Switch>
<span <NotificationListItem__Switch
aria-hidden="true" aria-label="Toggle notification failure"
className="pf-c-switch__label pf-m-on" id="notification-9000-error-toggle"
> isChecked={false}
Successful isDisabled={false}
</span> label="Failure"
<span onChange={[Function]}
aria-hidden="true" >
className="pf-c-switch__label pf-m-off" <StyledComponent
>
Successful
</span>
</label>
</Switch>
</StyledComponent>
</NotificationListItem__Switch>
<NotificationListItem__Switch
aria-label="Toggle notification failure" aria-label="Toggle notification failure"
forwardedComponent={
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__Switch-j7c411-1",
"isStatic": true,
"lastClassName": "ceuHGn",
"rules": Array [
"display:flex;flex-wrap:no-wrap;",
],
},
"displayName": "NotificationListItem__Switch",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__Switch-j7c411-1",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
id="notification-9000-error-toggle" id="notification-9000-error-toggle"
isChecked={false} isChecked={false}
isDisabled={false} isDisabled={false}
label="Failure" label="Failure"
onChange={[Function]} onChange={[Function]}
> >
<StyledComponent <Switch
aria-label="Toggle notification failure" aria-label="Toggle notification failure"
forwardedComponent={ className="NotificationListItem__Switch-j7c411-1 ceuHGn"
Object {
"$$typeof": Symbol(react.forward_ref),
"attrs": Array [],
"componentStyle": ComponentStyle {
"componentId": "NotificationListItem__Switch-j7c411-1",
"isStatic": true,
"lastClassName": "ceuHGn",
"rules": Array [
"display:flex;flex-wrap:no-wrap;",
],
},
"displayName": "NotificationListItem__Switch",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__Switch-j7c411-1",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
id="notification-9000-error-toggle" id="notification-9000-error-toggle"
isChecked={false} isChecked={false}
isDisabled={false} isDisabled={false}
label="Failure" label="Failure"
onChange={[Function]} onChange={[Function]}
> >
<Switch <label
aria-label="Toggle notification failure" className="pf-c-switch NotificationListItem__Switch-j7c411-1 ceuHGn"
className="NotificationListItem__Switch-j7c411-1 ceuHGn" htmlFor="notification-9000-error-toggle"
id="notification-9000-error-toggle"
isChecked={false}
isDisabled={false}
label="Failure"
onChange={[Function]}
> >
<label <input
className="pf-c-switch NotificationListItem__Switch-j7c411-1 ceuHGn" aria-label="Toggle notification failure"
htmlFor="notification-9000-error-toggle" checked={false}
className="pf-c-switch__input"
disabled={false}
id="notification-9000-error-toggle"
onChange={[Function]}
type="checkbox"
/>
<span
className="pf-c-switch__toggle"
/>
<span
aria-hidden="true"
className="pf-c-switch__label pf-m-on"
> >
<input Failure
aria-label="Toggle notification failure" </span>
checked={false} <span
className="pf-c-switch__input" aria-hidden="true"
disabled={false} className="pf-c-switch__label pf-m-off"
id="notification-9000-error-toggle" >
onChange={[Function]} Failure
type="checkbox" </span>
/> </label>
<span </Switch>
className="pf-c-switch__toggle" </StyledComponent>
/> </NotificationListItem__Switch>
<span </div>
aria-hidden="true" </DataListCell>
className="pf-c-switch__label pf-m-on" </StyledComponent>
> </NotificationListItem__DataListCell>
Failure </div>
</span> </DataListItemCells>
<span <NotificationListItem__Switch
aria-hidden="true" aria-label="Toggle notification failure"
className="pf-c-switch__label pf-m-off" id="notification-9000-error-toggle"
> isChecked={false}
Failure isDisabled={false}
</span> key=".1"
</label> label="Failure"
</Switch> onChange={[Function]}
</StyledComponent> rowid="items-list-item-9000"
</NotificationListItem__Switch> >
</div> <StyledComponent
</DataListCell> aria-label="Toggle notification failure"
</StyledComponent> forwardedComponent={
</NotificationListItem__DataListCell> Object {
</div> "$$typeof": Symbol(react.forward_ref),
</DataListItemCells> "attrs": Array [],
</div> "componentStyle": ComponentStyle {
</DataListItemRow> "componentId": "NotificationListItem__Switch-j7c411-1",
</li> "isStatic": true,
</DataListItem> "lastClassName": "ceuHGn",
</I18n> "rules": Array [
"display:flex;flex-wrap:no-wrap;",
],
},
"displayName": "NotificationListItem__Switch",
"foldedComponentIds": Array [],
"render": [Function],
"styledComponentId": "NotificationListItem__Switch-j7c411-1",
"target": [Function],
"toString": [Function],
"warnTooManyClasses": [Function],
"withComponent": [Function],
}
}
forwardedRef={null}
id="notification-9000-error-toggle"
isChecked={false}
isDisabled={false}
label="Failure"
onChange={[Function]}
rowid="items-list-item-9000"
>
<Switch
aria-label="Toggle notification failure"
className="NotificationListItem__Switch-j7c411-1 ceuHGn"
id="notification-9000-error-toggle"
isChecked={false}
isDisabled={false}
label="Failure"
onChange={[Function]}
rowid="items-list-item-9000"
>
<label
className="pf-c-switch NotificationListItem__Switch-j7c411-1 ceuHGn"
htmlFor="notification-9000-error-toggle"
>
<input
aria-label="Toggle notification failure"
checked={false}
className="pf-c-switch__input"
disabled={false}
id="notification-9000-error-toggle"
onChange={[Function]}
rowid="items-list-item-9000"
type="checkbox"
/>
<span
className="pf-c-switch__toggle"
/>
<span
aria-hidden="true"
className="pf-c-switch__label pf-m-on"
>
Failure
</span>
<span
aria-hidden="true"
className="pf-c-switch__label pf-m-off"
>
Failure
</span>
</label>
</Switch>
</StyledComponent>
</NotificationListItem__Switch>
</div>
</DataListItemRow>
</li>
</DataListItem>
</NotificationListItem> </NotificationListItem>
`; `;

View File

@@ -1,7 +1,6 @@
import React from 'react'; import React from 'react';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
// import { mount } from 'enzyme'; import { withI18n } from '@lingui/react';
import { I18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { mountWithContexts } from './enzymeHelpers'; import { mountWithContexts } from './enzymeHelpers';
import { Config } from '../src/contexts/Config'; import { Config } from '../src/contexts/Config';
@@ -11,26 +10,19 @@ import { withRootDialog } from '../src/contexts/RootDialog';
describe('mountWithContexts', () => { describe('mountWithContexts', () => {
describe('injected I18nProvider', () => { describe('injected I18nProvider', () => {
test('should mount and render', () => { test('should mount and render', () => {
const wrapper = mountWithContexts( const Child = withI18n()(({ i18n }) => (
<div> <div>
<I18n> <span>{i18n._(t`Text content`)}</span>
{({ i18n }) => (
<span>{i18n._(t`Text content`)}</span>
)}
</I18n>
</div> </div>
); ));
const wrapper = mountWithContexts(<Child />);
expect(wrapper.find('div')).toMatchSnapshot(); expect(wrapper.find('div')).toMatchSnapshot();
}); });
test('should mount and render deeply nested consumer', () => { test('should mount and render deeply nested consumer', () => {
const Child = () => ( const Child = withI18n()(({ i18n }) => (
<I18n> <div>{i18n._(t`Text content`)}</div>
{({ i18n }) => ( ));
<div>{i18n._(t`Text content`)}</div>
)}
</I18n>
);
const Parent = () => (<Child />); const Parent = () => (<Child />);
const wrapper = mountWithContexts(<Parent />); const wrapper = mountWithContexts(<Parent />);
expect(wrapper.find('Parent')).toMatchSnapshot(); expect(wrapper.find('Parent')).toMatchSnapshot();

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Applications from '../../src/pages/Applications'; import Applications from '../../src/pages/Applications';
describe('<Applications />', () => { describe('<Applications />', () => {
@@ -8,7 +8,7 @@ describe('<Applications />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Applications />); pageWrapper = mountWithContexts(<Applications />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import AuthSettings from '../../src/pages/AuthSettings'; import AuthSettings from '../../src/pages/AuthSettings';
describe('<AuthSettings />', () => { describe('<AuthSettings />', () => {
@@ -8,7 +8,7 @@ describe('<AuthSettings />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<AuthSettings />); pageWrapper = mountWithContexts(<AuthSettings />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import CredentialTypes from '../../src/pages/CredentialTypes'; import CredentialTypes from '../../src/pages/CredentialTypes';
describe('<CredentialTypes />', () => { describe('<CredentialTypes />', () => {
@@ -8,7 +8,7 @@ describe('<CredentialTypes />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<CredentialTypes />); pageWrapper = mountWithContexts(<CredentialTypes />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Credentials from '../../src/pages/Credentials'; import Credentials from '../../src/pages/Credentials';
describe('<Credentials />', () => { describe('<Credentials />', () => {
@@ -8,7 +8,7 @@ describe('<Credentials />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Credentials />); pageWrapper = mountWithContexts(<Credentials />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Dashboard from '../../src/pages/Dashboard'; import Dashboard from '../../src/pages/Dashboard';
describe('<Dashboard />', () => { describe('<Dashboard />', () => {
@@ -8,7 +8,7 @@ describe('<Dashboard />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Dashboard />); pageWrapper = mountWithContexts(<Dashboard />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import InstanceGroups from '../../src/pages/InstanceGroups'; import InstanceGroups from '../../src/pages/InstanceGroups';
describe('<InstanceGroups />', () => { describe('<InstanceGroups />', () => {
@@ -8,7 +8,7 @@ describe('<InstanceGroups />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<InstanceGroups />); pageWrapper = mountWithContexts(<InstanceGroups />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Inventories from '../../src/pages/Inventories'; import Inventories from '../../src/pages/Inventories';
describe('<Inventories />', () => { describe('<Inventories />', () => {
@@ -8,7 +8,7 @@ describe('<Inventories />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Inventories />); pageWrapper = mountWithContexts(<Inventories />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import InventoryScripts from '../../src/pages/InventoryScripts'; import InventoryScripts from '../../src/pages/InventoryScripts';
describe('<InventoryScripts />', () => { describe('<InventoryScripts />', () => {
@@ -8,7 +8,7 @@ describe('<InventoryScripts />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<InventoryScripts />); pageWrapper = mountWithContexts(<InventoryScripts />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Jobs from '../../src/pages/Jobs'; import Jobs from '../../src/pages/Jobs';
describe('<Jobs />', () => { describe('<Jobs />', () => {
@@ -8,7 +8,7 @@ describe('<Jobs />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Jobs />); pageWrapper = mountWithContexts(<Jobs />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import JobsSettings from '../../src/pages/JobsSettings'; import JobsSettings from '../../src/pages/JobsSettings';
describe('<JobsSettings />', () => { describe('<JobsSettings />', () => {
@@ -8,7 +8,7 @@ describe('<JobsSettings />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<JobsSettings />); pageWrapper = mountWithContexts(<JobsSettings />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import License from '../../src/pages/License'; import License from '../../src/pages/License';
describe('<License />', () => { describe('<License />', () => {
@@ -8,7 +8,7 @@ describe('<License />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<License />); pageWrapper = mountWithContexts(<License />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,9 +1,7 @@
import React from 'react'; import React from 'react';
import { MemoryRouter } from 'react-router-dom'; import { mountWithContexts } from '../enzymeHelpers';
import { mount } from 'enzyme';
import { I18nProvider } from '@lingui/react';
import { asyncFlush } from '../../jest.setup'; import { asyncFlush } from '../../jest.setup';
import { _AWXLogin } from '../../src/pages/Login'; import AWXLogin from '../../src/pages/Login';
import APIClient from '../../src/api'; import APIClient from '../../src/api';
describe('<Login />', () => { describe('<Login />', () => {
@@ -18,6 +16,12 @@ describe('<Login />', () => {
const api = new APIClient({}); const api = new APIClient({});
const mountLogin = () => {
loginWrapper = mountWithContexts(<AWXLogin />, { context: { network: {
api, handleHttpError: () => {}
} } });
};
const findChildren = () => { const findChildren = () => {
awxLogin = loginWrapper.find('AWXLogin'); awxLogin = loginWrapper.find('AWXLogin');
loginPage = loginWrapper.find('LoginPage'); loginPage = loginWrapper.find('LoginPage');
@@ -28,22 +32,13 @@ describe('<Login />', () => {
loginHeaderLogo = loginPage.find('img'); loginHeaderLogo = loginPage.find('img');
}; };
beforeEach(() => {
loginWrapper = mount(
<MemoryRouter>
<I18nProvider>
<_AWXLogin api={api} clearRootDialogMessage={() => {}} handleHttpError={() => {}} />
</I18nProvider>
</MemoryRouter>
);
findChildren();
});
afterEach(() => { afterEach(() => {
loginWrapper.unmount(); loginWrapper.unmount();
}); });
test('initially renders without crashing', () => { test('initially renders without crashing', () => {
mountLogin();
findChildren();
expect(loginWrapper.length).toBe(1); expect(loginWrapper.length).toBe(1);
expect(loginPage.length).toBe(1); expect(loginPage.length).toBe(1);
expect(loginForm.length).toBe(1); expect(loginForm.length).toBe(1);
@@ -58,13 +53,7 @@ describe('<Login />', () => {
}); });
test('custom logo renders Brand component with correct src and alt', () => { test('custom logo renders Brand component with correct src and alt', () => {
loginWrapper = mount( loginWrapper = mountWithContexts(<AWXLogin logo="images/foo.jpg" alt="Foo Application" />);
<MemoryRouter>
<I18nProvider>
<_AWXLogin api={api} logo="images/foo.jpg" alt="Foo Application" />
</I18nProvider>
</MemoryRouter>
);
findChildren(); findChildren();
expect(loginHeaderLogo.length).toBe(1); expect(loginHeaderLogo.length).toBe(1);
expect(loginHeaderLogo.props().src).toBe('data:image/jpeg;images/foo.jpg'); expect(loginHeaderLogo.props().src).toBe('data:image/jpeg;images/foo.jpg');
@@ -72,13 +61,7 @@ describe('<Login />', () => {
}); });
test('default logo renders Brand component with correct src and alt', () => { test('default logo renders Brand component with correct src and alt', () => {
loginWrapper = mount( mountLogin();
<MemoryRouter>
<I18nProvider>
<_AWXLogin api={api} />
</I18nProvider>
</MemoryRouter>
);
findChildren(); findChildren();
expect(loginHeaderLogo.length).toBe(1); expect(loginHeaderLogo.length).toBe(1);
expect(loginHeaderLogo.props().src).toBe('tower-logo-header.svg'); expect(loginHeaderLogo.props().src).toBe('tower-logo-header.svg');
@@ -86,6 +69,8 @@ describe('<Login />', () => {
}); });
test('state maps to un/pw input value props', () => { test('state maps to un/pw input value props', () => {
mountLogin();
findChildren();
awxLogin.setState({ username: 'un', password: 'pw' }); awxLogin.setState({ username: 'un', password: 'pw' });
expect(awxLogin.state().username).toBe('un'); expect(awxLogin.state().username).toBe('un');
expect(awxLogin.state().password).toBe('pw'); expect(awxLogin.state().password).toBe('pw');
@@ -95,6 +80,8 @@ describe('<Login />', () => {
}); });
test('updating un/pw clears out error', () => { test('updating un/pw clears out error', () => {
mountLogin();
findChildren();
awxLogin.setState({ isInputValid: false }); awxLogin.setState({ isInputValid: false });
expect(loginWrapper.find('.pf-c-form__helper-text.pf-m-error').length).toBe(1); expect(loginWrapper.find('.pf-c-form__helper-text.pf-m-error').length).toBe(1);
usernameInput.instance().value = 'uname'; usernameInput.instance().value = 'uname';
@@ -113,6 +100,8 @@ describe('<Login />', () => {
test('api.login not called when loading', () => { test('api.login not called when loading', () => {
api.login = jest.fn().mockImplementation(() => Promise.resolve({})); api.login = jest.fn().mockImplementation(() => Promise.resolve({}));
mountLogin();
findChildren();
expect(awxLogin.state().isLoading).toBe(false); expect(awxLogin.state().isLoading).toBe(false);
awxLogin.setState({ isLoading: true }); awxLogin.setState({ isLoading: true });
submitButton.simulate('click'); submitButton.simulate('click');
@@ -121,6 +110,8 @@ describe('<Login />', () => {
test('submit calls api.login successfully', async () => { test('submit calls api.login successfully', async () => {
api.login = jest.fn().mockImplementation(() => Promise.resolve({})); api.login = jest.fn().mockImplementation(() => Promise.resolve({}));
mountLogin();
findChildren();
expect(awxLogin.state().isLoading).toBe(false); expect(awxLogin.state().isLoading).toBe(false);
awxLogin.setState({ username: 'unamee', password: 'pwordd' }); awxLogin.setState({ username: 'unamee', password: 'pwordd' });
submitButton.simulate('click'); submitButton.simulate('click');
@@ -137,6 +128,8 @@ describe('<Login />', () => {
err.response = { status: 401, message: 'problem' }; err.response = { status: 401, message: 'problem' };
return Promise.reject(err); return Promise.reject(err);
}); });
mountLogin();
findChildren();
expect(awxLogin.state().isLoading).toBe(false); expect(awxLogin.state().isLoading).toBe(false);
expect(awxLogin.state().isInputValid).toBe(true); expect(awxLogin.state().isInputValid).toBe(true);
awxLogin.setState({ username: 'unamee', password: 'pwordd' }); awxLogin.setState({ username: 'unamee', password: 'pwordd' });
@@ -155,6 +148,8 @@ describe('<Login />', () => {
err.response = { status: 500, message: 'problem' }; err.response = { status: 500, message: 'problem' };
return Promise.reject(err); return Promise.reject(err);
}); });
mountLogin();
findChildren();
expect(awxLogin.state().isLoading).toBe(false); expect(awxLogin.state().isLoading).toBe(false);
awxLogin.setState({ username: 'unamee', password: 'pwordd' }); awxLogin.setState({ username: 'unamee', password: 'pwordd' });
submitButton.simulate('click'); submitButton.simulate('click');
@@ -166,6 +161,8 @@ describe('<Login />', () => {
}); });
test('render Redirect to / when already authenticated', () => { test('render Redirect to / when already authenticated', () => {
mountLogin();
findChildren();
awxLogin.setState({ isAuthenticated: true }); awxLogin.setState({ isAuthenticated: true });
const redirectElem = loginWrapper.find('Redirect'); const redirectElem = loginWrapper.find('Redirect');
expect(redirectElem.length).toBe(1); expect(redirectElem.length).toBe(1);

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import ManagementJobs from '../../src/pages/ManagementJobs'; import ManagementJobs from '../../src/pages/ManagementJobs';
describe('<ManagementJobs />', () => { describe('<ManagementJobs />', () => {
@@ -8,7 +8,7 @@ describe('<ManagementJobs />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<ManagementJobs />); pageWrapper = mountWithContexts(<ManagementJobs />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import NotificationTemplates from '../../src/pages/NotificationTemplates'; import NotificationTemplates from '../../src/pages/NotificationTemplates';
describe('<NotificationTemplates />', () => { describe('<NotificationTemplates />', () => {
@@ -8,7 +8,7 @@ describe('<NotificationTemplates />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<NotificationTemplates />); pageWrapper = mountWithContexts(<NotificationTemplates />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -2,6 +2,7 @@
exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = ` exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = `
<DeleteRoleConfirmationModal <DeleteRoleConfirmationModal
i18n={"/i18n/"}
onCancel={[Function]} onCancel={[Function]}
onConfirm={[Function]} onConfirm={[Function]}
role={ role={
@@ -16,11 +17,47 @@ exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = `
} }
username="jane" username="jane"
> >
<I18n <_default
update={true} actions={
withHash={true} Array [
<Button
aria-label="Confirm delete"
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="danger"
>
Delete
</Button>,
<Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="secondary"
>
Cancel
</Button>,
]
}
isOpen={true}
onClose={[Function]}
title="Remove {0} Access"
variant="danger"
> >
<_default <Modal
actions={ actions={
Array [ Array [
<Button <Button
@@ -55,377 +92,237 @@ exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = `
</Button>, </Button>,
] ]
} }
ariaDescribedById=""
className="awx-c-modal at-c-alertModal at-c-alertModal--danger"
hideTitle={false}
isLarge={false}
isOpen={true} isOpen={true}
isSmall={false}
onClose={[Function]} onClose={[Function]}
title="Remove Team Access" title="Remove {0} Access"
variant="danger" width={null}
> >
<Modal <Portal
actions={ containerInfo={
Array [ <div>
<Button <div
aria-label="Confirm delete" class="pf-c-backdrop"
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="danger"
> >
Delete
</Button>,
<Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="secondary"
>
Cancel
</Button>,
]
}
ariaDescribedById=""
className="awx-c-modal at-c-alertModal at-c-alertModal--danger"
hideTitle={false}
isLarge={false}
isOpen={true}
isSmall={false}
onClose={[Function]}
title="Remove Team Access"
width={null}
>
<Portal
containerInfo={
<div>
<div <div
class="pf-c-backdrop" class="pf-l-bullseye"
> >
<div <div
class="pf-l-bullseye" class="pf-l-bullseye"
> >
<div <div
class="pf-l-bullseye" aria-describedby="pf-modal-0"
aria-label="Remove {0} Access"
aria-modal="true"
class="pf-c-modal-box awx-c-modal at-c-alertModal at-c-alertModal--danger"
role="dialog"
> >
<div <button
aria-describedby="pf-modal-0" aria-label="Close"
aria-label="Remove Team Access" class="pf-c-button pf-m-plain"
aria-modal="true" type="button"
class="pf-c-modal-box awx-c-modal at-c-alertModal at-c-alertModal--danger"
role="dialog"
> >
<svg
aria-hidden="true"
fill="currentColor"
height="1em"
role="img"
style="vertical-align: -0.125em;"
viewBox="0 0 352 512"
width="1em"
>
<path
d="M242.72 256l100.07-100.07c12.28-12.28 12.28-32.19 0-44.48l-22.24-22.24c-12.28-12.28-32.19-12.28-44.48 0L176 189.28 75.93 89.21c-12.28-12.28-32.19-12.28-44.48 0L9.21 111.45c-12.28 12.28-12.28 32.19 0 44.48L109.28 256 9.21 356.07c-12.28 12.28-12.28 32.19 0 44.48l22.24 22.24c12.28 12.28 32.2 12.28 44.48 0L176 322.72l100.07 100.07c12.28 12.28 32.2 12.28 44.48 0l22.24-22.24c12.28-12.28 12.28-32.19 0-44.48L242.72 256z"
transform=""
/>
</svg>
</button>
<h3
class="pf-c-title pf-m-2xl"
>
Remove {0} Access
</h3>
<div
class="pf-c-modal-box__body"
id="pf-modal-0"
>
Are you sure you want to remove {0} access from {1}? Doing so affects all members of the team.
<br />
<br />
If you {0} want to remove access for this particular user, please remove them from the team.
<svg
aria-hidden="true"
class="at-c-alertModal__icon"
fill="currentColor"
height="1em"
role="img"
style="vertical-align: -0.125em;"
viewBox="0 0 512 512"
width="1em"
>
<path
d="M504 256c0 136.997-111.043 248-248 248S8 392.997 8 256C8 119.083 119.043 8 256 8s248 111.083 248 248zm-248 50c-25.405 0-46 20.595-46 46s20.595 46 46 46 46-20.595 46-46-20.595-46-46-46zm-43.673-165.346l7.418 136c.347 6.364 5.609 11.346 11.982 11.346h48.546c6.373 0 11.635-4.982 11.982-11.346l7.418-136c.375-6.874-5.098-12.654-11.982-12.654h-63.383c-6.884 0-12.356 5.78-11.981 12.654z"
transform=""
/>
</svg>
</div>
<div
class="pf-c-modal-box__footer"
>
<button <button
aria-label="Close" aria-label="Confirm delete"
class="pf-c-button pf-m-plain" class="pf-c-button pf-m-danger"
type="button" type="button"
> >
<svg Delete
aria-hidden="true"
fill="currentColor"
height="1em"
role="img"
style="vertical-align: -0.125em;"
viewBox="0 0 352 512"
width="1em"
>
<path
d="M242.72 256l100.07-100.07c12.28-12.28 12.28-32.19 0-44.48l-22.24-22.24c-12.28-12.28-32.19-12.28-44.48 0L176 189.28 75.93 89.21c-12.28-12.28-32.19-12.28-44.48 0L9.21 111.45c-12.28 12.28-12.28 32.19 0 44.48L109.28 256 9.21 356.07c-12.28 12.28-12.28 32.19 0 44.48l22.24 22.24c12.28 12.28 32.2 12.28 44.48 0L176 322.72l100.07 100.07c12.28 12.28 32.2 12.28 44.48 0l22.24-22.24c12.28-12.28 12.28-32.19 0-44.48L242.72 256z"
transform=""
/>
</svg>
</button> </button>
<h3 <button
class="pf-c-title pf-m-2xl" class="pf-c-button pf-m-secondary"
type="button"
> >
Cancel
Remove Team Access </button>
</h3>
<div
class="pf-c-modal-box__body"
id="pf-modal-0"
>
<svg
aria-hidden="true"
class="at-c-alertModal__icon"
fill="currentColor"
height="1em"
role="img"
style="vertical-align: -0.125em;"
viewBox="0 0 512 512"
width="1em"
>
<path
d="M504 256c0 136.997-111.043 248-248 248S8 392.997 8 256C8 119.083 119.043 8 256 8s248 111.083 248 248zm-248 50c-25.405 0-46 20.595-46 46s20.595 46 46 46 46-20.595 46-46-20.595-46-46-46zm-43.673-165.346l7.418 136c.347 6.364 5.609 11.346 11.982 11.346h48.546c6.373 0 11.635-4.982 11.982-11.346l7.418-136c.375-6.874-5.098-12.654-11.982-12.654h-63.383c-6.884 0-12.356 5.78-11.981 12.654z"
transform=""
/>
</svg>
</div>
<div
class="pf-c-modal-box__footer"
>
<button
aria-label="Confirm delete"
class="pf-c-button pf-m-danger"
type="button"
>
Delete
</button>
<button
class="pf-c-button pf-m-secondary"
type="button"
>
Cancel
</button>
</div>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
</div> </div>
} </div>
> }
<ModalContent >
actions={ <ModalContent
Array [ actions={
<Button Array [
aria-label="Confirm delete" <Button
className="" aria-label="Confirm delete"
component="button" className=""
isActive={false} component="button"
isBlock={false} isActive={false}
isDisabled={false} isBlock={false}
isFocus={false} isDisabled={false}
isHover={false} isFocus={false}
onClick={[Function]} isHover={false}
type="button" onClick={[Function]}
variant="danger" type="button"
> variant="danger"
Delete
</Button>,
<Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="secondary"
>
Cancel
</Button>,
]
}
ariaDescribedById=""
className="awx-c-modal at-c-alertModal at-c-alertModal--danger"
hideTitle={false}
id="pf-modal-0"
isLarge={false}
isOpen={true}
isSmall={false}
onClose={[Function]}
title="Remove Team Access"
width={null}
>
<Backdrop
className=""
>
<div
className="pf-c-backdrop"
> >
<Bullseye Delete
className="" </Button>,
component="div" <Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="secondary"
>
Cancel
</Button>,
]
}
ariaDescribedById=""
className="awx-c-modal at-c-alertModal at-c-alertModal--danger"
hideTitle={false}
id="pf-modal-0"
isLarge={false}
isOpen={true}
isSmall={false}
onClose={[Function]}
title="Remove {0} Access"
width={null}
>
<Backdrop
className=""
>
<div
className="pf-c-backdrop"
>
<Bullseye
className=""
component="div"
>
<div
className="pf-l-bullseye"
> >
<div <FocusTrap
_createFocusTrap={[Function]}
active={true}
className="pf-l-bullseye" className="pf-l-bullseye"
> focusTrapOptions={
<FocusTrap Object {
_createFocusTrap={[Function]} "clickOutsideDeactivates": true,
active={true}
className="pf-l-bullseye"
focusTrapOptions={
Object {
"clickOutsideDeactivates": true,
}
} }
paused={false} }
tag="div" paused={false}
tag="div"
>
<div
className="pf-l-bullseye"
> >
<div <ModalBox
className="pf-l-bullseye" className="awx-c-modal at-c-alertModal at-c-alertModal--danger"
id="pf-modal-0"
isLarge={false}
isSmall={false}
style={
Object {
"width": null,
}
}
title="Remove {0} Access"
> >
<ModalBox <div
className="awx-c-modal at-c-alertModal at-c-alertModal--danger" aria-describedby="pf-modal-0"
id="pf-modal-0" aria-label="Remove {0} Access"
isLarge={false} aria-modal="true"
isSmall={false} className="pf-c-modal-box awx-c-modal at-c-alertModal at-c-alertModal--danger"
role="dialog"
style={ style={
Object { Object {
"width": null, "width": null,
} }
} }
title="Remove Team Access"
> >
<div <ModalBoxCloseButton
aria-describedby="pf-modal-0" className=""
aria-label="Remove Team Access" onClose={[Function]}
aria-modal="true"
className="pf-c-modal-box awx-c-modal at-c-alertModal at-c-alertModal--danger"
role="dialog"
style={
Object {
"width": null,
}
}
> >
<ModalBoxCloseButton <Button
aria-label="Close"
className="" className=""
onClose={[Function]} component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]}
type="button"
variant="plain"
> >
<Button <button
aria-disabled={null}
aria-label="Close" aria-label="Close"
className="" className="pf-c-button pf-m-plain"
component="button" disabled={false}
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]} onClick={[Function]}
tabIndex={null}
type="button" type="button"
variant="plain"
> >
<button <TimesIcon
aria-disabled={null}
aria-label="Close"
className="pf-c-button pf-m-plain"
disabled={false}
onClick={[Function]}
tabIndex={null}
type="button"
>
<TimesIcon
color="currentColor"
size="sm"
title={null}
>
<svg
aria-hidden={true}
aria-labelledby={null}
fill="currentColor"
height="1em"
role="img"
style={
Object {
"verticalAlign": "-0.125em",
}
}
viewBox="0 0 352 512"
width="1em"
>
<path
d="M242.72 256l100.07-100.07c12.28-12.28 12.28-32.19 0-44.48l-22.24-22.24c-12.28-12.28-32.19-12.28-44.48 0L176 189.28 75.93 89.21c-12.28-12.28-32.19-12.28-44.48 0L9.21 111.45c-12.28 12.28-12.28 32.19 0 44.48L109.28 256 9.21 356.07c-12.28 12.28-12.28 32.19 0 44.48l22.24 22.24c12.28 12.28 32.2 12.28 44.48 0L176 322.72l100.07 100.07c12.28 12.28 32.2 12.28 44.48 0l22.24-22.24c12.28-12.28 12.28-32.19 0-44.48L242.72 256z"
transform=""
/>
</svg>
</TimesIcon>
</button>
</Button>
</ModalBoxCloseButton>
<ModalBoxHeader
className=""
hideTitle={false}
>
<Title
className=""
headingLevel="h3"
size="2xl"
>
<h3
className="pf-c-title pf-m-2xl"
>
Remove Team Access
</h3>
</Title>
</ModalBoxHeader>
<ModalBoxBody
className=""
id="pf-modal-0"
>
<div
className="pf-c-modal-box__body"
id="pf-modal-0"
>
<WithI18n
components={
Array [
<b />,
<b />,
<br />,
<br />,
<b />,
<i />,
]
}
id="Are you sure you want to remove<0> {0} </0>access from<1> {1}</1>? Doing so affects all members of the team.<2/><3/>If you<4><5> only </5></4>want to remove access for this particular user, please remove them from the team."
values={
Object {
"0": "Member",
"1": "The Team",
}
}
>
<I18n
update={true}
withHash={true}
>
<Trans
components={
Array [
<b />,
<b />,
<br />,
<br />,
<b />,
<i />,
]
}
i18n={"/i18n/"}
id="Are you sure you want to remove<0> {0} </0>access from<1> {1}</1>? Doing so affects all members of the team.<2/><3/>If you<4><5> only </5></4>want to remove access for this particular user, please remove them from the team."
values={
Object {
"0": "Member",
"1": "The Team",
}
}
>
<Render
value={null}
/>
</Trans>
</I18n>
</WithI18n>
<ExclamationCircleIcon
className="at-c-alertModal__icon"
color="currentColor" color="currentColor"
size="sm" size="sm"
title={null} title={null}
@@ -433,7 +330,6 @@ exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = `
<svg <svg
aria-hidden={true} aria-hidden={true}
aria-labelledby={null} aria-labelledby={null}
className="at-c-alertModal__icon"
fill="currentColor" fill="currentColor"
height="1em" height="1em"
role="img" role="img"
@@ -442,91 +338,150 @@ exports[`<DeleteRoleConfirmationModal /> should render initially 1`] = `
"verticalAlign": "-0.125em", "verticalAlign": "-0.125em",
} }
} }
viewBox="0 0 512 512" viewBox="0 0 352 512"
width="1em" width="1em"
> >
<path <path
d="M504 256c0 136.997-111.043 248-248 248S8 392.997 8 256C8 119.083 119.043 8 256 8s248 111.083 248 248zm-248 50c-25.405 0-46 20.595-46 46s20.595 46 46 46 46-20.595 46-46-20.595-46-46-46zm-43.673-165.346l7.418 136c.347 6.364 5.609 11.346 11.982 11.346h48.546c6.373 0 11.635-4.982 11.982-11.346l7.418-136c.375-6.874-5.098-12.654-11.982-12.654h-63.383c-6.884 0-12.356 5.78-11.981 12.654z" d="M242.72 256l100.07-100.07c12.28-12.28 12.28-32.19 0-44.48l-22.24-22.24c-12.28-12.28-32.19-12.28-44.48 0L176 189.28 75.93 89.21c-12.28-12.28-32.19-12.28-44.48 0L9.21 111.45c-12.28 12.28-12.28 32.19 0 44.48L109.28 256 9.21 356.07c-12.28 12.28-12.28 32.19 0 44.48l22.24 22.24c12.28 12.28 32.2 12.28 44.48 0L176 322.72l100.07 100.07c12.28 12.28 32.2 12.28 44.48 0l22.24-22.24c12.28-12.28 12.28-32.19 0-44.48L242.72 256z"
transform="" transform=""
/> />
</svg> </svg>
</ExclamationCircleIcon> </TimesIcon>
</div> </button>
</ModalBoxBody> </Button>
<ModalBoxFooter </ModalBoxCloseButton>
<ModalBoxHeader
className=""
hideTitle={false}
>
<Title
className="" className=""
headingLevel="h3"
size="2xl"
> >
<div <h3
className="pf-c-modal-box__footer" className="pf-c-title pf-m-2xl"
> >
<Button Remove {0} Access
aria-label="Confirm delete"
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
key="delete"
onClick={[Function]}
type="button"
variant="danger"
>
<button
aria-disabled={null}
aria-label="Confirm delete"
className="pf-c-button pf-m-danger"
disabled={false}
onClick={[Function]}
tabIndex={null}
type="button"
>
Delete
</button>
</Button>
<Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
key="cancel"
onClick={[Function]}
type="button"
variant="secondary"
>
<button
aria-disabled={null}
aria-label={null}
className="pf-c-button pf-m-secondary"
disabled={false}
onClick={[Function]}
tabIndex={null}
type="button"
>
Cancel
</button>
</Button>
</div> </h3>
</ModalBoxFooter> </Title>
</div> </ModalBoxHeader>
</ModalBox> <ModalBoxBody
</div> className=""
</FocusTrap> id="pf-modal-0"
</div> >
</Bullseye> <div
</div> className="pf-c-modal-box__body"
</Backdrop> id="pf-modal-0"
</ModalContent> >
</Portal> Are you sure you want to remove {0} access from {1}? Doing so affects all members of the team.
</Modal> <br />
</_default> <br />
</I18n> If you {0} want to remove access for this particular user, please remove them from the team.
<ExclamationCircleIcon
className="at-c-alertModal__icon"
color="currentColor"
size="sm"
title={null}
>
<svg
aria-hidden={true}
aria-labelledby={null}
className="at-c-alertModal__icon"
fill="currentColor"
height="1em"
role="img"
style={
Object {
"verticalAlign": "-0.125em",
}
}
viewBox="0 0 512 512"
width="1em"
>
<path
d="M504 256c0 136.997-111.043 248-248 248S8 392.997 8 256C8 119.083 119.043 8 256 8s248 111.083 248 248zm-248 50c-25.405 0-46 20.595-46 46s20.595 46 46 46 46-20.595 46-46-20.595-46-46-46zm-43.673-165.346l7.418 136c.347 6.364 5.609 11.346 11.982 11.346h48.546c6.373 0 11.635-4.982 11.982-11.346l7.418-136c.375-6.874-5.098-12.654-11.982-12.654h-63.383c-6.884 0-12.356 5.78-11.981 12.654z"
transform=""
/>
</svg>
</ExclamationCircleIcon>
</div>
</ModalBoxBody>
<ModalBoxFooter
className=""
>
<div
className="pf-c-modal-box__footer"
>
<Button
aria-label="Confirm delete"
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
key="delete"
onClick={[Function]}
type="button"
variant="danger"
>
<button
aria-disabled={null}
aria-label="Confirm delete"
className="pf-c-button pf-m-danger"
disabled={false}
onClick={[Function]}
tabIndex={null}
type="button"
>
Delete
</button>
</Button>
<Button
aria-label={null}
className=""
component="button"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
key="cancel"
onClick={[Function]}
type="button"
variant="secondary"
>
<button
aria-disabled={null}
aria-label={null}
className="pf-c-button pf-m-secondary"
disabled={false}
onClick={[Function]}
tabIndex={null}
type="button"
>
Cancel
</button>
</Button>
</div>
</ModalBoxFooter>
</div>
</ModalBox>
</div>
</FocusTrap>
</div>
</Bullseye>
</div>
</Backdrop>
</ModalContent>
</Portal>
</Modal>
</_default>
</DeleteRoleConfirmationModal> </DeleteRoleConfirmationModal>
`; `;

View File

@@ -29,43 +29,146 @@ exports[`<OrganizationAccessItem /> initially renders succesfully 1`] = `
"username": "jane", "username": "jane",
} }
} }
i18n={"/i18n/"}
onRoleDelete={[Function]} onRoleDelete={[Function]}
> >
<I18n <DataListItem
update={true} aria-labelledby="access-list-item"
withHash={true} className=""
isExpanded={false}
key="2"
> >
<DataListItem <li
aria-labelledby="access-list-item" aria-labelledby="access-list-item"
className="" className="pf-c-data-list__item"
isExpanded={false}
key="2"
> >
<li <DataListItemRow
aria-labelledby="access-list-item" className=""
className="pf-c-data-list__item" key=".0"
rowid="access-list-item"
> >
<DataListItemRow <div
className="" className="pf-c-data-list__item-row"
key=".0"
rowid="access-list-item"
> >
<div <DataListItemCells
className="pf-c-data-list__item-row" className=""
> dataListCells={
<DataListItemCells Array [
className="" <DataListCell
dataListCells={ alignRight={false}
Array [ className=""
<DataListCell isFilled={true}
alignRight={false} isIcon={false}
width={1}
>
<TextContent
className="" className=""
isFilled={true} style={
isIcon={false} Object {
width={1} "display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
> >
<TextContent <Link
replace={false}
to={
Object {
"pathname": "/bar",
}
}
>
<Text
className=""
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
jane
</Text>
</Link>
</TextContent>
<Detail
customStyles={null}
label="Name"
url={null}
value="jane brown"
/>
</DataListCell>,
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
width={1}
>
<ul
style={
Object {
"display": "flex",
"flexWrap": "wrap",
}
}
>
<Text
className="" className=""
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
Team Roles
</Text>
<Chip
className="awx-c-chip"
closeBtnAriaLabel="close"
isOverflowChip={false}
onClick={[Function]}
tooltipPosition="top"
>
Member
</Chip>
</ul>
</DataListCell>,
]
}
key=".0"
rowid="access-list-item"
>
<div
className="pf-c-data-list__item-content"
>
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
key="name"
width={1}
>
<div
className="pf-c-data-list__cell"
>
<TextContent
className=""
style={
Object {
"display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
>
<div
className="pf-c-content"
style={ style={
Object { Object {
"display": "grid", "display": "grid",
@@ -81,167 +184,8 @@ exports[`<OrganizationAccessItem /> initially renders succesfully 1`] = `
} }
} }
> >
<Text <a
className="" onClick={[Function]}
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
jane
</Text>
</Link>
</TextContent>
<Detail
customStyles={null}
label="Name"
url={null}
value="jane brown"
/>
</DataListCell>,
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
width={1}
>
<ul
style={
Object {
"display": "flex",
"flexWrap": "wrap",
}
}
>
<Text
className=""
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
Team Roles
</Text>
<Chip
className="awx-c-chip"
closeBtnAriaLabel="close"
isOverflowChip={false}
onClick={[Function]}
tooltipPosition="top"
>
Member
</Chip>
</ul>
</DataListCell>,
]
}
key=".0"
rowid="access-list-item"
>
<div
className="pf-c-data-list__item-content"
>
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
key="name"
width={1}
>
<div
className="pf-c-data-list__cell"
>
<TextContent
className=""
style={
Object {
"display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
>
<div
className="pf-c-content"
style={
Object {
"display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
>
<Link
replace={false}
to={
Object {
"pathname": "/bar",
}
}
>
<a
onClick={[Function]}
>
<Text
className=""
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
<h6
className=""
data-pf-content={true}
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
jane
</h6>
</Text>
</a>
</Link>
</div>
</TextContent>
<Detail
customStyles={null}
label="Name"
url={null}
value="jane brown"
>
<TextContent
className=""
style={
Object {
"display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
>
<div
className="pf-c-content"
style={
Object {
"display": "grid",
"gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
}
}
> >
<Text <Text
className="" className=""
@@ -265,70 +209,40 @@ exports[`<OrganizationAccessItem /> initially renders succesfully 1`] = `
} }
} }
> >
Name jane
</h6> </h6>
</Text> </Text>
<Text </a>
className="" </Link>
component="p" </div>
style={ </TextContent>
Object { <Detail
"lineHeight": "28px", customStyles={null}
"overflow": "visible", label="Name"
} url={null}
} value="jane brown"
>
<p
className=""
data-pf-content={true}
style={
Object {
"lineHeight": "28px",
"overflow": "visible",
}
}
>
jane brown
</p>
</Text>
</div>
</TextContent>
</Detail>
</div>
</DataListCell>
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
key="roles"
width={1}
>
<div
className="pf-c-data-list__cell"
> >
<ul <TextContent
className=""
style={ style={
Object { Object {
"display": "flex", "display": "grid",
"flexWrap": "wrap", "gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
} }
} }
> >
<Text <div
className="" className="pf-c-content"
component="h6"
style={ style={
Object { Object {
"fontWeight": "700", "display": "grid",
"lineHeight": "24px", "gridTemplateColumns": "minmax(70px, max-content) minmax(60px, max-content)",
"marginRight": "20px",
} }
} }
> >
<h6 <Text
className="" className=""
data-pf-content={true} component="h6"
style={ style={
Object { Object {
"fontWeight": "700", "fontWeight": "700",
@@ -337,103 +251,185 @@ exports[`<OrganizationAccessItem /> initially renders succesfully 1`] = `
} }
} }
> >
Team Roles <h6
</h6> className=""
</Text> data-pf-content={true}
<Chip style={
className="awx-c-chip" Object {
closeBtnAriaLabel="close" "fontWeight": "700",
isOverflowChip={false} "lineHeight": "24px",
key="3" "marginRight": "20px",
onClick={[Function]} }
tooltipPosition="top" }
>
<GenerateId
prefix="pf-random-id-"
>
<li
className="pf-c-chip awx-c-chip"
> >
<span Name
className="pf-c-chip__text" </h6>
id="pf-random-id-0" </Text>
> <Text
Member className=""
</span> component="p"
<ChipButton style={
Object {
"lineHeight": "28px",
"overflow": "visible",
}
}
>
<p
className=""
data-pf-content={true}
style={
Object {
"lineHeight": "28px",
"overflow": "visible",
}
}
>
jane brown
</p>
</Text>
</div>
</TextContent>
</Detail>
</div>
</DataListCell>
<DataListCell
alignRight={false}
className=""
isFilled={true}
isIcon={false}
key="roles"
width={1}
>
<div
className="pf-c-data-list__cell"
>
<ul
style={
Object {
"display": "flex",
"flexWrap": "wrap",
}
}
>
<Text
className=""
component="h6"
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
<h6
className=""
data-pf-content={true}
style={
Object {
"fontWeight": "700",
"lineHeight": "24px",
"marginRight": "20px",
}
}
>
Team Roles
</h6>
</Text>
<Chip
className="awx-c-chip"
closeBtnAriaLabel="close"
isOverflowChip={false}
key="3"
onClick={[Function]}
tooltipPosition="top"
>
<GenerateId
prefix="pf-random-id-"
>
<li
className="pf-c-chip awx-c-chip"
>
<span
className="pf-c-chip__text"
id="pf-random-id-0"
>
Member
</span>
<ChipButton
aria-labelledby="remove_pf-random-id-0 pf-random-id-0"
ariaLabel="close"
className=""
id="remove_pf-random-id-0"
onClick={[Function]}
>
<Button
aria-label="close"
aria-labelledby="remove_pf-random-id-0 pf-random-id-0" aria-labelledby="remove_pf-random-id-0 pf-random-id-0"
ariaLabel="close"
className="" className=""
component="button"
id="remove_pf-random-id-0" id="remove_pf-random-id-0"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]} onClick={[Function]}
type="button"
variant="plain"
> >
<Button <button
aria-disabled={null}
aria-label="close" aria-label="close"
aria-labelledby="remove_pf-random-id-0 pf-random-id-0" aria-labelledby="remove_pf-random-id-0 pf-random-id-0"
className="" className="pf-c-button pf-m-plain"
component="button" disabled={false}
id="remove_pf-random-id-0" id="remove_pf-random-id-0"
isActive={false}
isBlock={false}
isDisabled={false}
isFocus={false}
isHover={false}
onClick={[Function]} onClick={[Function]}
tabIndex={null}
type="button" type="button"
variant="plain"
> >
<button <TimesCircleIcon
aria-disabled={null} aria-hidden="true"
aria-label="close" color="currentColor"
aria-labelledby="remove_pf-random-id-0 pf-random-id-0" size="sm"
className="pf-c-button pf-m-plain" title={null}
disabled={false}
id="remove_pf-random-id-0"
onClick={[Function]}
tabIndex={null}
type="button"
> >
<TimesCircleIcon <svg
aria-hidden="true" aria-hidden="true"
color="currentColor" aria-labelledby={null}
size="sm" fill="currentColor"
title={null} height="1em"
> role="img"
<svg style={
aria-hidden="true" Object {
aria-labelledby={null} "verticalAlign": "-0.125em",
fill="currentColor"
height="1em"
role="img"
style={
Object {
"verticalAlign": "-0.125em",
}
} }
viewBox="0 0 512 512" }
width="1em" viewBox="0 0 512 512"
> width="1em"
<path >
d="M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm121.6 313.1c4.7 4.7 4.7 12.3 0 17L338 377.6c-4.7 4.7-12.3 4.7-17 0L256 312l-65.1 65.6c-4.7 4.7-12.3 4.7-17 0L134.4 338c-4.7-4.7-4.7-12.3 0-17l65.6-65-65.6-65.1c-4.7-4.7-4.7-12.3 0-17l39.6-39.6c4.7-4.7 12.3-4.7 17 0l65 65.7 65.1-65.6c4.7-4.7 12.3-4.7 17 0l39.6 39.6c4.7 4.7 4.7 12.3 0 17L312 256l65.6 65.1z" <path
transform="" d="M256 8C119 8 8 119 8 256s111 248 248 248 248-111 248-248S393 8 256 8zm121.6 313.1c4.7 4.7 4.7 12.3 0 17L338 377.6c-4.7 4.7-12.3 4.7-17 0L256 312l-65.1 65.6c-4.7 4.7-12.3 4.7-17 0L134.4 338c-4.7-4.7-4.7-12.3 0-17l65.6-65-65.6-65.1c-4.7-4.7-4.7-12.3 0-17l39.6-39.6c4.7-4.7 12.3-4.7 17 0l65 65.7 65.1-65.6c4.7-4.7 12.3-4.7 17 0l39.6 39.6c4.7 4.7 4.7 12.3 0 17L312 256l65.6 65.1z"
/> transform=""
</svg> />
</TimesCircleIcon> </svg>
</button> </TimesCircleIcon>
</Button> </button>
</ChipButton> </Button>
</li> </ChipButton>
</GenerateId> </li>
</Chip> </GenerateId>
</ul> </Chip>
</div> </ul>
</DataListCell> </div>
</div> </DataListCell>
</DataListItemCells> </div>
</div> </DataListItemCells>
</DataListItemRow> </div>
</li> </DataListItemRow>
</DataListItem> </li>
</I18n> </DataListItem>
</OrganizationAccessItem> </OrganizationAccessItem>
`; `;

View File

@@ -5,6 +5,7 @@ exports[`<OrganizationAccess /> initially renders succesfully 1`] = `
api={"/api/"} api={"/api/"}
handleHttpError={[Function]} handleHttpError={[Function]}
history={"/history/"} history={"/history/"}
i18n={"/i18n/"}
location={ location={
Object { Object {
"hash": "", "hash": "",

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Portal from '../../src/pages/Portal'; import Portal from '../../src/pages/Portal';
describe('<Portal />', () => { describe('<Portal />', () => {
@@ -8,7 +8,7 @@ describe('<Portal />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Portal />); pageWrapper = mountWithContexts(<Portal />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Projects from '../../src/pages/Projects'; import Projects from '../../src/pages/Projects';
describe('<Projects />', () => { describe('<Projects />', () => {
@@ -8,7 +8,7 @@ describe('<Projects />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Projects />); pageWrapper = mountWithContexts(<Projects />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Schedules from '../../src/pages/Schedules'; import Schedules from '../../src/pages/Schedules';
describe('<Schedules />', () => { describe('<Schedules />', () => {
@@ -8,7 +8,7 @@ describe('<Schedules />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Schedules />); pageWrapper = mountWithContexts(<Schedules />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import SystemSettings from '../../src/pages/SystemSettings'; import SystemSettings from '../../src/pages/SystemSettings';
describe('<SystemSettings />', () => { describe('<SystemSettings />', () => {
@@ -8,7 +8,7 @@ describe('<SystemSettings />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<SystemSettings />); pageWrapper = mountWithContexts(<SystemSettings />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Teams from '../../src/pages/Teams'; import Teams from '../../src/pages/Teams';
describe('<Teams />', () => { describe('<Teams />', () => {
@@ -8,7 +8,7 @@ describe('<Teams />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Teams />); pageWrapper = mountWithContexts(<Teams />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Templates from '../../src/pages/Templates'; import Templates from '../../src/pages/Templates';
describe('<Templates />', () => { describe('<Templates />', () => {
@@ -8,7 +8,7 @@ describe('<Templates />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Templates />); pageWrapper = mountWithContexts(<Templates />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import UISettings from '../../src/pages/UISettings'; import UISettings from '../../src/pages/UISettings';
describe('<UISettings />', () => { describe('<UISettings />', () => {
@@ -8,7 +8,7 @@ describe('<UISettings />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<UISettings />); pageWrapper = mountWithContexts(<UISettings />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,5 +1,5 @@
import React from 'react'; import React from 'react';
import { mount } from 'enzyme'; import { mountWithContexts } from '../enzymeHelpers';
import Users from '../../src/pages/Users'; import Users from '../../src/pages/Users';
describe('<Users />', () => { describe('<Users />', () => {
@@ -8,7 +8,7 @@ describe('<Users />', () => {
let title; let title;
beforeEach(() => { beforeEach(() => {
pageWrapper = mount(<Users />); pageWrapper = mountWithContexts(<Users />);
pageSections = pageWrapper.find('PageSection'); pageSections = pageWrapper.find('PageSection');
title = pageWrapper.find('Title'); title = pageWrapper.find('Title');
}); });

View File

@@ -1,34 +1,36 @@
import { required, maxLength } from '../../src/util/validators'; import { required, maxLength } from '../../src/util/validators';
const i18n = { _: val => val };
describe('validators', () => { describe('validators', () => {
test('required returns undefined if value given', () => { test('required returns undefined if value given', () => {
expect(required()('some value')).toBeUndefined(); expect(required(null, i18n)('some value')).toBeUndefined();
expect(required('oops')('some value')).toBeUndefined(); expect(required('oops', i18n)('some value')).toBeUndefined();
}); });
test('required returns default message if value missing', () => { test('required returns default message if value missing', () => {
expect(required()('')).toEqual('This field must not be blank'); expect(required(null, i18n)('')).toEqual({ id: 'This field must not be blank' });
}); });
test('required returns custom message if value missing', () => { test('required returns custom message if value missing', () => {
expect(required('oops')('')).toEqual('oops'); expect(required('oops', i18n)('')).toEqual('oops');
}); });
test('required interprets white space as empty value', () => { test('required interprets white space as empty value', () => {
expect(required()(' ')).toEqual('This field must not be blank'); expect(required(null, i18n)(' ')).toEqual({ id: 'This field must not be blank' });
expect(required()('\t')).toEqual('This field must not be blank'); expect(required(null, i18n)('\t')).toEqual({ id: 'This field must not be blank' });
}); });
test('maxLength accepts value below max', () => { test('maxLength accepts value below max', () => {
expect(maxLength(10)('snazzy')).toBeUndefined(); expect(maxLength(10, i18n)('snazzy')).toBeUndefined();
}); });
test('maxLength accepts value equal to max', () => { test('maxLength accepts value equal to max', () => {
expect(maxLength(10)('abracadbra')).toBeUndefined(); expect(maxLength(10, i18n)('abracadbra')).toBeUndefined();
}); });
test('maxLength rejects value above max', () => { test('maxLength rejects value above max', () => {
expect(maxLength(8)('abracadbra')) expect(maxLength(8, i18n)('abracadbra'))
.toEqual('This field must not exceed 8 characters'); .toEqual({ id: 'This field must not exceed {max} characters', values: { max: 8 } });
}); });
}); });

View File

@@ -13,6 +13,10 @@ msgstr ""
"Language-Team: \n" "Language-Team: \n"
"Plural-Forms: \n" "Plural-Forms: \n"
#: src/contexts/Network.jsx:56
msgid "404"
msgstr ""
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:60 #: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:60
#~ msgid "> add" #~ msgid "> add"
#~ msgstr "" #~ msgstr ""
@@ -25,20 +29,34 @@ msgstr ""
msgid "About" msgid "About"
msgstr "" msgstr ""
#: src/components/About.jsx:60 #: src/components/About.jsx:59
msgid "AboutModal Logo" msgid "AboutModal Logo"
msgstr "" msgstr ""
#: src/index.jsx:175 #: src/index.jsx:140
#: src/pages/Organizations/Organizations.jsx:43
#: src/pages/Organizations/screens/Organization/Organization.jsx:131
msgid "Access" msgid "Access"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:266 #: src/components/PaginatedDataList/ToolbarAddButton.jsx:33
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:68 #: src/components/PaginatedDataList/ToolbarAddButton.jsx:44
msgid "Add" msgid "Add"
msgstr "" msgstr ""
#: src/index.jsx:196 #: src/components/AddRole/AddResourceRole.jsx:153
msgid "Add Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:150
msgid "Add Team Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:147
msgid "Add User Roles"
msgstr ""
#: src/index.jsx:161
msgid "Administration" msgid "Administration"
msgstr "" msgstr ""
@@ -46,67 +64,154 @@ msgstr ""
#~ msgid "Admins" #~ msgid "Admins"
#~ msgstr "" #~ msgstr ""
#: src/components/About.jsx:77 #: src/pages/Organizations/components/OrganizationForm.jsx:123
#: src/pages/Organizations/components/OrganizationForm.jsx:128
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:142
msgid "Ansible Environment"
msgstr ""
#: src/components/About.jsx:75
msgid "Ansible Version" msgid "Ansible Version"
msgstr "" msgstr ""
#: src/pages/Applications.jsx:17 #: src/pages/Applications.jsx:19
msgid "Applications" msgid "Applications"
msgstr "" msgstr ""
#: src/index.jsx:231 #: src/components/AddRole/AddResourceRole.jsx:209
msgid "Apply roles"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:165
msgid "Are you sure you want to delete:"
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:51
msgid "Are you sure you want to remove {0} access from {1}? Doing so affects all members of the team."
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:58
msgid "Are you sure you want to remove {0} access from {username}?"
msgstr ""
#: src/index.jsx:196
msgid "Authentication" msgid "Authentication"
msgstr "" msgstr ""
#: src/pages/AuthSettings.jsx:17 #: src/pages/AuthSettings.jsx:19
msgid "Authentication Settings" msgid "Authentication Settings"
msgstr "" msgstr ""
#: src/components/About.jsx:58 #: src/components/About.jsx:57
msgid "Brand Image" msgid "Brand Image"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:241 #: src/components/FormActionGroup/FormActionGroup.jsx:30
#: src/components/FormActionGroup/FormActionGroup.jsx:30
#: src/components/Lookup/Lookup.jsx:169
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:161
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:45
msgid "Cancel"
msgstr ""
#: src/pages/Organizations/Organizations.jsx:77
msgid "Cannot find organization with ID"
msgstr ""
#: src/contexts/Network.jsx:57
msgid "Cannot find resource."
msgstr ""
#: src/components/NotifyAndRedirect.jsx:23
msgid "Cannot find route {0}."
msgstr ""
#: src/App.jsx:94
#: src/components/CardCloseButton.jsx:14
#: src/components/CardCloseButton.jsx:15
#: src/components/CardCloseButton.jsx:27
#: src/components/Lookup/Lookup.jsx:169
#: src/pages/Organizations/screens/OrganizationAdd.jsx:67
msgid "Close"
msgstr ""
#: src/components/ExpandCollapse/ExpandCollapse.jsx:34
msgid "Collapse" msgid "Collapse"
msgstr "" msgstr ""
#: src/components/About.jsx:56 #: src/components/About.jsx:55
msgid "Copyright 2018 Red Hat, Inc." msgid "Copyright 2018 Red Hat, Inc."
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:29 #: src/pages/Organizations/Organizations.jsx:39
#: src/pages/Organizations/Organizations.jsx:25
msgid "Create New Organization"
msgstr ""
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:58
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:146
#: src/pages/Organizations/screens/OrganizationsList.jsx:164
msgid "Created" msgid "Created"
msgstr "" msgstr ""
#: src/index.jsx:200 #: src/index.jsx:165
#: src/pages/CredentialTypes.jsx:17 #: src/pages/CredentialTypes.jsx:19
msgid "Credential Types" msgid "Credential Types"
msgstr "" msgstr ""
#: src/index.jsx:153 #: src/index.jsx:118
#: src/pages/Credentials.jsx:17 #: src/pages/Credentials.jsx:19
msgid "Credentials" msgid "Credentials"
msgstr "" msgstr ""
#: src/index.jsx:122 #: src/index.jsx:87
#: src/pages/Dashboard.jsx:17 #: src/pages/Dashboard.jsx:19
msgid "Dashboard" msgid "Dashboard"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:252 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:109
#: src/components/DataListToolbar/DataListToolbar.jsx:257 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:130
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:153
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:42
msgid "Delete" msgid "Delete"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:59 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:142
msgid "Delete {0}"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:141
msgid "Delete {itemName}"
msgstr ""
#: src/pages/Organizations/components/OrganizationForm.jsx:113
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:138
msgid "Description"
msgstr ""
#: src/pages/Organizations/Organizations.jsx:42
#: src/pages/Organizations/screens/Organization/Organization.jsx:130
msgid "Details"
msgstr ""
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:171
msgid "Edit" msgid "Edit"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:233 #: src/pages/Organizations/Organizations.jsx:41
msgid "Edit Details"
msgstr ""
#: src/components/ExpandCollapse/ExpandCollapse.jsx:44
msgid "Expand" msgid "Expand"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:169 #: src/components/NotificationsList/NotificationListItem.jsx:82
#: src/components/NotificationsList/NotificationListItem.jsx:97
msgid "Failure"
msgstr ""
#: src/components/Pagination/Pagination.jsx:170
msgid "First" msgid "First"
msgstr "" msgstr ""
@@ -114,260 +219,436 @@ msgstr ""
msgid "Help" msgid "Help"
msgstr "" msgstr ""
#: src/index.jsx:215 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:54
#: src/pages/InstanceGroups.jsx:17 msgid "If you {0} want to remove access for this particular user, please remove them from the team."
msgstr ""
#: src/components/PageHeaderToolbar.jsx:66
msgid "Info"
msgstr ""
#: src/index.jsx:180
#: src/pages/InstanceGroups.jsx:19
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:32
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:50
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:159
msgid "Instance Groups" msgid "Instance Groups"
msgstr "" msgstr ""
#: src/index.jsx:220 #: src/index.jsx:185
msgid "Integrations" msgid "Integrations"
msgstr "" msgstr ""
#: src/pages/Login.jsx:80 #: src/pages/Login.jsx:84
msgid "Invalid username or password. Please try again." msgid "Invalid username or password. Please try again."
msgstr "" msgstr ""
#: src/index.jsx:163 #: src/index.jsx:128
#: src/pages/Inventories.jsx:17 #: src/pages/Inventories.jsx:19
msgid "Inventories" msgid "Inventories"
msgstr "" msgstr ""
#: src/index.jsx:168 #: src/index.jsx:133
#: src/pages/InventoryScripts.jsx:17 #: src/pages/InventoryScripts.jsx:19
msgid "Inventory Scripts" msgid "Inventory Scripts"
msgstr "" msgstr ""
#: src/index.jsx:127 #: src/components/Pagination/Pagination.jsx:142
#: src/index.jsx:236 msgid "Items Per Page"
#: src/pages/Jobs.jsx:17 msgstr ""
#: src/components/Pagination/Pagination.jsx:162
msgid "Items {itemMin} {itemMax} of {count}"
msgstr ""
#: src/index.jsx:92
#: src/index.jsx:201
#: src/pages/Jobs.jsx:19
msgid "Jobs" msgid "Jobs"
msgstr "" msgstr ""
#: src/pages/JobsSettings.jsx:17 #: src/pages/JobsSettings.jsx:19
msgid "Jobs Settings" msgid "Jobs Settings"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:221 #: src/components/Pagination/Pagination.jsx:213
msgid "Last" msgid "Last"
msgstr "" msgstr ""
#: src/index.jsx:251 #: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:150
#: src/pages/License.jsx:17 msgid "Last Modified"
msgstr ""
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:173
msgid "Last Name"
msgstr ""
#: src/index.jsx:216
#: src/pages/License.jsx:19
msgid "License" msgid "License"
msgstr "" msgstr ""
#: src/components/PageHeaderToolbar.jsx:116 #: src/components/AddRole/SelectResourceStep.jsx:96
msgid "Loading..."
msgstr ""
#: src/components/PageHeaderToolbar.jsx:120
msgid "Logout" msgid "Logout"
msgstr "" msgstr ""
#: src/index.jsx:210 #: src/index.jsx:175
#: src/pages/ManagementJobs.jsx:17 #: src/pages/ManagementJobs.jsx:19
msgid "Management Jobs" msgid "Management Jobs"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:28 #: src/pages/Organizations/components/OrganizationListItem.jsx:91
msgid "Members"
msgstr ""
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:57
#: src/pages/Organizations/screens/OrganizationsList.jsx:163
msgid "Modified" msgid "Modified"
msgstr "" msgstr ""
#: src/index.jsx:137 #: src/index.jsx:102
#: src/pages/Portal.jsx:17 #: src/pages/Portal.jsx:19
msgid "My View" msgid "My View"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:27 #: src/components/AddRole/AddResourceRole.jsx:140
#: src/components/PaginatedDataList/PaginatedDataList.jsx:116
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:56
#: src/pages/Organizations/components/OrganizationAccessItem.jsx:109
#: src/pages/Organizations/components/OrganizationForm.jsx:105
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:171
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:134
#: src/pages/Organizations/screens/OrganizationsList.jsx:162
msgid "Name" msgid "Name"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:212 #: src/components/Pagination/Pagination.jsx:204
msgid "Next" msgid "Next"
msgstr "" msgstr ""
#: src/pages/NotificationTemplates.jsx:17 #: src/components/PaginatedDataList/PaginatedDataList.jsx:133
#: src/components/PaginatedDataList/PaginatedDataList.jsx:196
msgid "No {0} Found"
msgstr ""
#: src/pages/NotificationTemplates.jsx:19
msgid "Notification Templates" msgid "Notification Templates"
msgstr "" msgstr ""
#: src/index.jsx:205 #: src/index.jsx:170
#: src/pages/Organizations/Organizations.jsx:45
#: src/pages/Organizations/screens/Organization/Organization.jsx:137
msgid "Notifications" msgid "Notifications"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organization.add.jsx:79 #: src/pages/Organizations/views/Organization.add.jsx:79
msgid "Organization Add" #~ msgid "Organization Add"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:72 #: src/pages/Organizations/screens/Organization/Organization.jsx:153
msgid "Organization detail tabs" msgid "Organization detail tabs"
msgstr "" msgstr ""
#: src/index.jsx:179 #: src/index.jsx:144
#: src/pages/Organizations/views/Organization.view.jsx:63 #: src/pages/Organizations/Organizations.jsx:38
#: src/pages/Organizations/views/Organizations.list.jsx:196 #: src/pages/Organizations/Organizations.jsx:24
#: src/pages/Organizations/views/Organizations.list.jsx:202
msgid "Organizations" msgid "Organizations"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:218 #: src/pages/Organizations/views/Organizations.list.jsx:218
msgid "Organizations List" #~ msgid "Organizations List"
#~ msgstr ""
#: src/components/Pagination/Pagination.jsx:190
msgid "Page"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:189 #: src/components/Pagination/Pagination.jsx:189
msgid "Page <0/> of {pageCount}" #~ msgid "Page <0/> of {pageCount}"
msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:192 #: src/components/Pagination/Pagination.jsx:193
msgid "Page Number" msgid "Page Number"
msgstr "" msgstr ""
#: src/pages/Login.jsx:79 #: src/pages/Login.jsx:82
msgid "Password" msgid "Password"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:158 #: src/components/Pagination/Pagination.jsx:158
msgid "Per Page" #~ msgid "Per Page"
#~ msgstr ""
#: src/components/PaginatedDataList/PaginatedDataList.jsx:136
#: src/components/PaginatedDataList/PaginatedDataList.jsx:199
msgid "Please add {0} {itemName} to populate this list"
msgstr "" msgstr ""
#: src/App.jsx:203 #: src/App.jsx:203
#~ msgid "Portal Mode" #~ msgid "Portal Mode"
#~ msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:178 #: src/components/Pagination/Pagination.jsx:179
msgid "Previous" msgid "Previous"
msgstr "" msgstr ""
#: src/index.jsx:115 #: src/index.jsx:80
msgid "Primary Navigation" msgid "Primary Navigation"
msgstr "" msgstr ""
#: src/index.jsx:158 #: src/index.jsx:123
#: src/pages/Projects.jsx:17 #: src/pages/Projects.jsx:19
msgid "Projects" msgid "Projects"
msgstr "" msgstr ""
#: src/index.jsx:144 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
msgid "Remove {0} Access"
msgstr ""
#: src/index.jsx:109
msgid "Resources" msgid "Resources"
msgstr "" msgstr ""
#: src/index.jsx:132 #: src/components/AddRole/AddResourceRole.jsx:220
#: src/pages/Schedules.jsx:17 #: src/components/FormActionGroup/FormActionGroup.jsx:27
#: src/components/FormActionGroup/FormActionGroup.jsx:27
#: src/components/Lookup/Lookup.jsx:168
msgid "Save"
msgstr ""
#: src/index.jsx:97
#: src/pages/Schedules.jsx:19
msgid "Schedules" msgid "Schedules"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:191 #: src/components/Search/Search.jsx:138
msgid "Search" msgid "Search"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:185 #: src/components/Search/Search.jsx:131
msgid "Search text input" msgid "Search text input"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:160 #: src/components/AnsibleSelect/AnsibleSelect.jsx:28
msgid "Select Input"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:159
msgid "Select Users Or Teams"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:111
msgid "Select a row to delete"
msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:102
msgid "Select all" msgid "Select all"
msgstr "" msgstr ""
#: src/index.jsx:227 #: src/components/AddRole/AddResourceRole.jsx:178
msgid "Select items from list"
msgstr ""
#: src/pages/Organizations/components/OrganizationForm.jsx:141
msgid "Select the Instance Groups for this Organization to run on."
msgstr ""
#: src/components/Lookup/Lookup.jsx:164
msgid "Select {header}"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:187
#: src/components/AddRole/AddResourceRole.jsx:198
#: src/components/AddRole/AddResourceRole.jsx:215
#: src/components/AddRole/SelectRoleStep.jsx:29
#: src/components/Lookup/Lookup.jsx:193
msgid "Selected"
msgstr ""
#: src/index.jsx:192
msgid "Settings" msgid "Settings"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:222 #: src/components/Sort/Sort.jsx:135
msgid "Sort" msgid "Sort"
msgstr "" msgstr ""
#: src/index.jsx:241 #: src/components/NotificationsList/NotificationListItem.jsx:70
msgid "Successful"
msgstr ""
#: src/index.jsx:206
msgid "System" msgid "System"
msgstr "" msgstr ""
#: src/pages/SystemSettings.jsx:17 #: src/pages/SystemSettings.jsx:19
msgid "System Settings" msgid "System Settings"
msgstr "" msgstr ""
#: src/index.jsx:189 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
#: src/pages/Organizations/components/OrganizationListItem.jsx:57 msgid "Team"
#: src/pages/Teams.jsx:17 msgstr ""
#: src/pages/Organizations/components/OrganizationAccessItem.jsx:144
msgid "Team Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:169
#: src/index.jsx:154
#: src/pages/Organizations/Organizations.jsx:44
#: src/pages/Organizations/components/OrganizationListItem.jsx:99
#: src/pages/Organizations/screens/Organization/Organization.jsx:132
#: src/pages/Teams.jsx:19
msgid "Teams" msgid "Teams"
msgstr "" msgstr ""
#: src/index.jsx:148 #: src/index.jsx:113
#: src/pages/Templates.jsx:17 #: src/pages/Templates.jsx:19
msgid "Templates" msgid "Templates"
msgstr "" msgstr ""
#: src/util/validators.jsx:6
msgid "This field must not be blank"
msgstr ""
#: src/util/validators.jsx:16
msgid "This field must not exceed {max} characters"
msgstr ""
#: src/components/NotificationsList/NotificationListItem.jsx:90
#: src/components/NotificationsList/NotificationListItem.jsx:105
msgid "Toggle notification failure"
msgstr ""
#: src/components/NotificationsList/NotificationListItem.jsx:78
msgid "Toggle notification success"
msgstr ""
#: src/components/TowerLogo/TowerLogo.jsx:48 #: src/components/TowerLogo/TowerLogo.jsx:48
msgid "Tower Brand Image" msgid "Tower Brand Image"
msgstr "" msgstr ""
#: src/components/PageHeaderToolbar.jsx:108 #: src/components/AnsibleSelect/AnsibleSelect.jsx:35
msgid "Use Default {label}"
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
msgid "User"
msgstr ""
#: src/components/PageHeaderToolbar.jsx:113
msgid "User Details" msgid "User Details"
msgstr "" msgstr ""
#: src/index.jsx:246 #: src/index.jsx:211
msgid "User Interface" msgid "User Interface"
msgstr "" msgstr ""
#: src/pages/UISettings.jsx:17 #: src/pages/UISettings.jsx:19
msgid "User Interface Settings" msgid "User Interface Settings"
msgstr "" msgstr ""
#: src/pages/Login.jsx:78 #: src/pages/Organizations/components/OrganizationAccessItem.jsx:122
msgid "User Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:136
#: src/pages/Login.jsx:81
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:172
msgid "Username" msgid "Username"
msgstr "" msgstr ""
#: src/index.jsx:184 #: src/components/AddRole/AddResourceRole.jsx:164
#: src/pages/Organizations/components/OrganizationListItem.jsx:49 #: src/index.jsx:149
#: src/pages/Users.jsx:17 #: src/pages/Users.jsx:19
msgid "Users" msgid "Users"
msgstr "" msgstr ""
#: src/index.jsx:118 #: src/index.jsx:83
msgid "Views" msgid "Views"
msgstr "" msgstr ""
#: src/pages/Login.jsx:74 #: src/pages/Login.jsx:76
msgid "Welcome to Ansible Tower! Please Sign In." msgid "Welcome to Ansible Tower! Please Sign In."
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:56 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:104
msgid "add {currentTab}" msgid "You do not have permission to delete the following {0}: {itemsUnableToDelete}"
msgstr "" msgstr ""
#: src/contexts/Network.jsx:44
msgid "You have been logged out."
msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:56
#~ msgid "add {currentTab}"
#~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:45 #: src/pages/Organizations/components/OrganizationDetail.jsx:45
msgid "adding {currentTab}" #~ msgid "adding {currentTab}"
#~ msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:158
msgid "cancel delete"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:150
msgid "confirm delete"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:38 #: src/pages/Organizations/components/OrganizationDetail.jsx:38
msgid "confirm removal of {currentTab}/cancel and go back to {currentTab} view." #~ msgid "confirm removal of {currentTab}/cancel and go back to {currentTab} view."
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:60 #: src/pages/Organizations/components/OrganizationDetail.jsx:60
msgid "delete {currentTab}" #~ msgid "delete {currentTab}"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:36 #: src/pages/Organizations/components/OrganizationDetail.jsx:36
msgid "deleting {currentTab} association with orgs" #~ msgid "deleting {currentTab} association with orgs"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationEdit.jsx:20 #: src/pages/Organizations/components/OrganizationEdit.jsx:20
msgid "edit view" #~ msgid "edit view"
#~ msgstr ""
#: src/components/Lookup/Lookup.jsx:135
msgid "items"
msgstr ""
#: src/components/Pagination/Pagination.jsx:198
msgid "of {pageCount}"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationEdit.jsx:22 #: src/pages/Organizations/components/OrganizationEdit.jsx:22
msgid "save/cancel and go back to view" #~ msgid "save/cancel and go back to view"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:47 #: src/pages/Organizations/components/OrganizationDetail.jsx:47
msgid "save/cancel and go back to {currentTab} view" #~ msgid "save/cancel and go back to {currentTab} view"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationListItem.jsx:29 #: src/pages/Organizations/components/OrganizationListItem.jsx:29
msgid "select organization {itemId}" #~ msgid "select organization {itemId}"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:82 #: src/pages/Organizations/components/OrganizationDetail.jsx:82
msgid "{0}" #~ msgid "{0}"
#~ msgstr ""
#: src/components/PaginatedDataList/PaginatedDataList.jsx:153
#: src/components/PaginatedDataList/PaginatedDataList.jsx:215
msgid "{0} List"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:54 #: src/pages/Organizations/components/OrganizationDetail.jsx:54
msgid "{currentTab} detail view" #~ msgid "{currentTab} detail view"
msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:163 #: src/components/Pagination/Pagination.jsx:163
msgid "{itemMin} - {itemMax} of {count}" #~ msgid "{itemMin} - {itemMax} of {count}"
msgstr "" #~ msgstr ""

View File

@@ -13,6 +13,10 @@ msgstr ""
"Language-Team: \n" "Language-Team: \n"
"Plural-Forms: \n" "Plural-Forms: \n"
#: src/contexts/Network.jsx:56
msgid "404"
msgstr ""
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:60 #: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:60
#~ msgid "> add" #~ msgid "> add"
#~ msgstr "" #~ msgstr ""
@@ -25,20 +29,34 @@ msgstr ""
msgid "About" msgid "About"
msgstr "" msgstr ""
#: src/components/About.jsx:60 #: src/components/About.jsx:59
msgid "AboutModal Logo" msgid "AboutModal Logo"
msgstr "" msgstr ""
#: src/index.jsx:175 #: src/index.jsx:140
#: src/pages/Organizations/Organizations.jsx:43
#: src/pages/Organizations/screens/Organization/Organization.jsx:131
msgid "Access" msgid "Access"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:266 #: src/components/PaginatedDataList/ToolbarAddButton.jsx:33
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:68 #: src/components/PaginatedDataList/ToolbarAddButton.jsx:44
msgid "Add" msgid "Add"
msgstr "" msgstr ""
#: src/index.jsx:196 #: src/components/AddRole/AddResourceRole.jsx:153
msgid "Add Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:150
msgid "Add Team Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:147
msgid "Add User Roles"
msgstr ""
#: src/index.jsx:161
msgid "Administration" msgid "Administration"
msgstr "" msgstr ""
@@ -46,67 +64,154 @@ msgstr ""
#~ msgid "Admins" #~ msgid "Admins"
#~ msgstr "" #~ msgstr ""
#: src/components/About.jsx:77 #: src/pages/Organizations/components/OrganizationForm.jsx:123
#: src/pages/Organizations/components/OrganizationForm.jsx:128
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:142
msgid "Ansible Environment"
msgstr ""
#: src/components/About.jsx:75
msgid "Ansible Version" msgid "Ansible Version"
msgstr "" msgstr ""
#: src/pages/Applications.jsx:17 #: src/pages/Applications.jsx:19
msgid "Applications" msgid "Applications"
msgstr "" msgstr ""
#: src/index.jsx:231 #: src/components/AddRole/AddResourceRole.jsx:209
msgid "Apply roles"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:165
msgid "Are you sure you want to delete:"
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:51
msgid "Are you sure you want to remove {0} access from {1}? Doing so affects all members of the team."
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:58
msgid "Are you sure you want to remove {0} access from {username}?"
msgstr ""
#: src/index.jsx:196
msgid "Authentication" msgid "Authentication"
msgstr "" msgstr ""
#: src/pages/AuthSettings.jsx:17 #: src/pages/AuthSettings.jsx:19
msgid "Authentication Settings" msgid "Authentication Settings"
msgstr "" msgstr ""
#: src/components/About.jsx:58 #: src/components/About.jsx:57
msgid "Brand Image" msgid "Brand Image"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:241 #: src/components/FormActionGroup/FormActionGroup.jsx:30
#: src/components/FormActionGroup/FormActionGroup.jsx:30
#: src/components/Lookup/Lookup.jsx:169
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:161
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:45
msgid "Cancel"
msgstr ""
#: src/pages/Organizations/Organizations.jsx:77
msgid "Cannot find organization with ID"
msgstr ""
#: src/contexts/Network.jsx:57
msgid "Cannot find resource."
msgstr ""
#: src/components/NotifyAndRedirect.jsx:23
msgid "Cannot find route {0}."
msgstr ""
#: src/App.jsx:94
#: src/components/CardCloseButton.jsx:14
#: src/components/CardCloseButton.jsx:15
#: src/components/CardCloseButton.jsx:27
#: src/components/Lookup/Lookup.jsx:169
#: src/pages/Organizations/screens/OrganizationAdd.jsx:67
msgid "Close"
msgstr ""
#: src/components/ExpandCollapse/ExpandCollapse.jsx:34
msgid "Collapse" msgid "Collapse"
msgstr "" msgstr ""
#: src/components/About.jsx:56 #: src/components/About.jsx:55
msgid "Copyright 2018 Red Hat, Inc." msgid "Copyright 2018 Red Hat, Inc."
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:29 #: src/pages/Organizations/Organizations.jsx:39
#: src/pages/Organizations/Organizations.jsx:25
msgid "Create New Organization"
msgstr ""
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:58
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:146
#: src/pages/Organizations/screens/OrganizationsList.jsx:164
msgid "Created" msgid "Created"
msgstr "" msgstr ""
#: src/index.jsx:200 #: src/index.jsx:165
#: src/pages/CredentialTypes.jsx:17 #: src/pages/CredentialTypes.jsx:19
msgid "Credential Types" msgid "Credential Types"
msgstr "" msgstr ""
#: src/index.jsx:153 #: src/index.jsx:118
#: src/pages/Credentials.jsx:17 #: src/pages/Credentials.jsx:19
msgid "Credentials" msgid "Credentials"
msgstr "" msgstr ""
#: src/index.jsx:122 #: src/index.jsx:87
#: src/pages/Dashboard.jsx:17 #: src/pages/Dashboard.jsx:19
msgid "Dashboard" msgid "Dashboard"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:252 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:109
#: src/components/DataListToolbar/DataListToolbar.jsx:257 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:130
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:153
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:42
msgid "Delete" msgid "Delete"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationBreadcrumb.jsx:59 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:142
msgid "Delete {0}"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:141
msgid "Delete {itemName}"
msgstr ""
#: src/pages/Organizations/components/OrganizationForm.jsx:113
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:138
msgid "Description"
msgstr ""
#: src/pages/Organizations/Organizations.jsx:42
#: src/pages/Organizations/screens/Organization/Organization.jsx:130
msgid "Details"
msgstr ""
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:171
msgid "Edit" msgid "Edit"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:233 #: src/pages/Organizations/Organizations.jsx:41
msgid "Edit Details"
msgstr ""
#: src/components/ExpandCollapse/ExpandCollapse.jsx:44
msgid "Expand" msgid "Expand"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:169 #: src/components/NotificationsList/NotificationListItem.jsx:82
#: src/components/NotificationsList/NotificationListItem.jsx:97
msgid "Failure"
msgstr ""
#: src/components/Pagination/Pagination.jsx:170
msgid "First" msgid "First"
msgstr "" msgstr ""
@@ -114,260 +219,436 @@ msgstr ""
msgid "Help" msgid "Help"
msgstr "" msgstr ""
#: src/index.jsx:215 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:54
#: src/pages/InstanceGroups.jsx:17 msgid "If you {0} want to remove access for this particular user, please remove them from the team."
msgstr ""
#: src/components/PageHeaderToolbar.jsx:66
msgid "Info"
msgstr ""
#: src/index.jsx:180
#: src/pages/InstanceGroups.jsx:19
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:32
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:50
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:159
msgid "Instance Groups" msgid "Instance Groups"
msgstr "" msgstr ""
#: src/index.jsx:220 #: src/index.jsx:185
msgid "Integrations" msgid "Integrations"
msgstr "" msgstr ""
#: src/pages/Login.jsx:80 #: src/pages/Login.jsx:84
msgid "Invalid username or password. Please try again." msgid "Invalid username or password. Please try again."
msgstr "" msgstr ""
#: src/index.jsx:163 #: src/index.jsx:128
#: src/pages/Inventories.jsx:17 #: src/pages/Inventories.jsx:19
msgid "Inventories" msgid "Inventories"
msgstr "" msgstr ""
#: src/index.jsx:168 #: src/index.jsx:133
#: src/pages/InventoryScripts.jsx:17 #: src/pages/InventoryScripts.jsx:19
msgid "Inventory Scripts" msgid "Inventory Scripts"
msgstr "" msgstr ""
#: src/index.jsx:127 #: src/components/Pagination/Pagination.jsx:142
#: src/index.jsx:236 msgid "Items Per Page"
#: src/pages/Jobs.jsx:17 msgstr ""
#: src/components/Pagination/Pagination.jsx:162
msgid "Items {itemMin} {itemMax} of {count}"
msgstr ""
#: src/index.jsx:92
#: src/index.jsx:201
#: src/pages/Jobs.jsx:19
msgid "Jobs" msgid "Jobs"
msgstr "" msgstr ""
#: src/pages/JobsSettings.jsx:17 #: src/pages/JobsSettings.jsx:19
msgid "Jobs Settings" msgid "Jobs Settings"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:221 #: src/components/Pagination/Pagination.jsx:213
msgid "Last" msgid "Last"
msgstr "" msgstr ""
#: src/index.jsx:251 #: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:150
#: src/pages/License.jsx:17 msgid "Last Modified"
msgstr ""
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:173
msgid "Last Name"
msgstr ""
#: src/index.jsx:216
#: src/pages/License.jsx:19
msgid "License" msgid "License"
msgstr "" msgstr ""
#: src/components/PageHeaderToolbar.jsx:116 #: src/components/AddRole/SelectResourceStep.jsx:96
msgid "Loading..."
msgstr ""
#: src/components/PageHeaderToolbar.jsx:120
msgid "Logout" msgid "Logout"
msgstr "" msgstr ""
#: src/index.jsx:210 #: src/index.jsx:175
#: src/pages/ManagementJobs.jsx:17 #: src/pages/ManagementJobs.jsx:19
msgid "Management Jobs" msgid "Management Jobs"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:28 #: src/pages/Organizations/components/OrganizationListItem.jsx:91
msgid "Members"
msgstr ""
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:57
#: src/pages/Organizations/screens/OrganizationsList.jsx:163
msgid "Modified" msgid "Modified"
msgstr "" msgstr ""
#: src/index.jsx:137 #: src/index.jsx:102
#: src/pages/Portal.jsx:17 #: src/pages/Portal.jsx:19
msgid "My View" msgid "My View"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:27 #: src/components/AddRole/AddResourceRole.jsx:140
#: src/components/PaginatedDataList/PaginatedDataList.jsx:116
#: src/pages/Organizations/components/InstanceGroupsLookup.jsx:56
#: src/pages/Organizations/components/OrganizationAccessItem.jsx:109
#: src/pages/Organizations/components/OrganizationForm.jsx:105
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:171
#: src/pages/Organizations/screens/Organization/OrganizationDetail.jsx:134
#: src/pages/Organizations/screens/OrganizationsList.jsx:162
msgid "Name" msgid "Name"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:212 #: src/components/Pagination/Pagination.jsx:204
msgid "Next" msgid "Next"
msgstr "" msgstr ""
#: src/pages/NotificationTemplates.jsx:17 #: src/components/PaginatedDataList/PaginatedDataList.jsx:133
#: src/components/PaginatedDataList/PaginatedDataList.jsx:196
msgid "No {0} Found"
msgstr ""
#: src/pages/NotificationTemplates.jsx:19
msgid "Notification Templates" msgid "Notification Templates"
msgstr "" msgstr ""
#: src/index.jsx:205 #: src/index.jsx:170
#: src/pages/Organizations/Organizations.jsx:45
#: src/pages/Organizations/screens/Organization/Organization.jsx:137
msgid "Notifications" msgid "Notifications"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organization.add.jsx:79 #: src/pages/Organizations/views/Organization.add.jsx:79
msgid "Organization Add" #~ msgid "Organization Add"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:72 #: src/pages/Organizations/screens/Organization/Organization.jsx:153
msgid "Organization detail tabs" msgid "Organization detail tabs"
msgstr "" msgstr ""
#: src/index.jsx:179 #: src/index.jsx:144
#: src/pages/Organizations/views/Organization.view.jsx:63 #: src/pages/Organizations/Organizations.jsx:38
#: src/pages/Organizations/views/Organizations.list.jsx:196 #: src/pages/Organizations/Organizations.jsx:24
#: src/pages/Organizations/views/Organizations.list.jsx:202
msgid "Organizations" msgid "Organizations"
msgstr "" msgstr ""
#: src/pages/Organizations/views/Organizations.list.jsx:218 #: src/pages/Organizations/views/Organizations.list.jsx:218
msgid "Organizations List" #~ msgid "Organizations List"
#~ msgstr ""
#: src/components/Pagination/Pagination.jsx:190
msgid "Page"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:189 #: src/components/Pagination/Pagination.jsx:189
msgid "Page <0/> of {pageCount}" #~ msgid "Page <0/> of {pageCount}"
msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:192 #: src/components/Pagination/Pagination.jsx:193
msgid "Page Number" msgid "Page Number"
msgstr "" msgstr ""
#: src/pages/Login.jsx:79 #: src/pages/Login.jsx:82
msgid "Password" msgid "Password"
msgstr "" msgstr ""
#: src/components/Pagination/Pagination.jsx:158 #: src/components/Pagination/Pagination.jsx:158
msgid "Per Page" #~ msgid "Per Page"
#~ msgstr ""
#: src/components/PaginatedDataList/PaginatedDataList.jsx:136
#: src/components/PaginatedDataList/PaginatedDataList.jsx:199
msgid "Please add {0} {itemName} to populate this list"
msgstr "" msgstr ""
#: src/App.jsx:203 #: src/App.jsx:203
#~ msgid "Portal Mode" #~ msgid "Portal Mode"
#~ msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:178 #: src/components/Pagination/Pagination.jsx:179
msgid "Previous" msgid "Previous"
msgstr "" msgstr ""
#: src/index.jsx:115 #: src/index.jsx:80
msgid "Primary Navigation" msgid "Primary Navigation"
msgstr "" msgstr ""
#: src/index.jsx:158 #: src/index.jsx:123
#: src/pages/Projects.jsx:17 #: src/pages/Projects.jsx:19
msgid "Projects" msgid "Projects"
msgstr "" msgstr ""
#: src/index.jsx:144 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
msgid "Remove {0} Access"
msgstr ""
#: src/index.jsx:109
msgid "Resources" msgid "Resources"
msgstr "" msgstr ""
#: src/index.jsx:132 #: src/components/AddRole/AddResourceRole.jsx:220
#: src/pages/Schedules.jsx:17 #: src/components/FormActionGroup/FormActionGroup.jsx:27
#: src/components/FormActionGroup/FormActionGroup.jsx:27
#: src/components/Lookup/Lookup.jsx:168
msgid "Save"
msgstr ""
#: src/index.jsx:97
#: src/pages/Schedules.jsx:19
msgid "Schedules" msgid "Schedules"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:191 #: src/components/Search/Search.jsx:138
msgid "Search" msgid "Search"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:185 #: src/components/Search/Search.jsx:131
msgid "Search text input" msgid "Search text input"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:160 #: src/components/AnsibleSelect/AnsibleSelect.jsx:28
msgid "Select Input"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:159
msgid "Select Users Or Teams"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:111
msgid "Select a row to delete"
msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:102
msgid "Select all" msgid "Select all"
msgstr "" msgstr ""
#: src/index.jsx:227 #: src/components/AddRole/AddResourceRole.jsx:178
msgid "Select items from list"
msgstr ""
#: src/pages/Organizations/components/OrganizationForm.jsx:141
msgid "Select the Instance Groups for this Organization to run on."
msgstr ""
#: src/components/Lookup/Lookup.jsx:164
msgid "Select {header}"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:187
#: src/components/AddRole/AddResourceRole.jsx:198
#: src/components/AddRole/AddResourceRole.jsx:215
#: src/components/AddRole/SelectRoleStep.jsx:29
#: src/components/Lookup/Lookup.jsx:193
msgid "Selected"
msgstr ""
#: src/index.jsx:192
msgid "Settings" msgid "Settings"
msgstr "" msgstr ""
#: src/components/DataListToolbar/DataListToolbar.jsx:222 #: src/components/Sort/Sort.jsx:135
msgid "Sort" msgid "Sort"
msgstr "" msgstr ""
#: src/index.jsx:241 #: src/components/NotificationsList/NotificationListItem.jsx:70
msgid "Successful"
msgstr ""
#: src/index.jsx:206
msgid "System" msgid "System"
msgstr "" msgstr ""
#: src/pages/SystemSettings.jsx:17 #: src/pages/SystemSettings.jsx:19
msgid "System Settings" msgid "System Settings"
msgstr "" msgstr ""
#: src/index.jsx:189 #: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
#: src/pages/Organizations/components/OrganizationListItem.jsx:57 msgid "Team"
#: src/pages/Teams.jsx:17 msgstr ""
#: src/pages/Organizations/components/OrganizationAccessItem.jsx:144
msgid "Team Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:169
#: src/index.jsx:154
#: src/pages/Organizations/Organizations.jsx:44
#: src/pages/Organizations/components/OrganizationListItem.jsx:99
#: src/pages/Organizations/screens/Organization/Organization.jsx:132
#: src/pages/Teams.jsx:19
msgid "Teams" msgid "Teams"
msgstr "" msgstr ""
#: src/index.jsx:148 #: src/index.jsx:113
#: src/pages/Templates.jsx:17 #: src/pages/Templates.jsx:19
msgid "Templates" msgid "Templates"
msgstr "" msgstr ""
#: src/util/validators.jsx:6
msgid "This field must not be blank"
msgstr ""
#: src/util/validators.jsx:16
msgid "This field must not exceed {max} characters"
msgstr ""
#: src/components/NotificationsList/NotificationListItem.jsx:90
#: src/components/NotificationsList/NotificationListItem.jsx:105
msgid "Toggle notification failure"
msgstr ""
#: src/components/NotificationsList/NotificationListItem.jsx:78
msgid "Toggle notification success"
msgstr ""
#: src/components/TowerLogo/TowerLogo.jsx:48 #: src/components/TowerLogo/TowerLogo.jsx:48
msgid "Tower Brand Image" msgid "Tower Brand Image"
msgstr "" msgstr ""
#: src/components/PageHeaderToolbar.jsx:108 #: src/components/AnsibleSelect/AnsibleSelect.jsx:35
msgid "Use Default {label}"
msgstr ""
#: src/pages/Organizations/components/DeleteRoleConfirmationModal.jsx:28
msgid "User"
msgstr ""
#: src/components/PageHeaderToolbar.jsx:113
msgid "User Details" msgid "User Details"
msgstr "" msgstr ""
#: src/index.jsx:246 #: src/index.jsx:211
msgid "User Interface" msgid "User Interface"
msgstr "" msgstr ""
#: src/pages/UISettings.jsx:17 #: src/pages/UISettings.jsx:19
msgid "User Interface Settings" msgid "User Interface Settings"
msgstr "" msgstr ""
#: src/pages/Login.jsx:78 #: src/pages/Organizations/components/OrganizationAccessItem.jsx:122
msgid "User Roles"
msgstr ""
#: src/components/AddRole/AddResourceRole.jsx:136
#: src/pages/Login.jsx:81
#: src/pages/Organizations/screens/Organization/OrganizationAccess.jsx:172
msgid "Username" msgid "Username"
msgstr "" msgstr ""
#: src/index.jsx:184 #: src/components/AddRole/AddResourceRole.jsx:164
#: src/pages/Organizations/components/OrganizationListItem.jsx:49 #: src/index.jsx:149
#: src/pages/Users.jsx:17 #: src/pages/Users.jsx:19
msgid "Users" msgid "Users"
msgstr "" msgstr ""
#: src/index.jsx:118 #: src/index.jsx:83
msgid "Views" msgid "Views"
msgstr "" msgstr ""
#: src/pages/Login.jsx:74 #: src/pages/Login.jsx:76
msgid "Welcome to Ansible Tower! Please Sign In." msgid "Welcome to Ansible Tower! Please Sign In."
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:56 #: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:104
msgid "add {currentTab}" msgid "You do not have permission to delete the following {0}: {itemsUnableToDelete}"
msgstr "" msgstr ""
#: src/contexts/Network.jsx:44
msgid "You have been logged out."
msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:56
#~ msgid "add {currentTab}"
#~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:45 #: src/pages/Organizations/components/OrganizationDetail.jsx:45
msgid "adding {currentTab}" #~ msgid "adding {currentTab}"
#~ msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:158
msgid "cancel delete"
msgstr ""
#: src/components/PaginatedDataList/ToolbarDeleteButton.jsx:150
msgid "confirm delete"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:38 #: src/pages/Organizations/components/OrganizationDetail.jsx:38
msgid "confirm removal of {currentTab}/cancel and go back to {currentTab} view." #~ msgid "confirm removal of {currentTab}/cancel and go back to {currentTab} view."
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:60 #: src/pages/Organizations/components/OrganizationDetail.jsx:60
msgid "delete {currentTab}" #~ msgid "delete {currentTab}"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:36 #: src/pages/Organizations/components/OrganizationDetail.jsx:36
msgid "deleting {currentTab} association with orgs" #~ msgid "deleting {currentTab} association with orgs"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationEdit.jsx:20 #: src/pages/Organizations/components/OrganizationEdit.jsx:20
msgid "edit view" #~ msgid "edit view"
#~ msgstr ""
#: src/components/Lookup/Lookup.jsx:135
msgid "items"
msgstr ""
#: src/components/Pagination/Pagination.jsx:198
msgid "of {pageCount}"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationEdit.jsx:22 #: src/pages/Organizations/components/OrganizationEdit.jsx:22
msgid "save/cancel and go back to view" #~ msgid "save/cancel and go back to view"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:47 #: src/pages/Organizations/components/OrganizationDetail.jsx:47
msgid "save/cancel and go back to {currentTab} view" #~ msgid "save/cancel and go back to {currentTab} view"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationListItem.jsx:29 #: src/pages/Organizations/components/OrganizationListItem.jsx:29
msgid "select organization {itemId}" #~ msgid "select organization {itemId}"
msgstr "" #~ msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:82 #: src/pages/Organizations/components/OrganizationDetail.jsx:82
msgid "{0}" #~ msgid "{0}"
#~ msgstr ""
#: src/components/PaginatedDataList/PaginatedDataList.jsx:153
#: src/components/PaginatedDataList/PaginatedDataList.jsx:215
msgid "{0} List"
msgstr "" msgstr ""
#: src/pages/Organizations/components/OrganizationDetail.jsx:54 #: src/pages/Organizations/components/OrganizationDetail.jsx:54
msgid "{currentTab} detail view" #~ msgid "{currentTab} detail view"
msgstr "" #~ msgstr ""
#: src/components/Pagination/Pagination.jsx:163 #: src/components/Pagination/Pagination.jsx:163
msgid "{itemMin} - {itemMax} of {count}" #~ msgid "{itemMin} - {itemMax} of {count}"
msgstr "" #~ msgstr ""

View File

@@ -9,7 +9,7 @@ import {
Button Button
} from '@patternfly/react-core'; } from '@patternfly/react-core';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { RootDialog } from './contexts/RootDialog'; import { RootDialog } from './contexts/RootDialog';
@@ -66,92 +66,88 @@ class App extends Component {
render () { render () {
const { isAboutModalOpen, isNavOpen } = this.state; const { isAboutModalOpen, isNavOpen } = this.state;
const { render, routeGroups = [], navLabel = '' } = this.props; const { render, routeGroups = [], navLabel = '', i18n } = this.props;
return ( return (
<Config> <Config>
{({ ansible_version, version, me }) => ( {({ ansible_version, version, me }) => (
<I18n> <RootDialog>
{({ i18n }) => ( {({
<RootDialog> title,
{({ bodyText,
title, variant = 'info',
bodyText, clearRootDialogMessage
variant = 'info', }) => (
clearRootDialogMessage <Fragment>
}) => ( {(title || bodyText) && (
<Fragment> <AlertModal
{(title || bodyText) && ( variant={variant}
<AlertModal isOpen={!!(title || bodyText)}
variant={variant} onClose={clearRootDialogMessage}
isOpen={!!(title || bodyText)} title={title}
onClose={clearRootDialogMessage} actions={[
title={title} <Button
actions={[ key="close"
<Button variant="secondary"
key="close" onClick={clearRootDialogMessage}
variant="secondary"
onClick={clearRootDialogMessage}
>
{i18n._(t`Close`)}
</Button>
]}
> >
{bodyText} {i18n._(t`Close`)}
</AlertModal> </Button>
)} ]}
<Page >
usecondensed="True" {bodyText}
header={( </AlertModal>
<PageHeader
showNavToggle
onNavToggle={this.onNavToggle}
logo={<TowerLogo linkTo="/" />}
toolbar={(
<PageHeaderToolbar
loggedInUser={me}
isAboutDisabled={!version}
onAboutClick={this.onAboutModalOpen}
onLogoutClick={this.onLogout}
/>
)}
/>
)}
sidebar={(
<PageSidebar
isNavOpen={isNavOpen}
nav={(
<Nav aria-label={navLabel}>
<NavList>
{routeGroups.map(
({ groupId, groupTitle, routes }) => (
<NavExpandableGroup
key={groupId}
groupId={groupId}
groupTitle={groupTitle}
routes={routes}
/>
)
)}
</NavList>
</Nav>
)}
/>
)}
>
{render && render({ routeGroups })}
</Page>
<About
ansible_version={ansible_version}
version={version}
isOpen={isAboutModalOpen}
onClose={this.onAboutModalClose}
/>
</Fragment>
)} )}
</RootDialog> <Page
usecondensed="True"
header={(
<PageHeader
showNavToggle
onNavToggle={this.onNavToggle}
logo={<TowerLogo linkTo="/" />}
toolbar={(
<PageHeaderToolbar
loggedInUser={me}
isAboutDisabled={!version}
onAboutClick={this.onAboutModalOpen}
onLogoutClick={this.onLogout}
/>
)}
/>
)}
sidebar={(
<PageSidebar
isNavOpen={isNavOpen}
nav={(
<Nav aria-label={navLabel}>
<NavList>
{routeGroups.map(
({ groupId, groupTitle, routes }) => (
<NavExpandableGroup
key={groupId}
groupId={groupId}
groupTitle={groupTitle}
routes={routes}
/>
)
)}
</NavList>
</Nav>
)}
/>
)}
>
{render && render({ routeGroups })}
</Page>
<About
ansible_version={ansible_version}
version={version}
isOpen={isAboutModalOpen}
onClose={this.onAboutModalClose}
/>
</Fragment>
)} )}
</I18n> </RootDialog>
)} )}
</Config> </Config>
); );
@@ -159,4 +155,4 @@ class App extends Component {
} }
export { App as _App }; export { App as _App };
export default withNetwork(App); export default withI18n()(withNetwork(App));

View File

@@ -1,7 +1,7 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { Trans, t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
AboutModal, AboutModal,
TextContent, TextContent,
@@ -41,46 +41,43 @@ class About extends React.Component {
ansible_version, ansible_version,
version, version,
isOpen, isOpen,
onClose onClose,
i18n
} = this.props; } = this.props;
const speechBubble = this.createSpeechBubble(version); const speechBubble = this.createSpeechBubble(version);
return ( return (
<I18n> <AboutModal
{({ i18n }) => ( isOpen={isOpen}
<AboutModal onClose={onClose}
isOpen={isOpen} productName="Ansible Tower"
onClose={onClose} trademark={i18n._(t`Copyright 2018 Red Hat, Inc.`)}
productName="Ansible Tower" brandImageSrc={brandImg}
trademark={i18n._(t`Copyright 2018 Red Hat, Inc.`)} brandImageAlt={i18n._(t`Brand Image`)}
brandImageSrc={brandImg} logoImageSrc={logoImg}
brandImageAlt={i18n._(t`Brand Image`)} logoImageAlt={i18n._(t`AboutModal Logo`)}
logoImageSrc={logoImg} >
logoImageAlt={i18n._(t`AboutModal Logo`)} <pre>
> { speechBubble }
<pre> {`
{ speechBubble } \\
{` \\ ^__^
\\ (oo)\\_______
\\ ^__^ (__) A )\\
(oo)\\_______ ||----w |
(__) A )\\ || ||
||----w | `}
|| || </pre>
`} <TextContent>
</pre> <TextList component="dl">
<TextContent> <TextListItem component="dt">
<TextList component="dl"> {i18n._(t`Ansible Version`)}
<TextListItem component="dt"> </TextListItem>
<Trans>Ansible Version</Trans> <TextListItem component="dd">{ ansible_version }</TextListItem>
</TextListItem> </TextList>
<TextListItem component="dd">{ ansible_version }</TextListItem> </TextContent>
</TextList> </AboutModal>
</TextContent>
</AboutModal>
)}
</I18n>
); );
} }
} }
@@ -98,4 +95,4 @@ About.defaultProps = {
version: null, version: null,
}; };
export default About; export default withI18n()(About);

View File

@@ -1,8 +1,9 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n, i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { Wizard } from '@patternfly/react-core'; import { Wizard } from '@patternfly/react-core';
import { withNetwork } from '../../contexts/Network';
import SelectResourceStep from './SelectResourceStep'; import SelectResourceStep from './SelectResourceStep';
import SelectRoleStep from './SelectRoleStep'; import SelectRoleStep from './SelectRoleStep';
import SelectableCard from './SelectableCard'; import SelectableCard from './SelectableCard';
@@ -127,113 +128,103 @@ class AddResourceRole extends React.Component {
} = this.state; } = this.state;
const { const {
onClose, onClose,
roles roles,
i18n
} = this.props; } = this.props;
const userColumns = [ const userColumns = [
{ name: i18nMark('Username'), key: 'username', isSortable: true } { name: i18n._(t`Username`), key: 'username', isSortable: true }
]; ];
const teamColumns = [ const teamColumns = [
{ name: i18nMark('Name'), key: 'name', isSortable: true } { name: i18n._(t`Name`), key: 'name', isSortable: true }
]; ];
let wizardTitle = ''; let wizardTitle = '';
switch (selectedResource) { switch (selectedResource) {
case 'users': case 'users':
wizardTitle = i18nMark('Add User Roles'); wizardTitle = i18n._(t`Add User Roles`);
break; break;
case 'teams': case 'teams':
wizardTitle = i18nMark('Add Team Roles'); wizardTitle = i18n._(t`Add Team Roles`);
break; break;
default: default:
wizardTitle = i18nMark('Add Roles'); wizardTitle = i18n._(t`Add Roles`);
} }
const steps = [ const steps = [
{ {
id: 1, id: 1,
name: i18nMark('Select Users Or Teams'), name: i18n._(t`Select Users Or Teams`),
component: ( component: (
<I18n> <div style={{ display: 'flex' }}>
{({ i18n }) => ( <SelectableCard
<div style={{ display: 'flex' }}> isSelected={selectedResource === 'users'}
<SelectableCard label={i18n._(t`Users`)}
isSelected={selectedResource === 'users'} onClick={() => this.handleResourceSelect('users')}
label={i18n._(t`Users`)} />
onClick={() => this.handleResourceSelect('users')} <SelectableCard
/> isSelected={selectedResource === 'teams'}
<SelectableCard label={i18n._(t`Teams`)}
isSelected={selectedResource === 'teams'} onClick={() => this.handleResourceSelect('teams')}
label={i18n._(t`Teams`)} />
onClick={() => this.handleResourceSelect('teams')} </div>
/>
</div>
)}
</I18n>
), ),
enableNext: selectedResource !== null enableNext: selectedResource !== null
}, },
{ {
id: 2, id: 2,
name: i18nMark('Select items from list'), name: i18n._(t`Select items from list`),
component: ( component: (
<I18n> <Fragment>
{({ i18n }) => ( {selectedResource === 'users' && (
<Fragment> <SelectResourceStep
{selectedResource === 'users' && ( columns={userColumns}
<SelectResourceStep displayKey="username"
columns={userColumns} onRowClick={this.handleResourceCheckboxClick}
displayKey="username" onSearch={this.readUsers}
onRowClick={this.handleResourceCheckboxClick} selectedLabel={i18n._(t`Selected`)}
onSearch={this.readUsers} selectedResourceRows={selectedResourceRows}
selectedLabel={i18n._(t`Selected`)} sortedColumnKey="username"
selectedResourceRows={selectedResourceRows} itemName="user"
sortedColumnKey="username" />
itemName="user"
/>
)}
{selectedResource === 'teams' && (
<SelectResourceStep
columns={teamColumns}
onRowClick={this.handleResourceCheckboxClick}
onSearch={this.readTeams}
selectedLabel={i18n._(t`Selected`)}
selectedResourceRows={selectedResourceRows}
itemName="team"
/>
)}
</Fragment>
)} )}
</I18n> {selectedResource === 'teams' && (
<SelectResourceStep
columns={teamColumns}
onRowClick={this.handleResourceCheckboxClick}
onSearch={this.readTeams}
selectedLabel={i18n._(t`Selected`)}
selectedResourceRows={selectedResourceRows}
itemName="team"
/>
)}
</Fragment>
), ),
enableNext: selectedResourceRows.length > 0 enableNext: selectedResourceRows.length > 0
}, },
{ {
id: 3, id: 3,
name: i18nMark('Apply roles'), name: i18n._(t`Apply roles`),
component: ( component: (
<I18n> <SelectRoleStep
{({ i18n }) => ( onRolesClick={this.handleRoleCheckboxClick}
<SelectRoleStep roles={roles}
onRolesClick={this.handleRoleCheckboxClick} selectedListKey={selectedResource === 'users' ? 'username' : 'name'}
roles={roles} selectedListLabel={i18n._(t`Selected`)}
selectedListKey={selectedResource === 'users' ? 'username' : 'name'} selectedResourceRows={selectedResourceRows}
selectedListLabel={i18n._(t`Selected`)} selectedRoleRows={selectedRoleRows}
selectedResourceRows={selectedResourceRows} />
selectedRoleRows={selectedRoleRows}
/>
)}
</I18n>
), ),
nextButtonText: i18nMark('Save'), nextButtonText: i18n._(t`Save`),
enableNext: selectedRoleRows.length > 0 enableNext: selectedRoleRows.length > 0
} }
]; ];
const currentStep = steps.find(step => step.id === currentStepId); const currentStep = steps.find(step => step.id === currentStepId);
// TODO: somehow internationalize steps and currentStep.nextButtonText
return ( return (
<Wizard <Wizard
style={{ overflow: 'scroll' }} style={{ overflow: 'scroll' }}
@@ -259,4 +250,5 @@ AddResourceRole.defaultProps = {
roles: {} roles: {}
}; };
export default AddResourceRole; export { AddResourceRole as _AddResourceRole };
export default withI18n()(withNetwork(AddResourceRole));

View File

@@ -1,7 +1,8 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import PaginatedDataList from '../PaginatedDataList'; import PaginatedDataList from '../PaginatedDataList';
import CheckboxListItem from '../ListItem'; import CheckboxListItem from '../ListItem';
import SelectedList from '../SelectedList'; import SelectedList from '../SelectedList';
@@ -87,11 +88,12 @@ class SelectResourceStep extends React.Component {
selectedLabel, selectedLabel,
selectedResourceRows, selectedResourceRows,
itemName, itemName,
i18n
} = this.props; } = this.props;
return ( return (
<Fragment> <Fragment>
{isLoading && (<div>Loading...</div>)} {isLoading && (<div>{i18n._(t`Loading...`)}</div>)}
{isInitialized && ( {isInitialized && (
<Fragment> <Fragment>
{selectedResourceRows.length > 0 && ( {selectedResourceRows.length > 0 && (
@@ -146,11 +148,11 @@ SelectResourceStep.propTypes = {
SelectResourceStep.defaultProps = { SelectResourceStep.defaultProps = {
displayKey: 'name', displayKey: 'name',
onRowClick: () => {}, onRowClick: () => {},
selectedLabel: i18nMark('Selected Items'), selectedLabel: null,
selectedResourceRows: [], selectedResourceRows: [],
sortedColumnKey: 'name', sortedColumnKey: 'name',
itemName: 'item', itemName: 'item',
}; };
export { SelectResourceStep as _SelectResourceStep }; export { SelectResourceStep as _SelectResourceStep };
export default withRouter(SelectResourceStep); export default withI18n()(withRouter(SelectResourceStep));

View File

@@ -1,7 +1,8 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import CheckboxCard from './CheckboxCard'; import CheckboxCard from './CheckboxCard';
import SelectedList from '../SelectedList'; import SelectedList from '../SelectedList';
@@ -14,7 +15,8 @@ class RolesStep extends React.Component {
selectedListKey, selectedListKey,
selectedListLabel, selectedListLabel,
selectedResourceRows, selectedResourceRows,
selectedRoleRows selectedRoleRows,
i18n
} = this.props; } = this.props;
return ( return (
@@ -24,7 +26,7 @@ class RolesStep extends React.Component {
<SelectedList <SelectedList
displayKey={selectedListKey} displayKey={selectedListKey}
isReadOnly isReadOnly
label={selectedListLabel} label={selectedListLabel || i18n._(t`Selected`)}
selected={selectedResourceRows} selected={selectedResourceRows}
showOverflowAfter={5} showOverflowAfter={5}
/> />
@@ -61,9 +63,9 @@ RolesStep.propTypes = {
RolesStep.defaultProps = { RolesStep.defaultProps = {
onRolesClick: () => {}, onRolesClick: () => {},
selectedListKey: 'name', selectedListKey: 'name',
selectedListLabel: i18nMark('Selected'), selectedListLabel: null,
selectedResourceRows: [], selectedResourceRows: [],
selectedRoleRows: [] selectedRoleRows: []
}; };
export default RolesStep; export default withI18n()(RolesStep);

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
FormSelect, FormSelect,
@@ -20,29 +20,25 @@ class AnsibleSelect extends React.Component {
} }
render () { render () {
const { label, value, data, defaultSelected } = this.props; const { label, value, data, defaultSelected, i18n } = this.props;
return ( return (
<I18n> <FormSelect
{({ i18n }) => ( value={value}
<FormSelect onChange={this.onSelectChange}
value={value} aria-label={i18n._(t`Select Input`)}
onChange={this.onSelectChange} >
aria-label={i18n._(t`Select Input`)} {data.map((datum) => (
> datum === defaultSelected ? (
{data.map((datum) => ( <FormSelectOption
datum === defaultSelected ? ( key=""
<FormSelectOption value=""
key="" label={i18n._(t`Use Default ${label}`)}
value="" />
label={i18n._(t`Use Default ${label}`)} ) : (
/> <FormSelectOption key={datum} value={datum} label={datum} />
) : ( )
<FormSelectOption key={datum} value={datum} label={datum} /> ))}
) </FormSelect>
))}
</FormSelect>
)}
</I18n>
); );
} }
} }
@@ -62,4 +58,5 @@ AnsibleSelect.propTypes = {
value: PropTypes.string.isRequired, value: PropTypes.string.isRequired,
}; };
export default AnsibleSelect; export { AnsibleSelect as _AnsibleSelect };
export default withI18n()(AnsibleSelect);

View File

@@ -3,40 +3,32 @@ import { string } from 'prop-types';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import { Button } from '@patternfly/react-core'; import { Button } from '@patternfly/react-core';
import { TimesIcon } from '@patternfly/react-icons'; import { TimesIcon } from '@patternfly/react-icons';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
function CardCloseButton ({ linkTo, ...props }) { function CardCloseButton ({ linkTo, i18n, i18nHash, ...props }) {
if (linkTo) { if (linkTo) {
return ( return (
<I18n> <Link
{({ i18n }) => ( className="pf-c-button pf-c-card__close"
<Link aria-label={i18n._(t`Close`)}
className="pf-c-button pf-c-card__close" title={i18n._(t`Close`)}
aria-label={i18n._(t`Close`)} to={linkTo}
title={i18n._(t`Close`)} {...props}
to={linkTo} >
{...props} <TimesIcon />
> </Link>
<TimesIcon />
</Link>
)}
</I18n>
); );
} }
return ( return (
<I18n> <Button
{({ i18n }) => ( variant="plain"
<Button className="pf-c-card__close"
variant="plain" aria-label={i18n._(t`Close`)}
className="pf-c-card__close" {...props}
aria-label={i18n._(t`Close`)} >
{...props} <TimesIcon />
> </Button>
<TimesIcon />
</Button>
)}
</I18n>
); );
} }
CardCloseButton.propTypes = { CardCloseButton.propTypes = {
@@ -46,4 +38,4 @@ CardCloseButton.defaultProps = {
linkTo: null, linkTo: null,
}; };
export default CardCloseButton; export default withI18n()(CardCloseButton);

View File

@@ -1,6 +1,6 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Checkbox, Checkbox,
@@ -85,70 +85,66 @@ class DataListToolbar extends React.Component {
sortOrder, sortOrder,
sortedColumnKey, sortedColumnKey,
additionalControls, additionalControls,
i18n
} = this.props; } = this.props;
const showExpandCollapse = (onCompact && onExpand); const showExpandCollapse = (onCompact && onExpand);
return ( return (
<I18n> <AWXToolbar>
{({ i18n }) => ( <Toolbar marginleft={noLeftMargin ? 1 : 0}>
<AWXToolbar> <ColumnLeft>
<Toolbar marginleft={noLeftMargin ? 1 : 0}> { showSelectAll && (
<ColumnLeft> <Fragment>
{ showSelectAll && ( <ToolbarItem>
<Fragment> <Checkbox
<ToolbarItem> checked={isAllSelected}
<Checkbox onChange={onSelectAll}
checked={isAllSelected} aria-label={i18n._(t`Select all`)}
onChange={onSelectAll} id="select-all"
aria-label={i18n._(t`Select all`)}
id="select-all"
/>
</ToolbarItem>
<VerticalSeparator />
</Fragment>
)}
<ToolbarItem css="flex-grow: 1;">
<Search
columns={columns}
onSearch={onSearch}
sortedColumnKey={sortedColumnKey}
/> />
</ToolbarItem> </ToolbarItem>
<VerticalSeparator /> <VerticalSeparator />
</ColumnLeft> </Fragment>
<ColumnRight> )}
<ToolbarItem> <ToolbarItem css="flex-grow: 1;">
<Sort <Search
columns={columns} columns={columns}
onSort={onSort} onSearch={onSearch}
sortOrder={sortOrder} sortedColumnKey={sortedColumnKey}
sortedColumnKey={sortedColumnKey} />
</ToolbarItem>
<VerticalSeparator />
</ColumnLeft>
<ColumnRight>
<ToolbarItem>
<Sort
columns={columns}
onSort={onSort}
sortOrder={sortOrder}
sortedColumnKey={sortedColumnKey}
/>
</ToolbarItem>
{showExpandCollapse && (
<Fragment>
<VerticalSeparator />
<ToolbarGroup>
<ExpandCollapse
isCompact={isCompact}
onCompact={onCompact}
onExpand={onExpand}
/> />
</ToolbarItem> </ToolbarGroup>
{showExpandCollapse && ( { additionalControls && (
<Fragment> <VerticalSeparator />
<VerticalSeparator />
<ToolbarGroup>
<ExpandCollapse
isCompact={isCompact}
onCompact={onCompact}
onExpand={onExpand}
/>
</ToolbarGroup>
{ additionalControls && (
<VerticalSeparator />
)}
</Fragment>
)} )}
<AdditionalControlsWrapper> </Fragment>
{additionalControls} )}
</AdditionalControlsWrapper> <AdditionalControlsWrapper>
</ColumnRight> {additionalControls}
</Toolbar> </AdditionalControlsWrapper>
</AWXToolbar> </ColumnRight>
</Toolbar>
)} </AWXToolbar>
</I18n>
); );
} }
} }
@@ -184,4 +180,4 @@ DataListToolbar.defaultProps = {
additionalControls: [], additionalControls: [],
}; };
export default DataListToolbar; export default withI18n()(DataListToolbar);

View File

@@ -1,6 +1,6 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Button, Button,
@@ -22,36 +22,33 @@ class ExpandCollapse extends React.Component {
const { const {
onCompact, onCompact,
onExpand, onExpand,
isCompact isCompact,
i18n
} = this.props; } = this.props;
return ( return (
<I18n> <Fragment>
{({ i18n }) => ( <ToolbarItem>
<Fragment> <Button
<ToolbarItem> variant="plain"
<Button aria-label={i18n._(t`Collapse`)}
variant="plain" onClick={onCompact}
aria-label={i18n._(t`Collapse`)} style={isCompact ? ToolbarActiveStyle : null}
onClick={onCompact} >
style={isCompact ? ToolbarActiveStyle : null} <BarsIcon />
> </Button>
<BarsIcon /> </ToolbarItem>
</Button> <ToolbarItem>
</ToolbarItem> <Button
<ToolbarItem> variant="plain"
<Button aria-label={i18n._(t`Expand`)}
variant="plain" onClick={onExpand}
aria-label={i18n._(t`Expand`)} style={!isCompact ? ToolbarActiveStyle : null}
onClick={onExpand} >
style={!isCompact ? ToolbarActiveStyle : null} <EqualsIcon />
> </Button>
<EqualsIcon /> </ToolbarItem>
</Button> </Fragment>
</ToolbarItem>
</Fragment>
)}
</I18n>
); );
} }
} }
@@ -64,4 +61,4 @@ ExpandCollapse.propTypes = {
ExpandCollapse.defaultProps = {}; ExpandCollapse.defaultProps = {};
export default ExpandCollapse; export default withI18n()(ExpandCollapse);

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
ActionGroup, ActionGroup,
@@ -20,21 +20,17 @@ const buttonGroupStyle = {
marginRight: '20px' marginRight: '20px'
}; };
const FormActionGroup = ({ onSubmit, submitDisabled, onCancel }) => ( const FormActionGroup = ({ onSubmit, submitDisabled, onCancel, i18n }) => (
<I18n> <ActionGroup style={formActionGroupStyle}>
{({ i18n }) => ( <Toolbar>
<ActionGroup style={formActionGroupStyle}> <ToolbarGroup style={buttonGroupStyle}>
<Toolbar> <Button aria-label={i18n._(t`Save`)} variant="primary" type="submit" onClick={onSubmit} isDisabled={submitDisabled}>{i18n._(t`Save`)}</Button>
<ToolbarGroup style={buttonGroupStyle}> </ToolbarGroup>
<Button aria-label={i18n._(t`Save`)} variant="primary" type="submit" onClick={onSubmit} isDisabled={submitDisabled}>{i18n._(t`Save`)}</Button> <ToolbarGroup>
</ToolbarGroup> <Button aria-label={i18n._(t`Cancel`)} variant="secondary" type="button" onClick={onCancel}>{i18n._(t`Cancel`)}</Button>
<ToolbarGroup> </ToolbarGroup>
<Button aria-label={i18n._(t`Cancel`)} variant="secondary" type="button" onClick={onCancel}>{i18n._(t`Cancel`)}</Button> </Toolbar>
</ToolbarGroup> </ActionGroup>
</Toolbar>
</ActionGroup>
)}
</I18n>
); );
FormActionGroup.propTypes = { FormActionGroup.propTypes = {
@@ -47,4 +43,4 @@ FormActionGroup.defaultProps = {
submitDisabled: false, submitDisabled: false,
}; };
export default FormActionGroup; export default withI18n()(FormActionGroup);

View File

@@ -9,7 +9,7 @@ import {
InputGroup, InputGroup,
Modal, Modal,
} from '@patternfly/react-core'; } from '@patternfly/react-core';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { withNetwork } from '../../contexts/Network'; import { withNetwork } from '../../contexts/Network';
@@ -130,7 +130,9 @@ class Lookup extends React.Component {
results, results,
count, count,
} = this.state; } = this.state;
const { id, lookupHeader = 'items', value, columns } = this.props; const { id, lookupHeader, value, columns, i18n } = this.props;
const header = lookupHeader || i18n._(t`items`);
const chips = value ? ( const chips = value ? (
<div className="pf-c-chip-group"> <div className="pf-c-chip-group">
@@ -143,64 +145,60 @@ class Lookup extends React.Component {
) : null; ) : null;
return ( return (
<I18n> <Fragment>
{({ i18n }) => ( <InputGroup className="awx-lookup">
<Fragment> <Button
<InputGroup className="awx-lookup"> aria-label="Search"
<Button id={id}
aria-label="Search" onClick={this.handleModalToggle}
id={id} variant={ButtonVariant.tertiary}
onClick={this.handleModalToggle} >
variant={ButtonVariant.tertiary} <SearchIcon />
> </Button>
<SearchIcon /> <div className="pf-c-form-control">
</Button> {chips}
<div className="pf-c-form-control"> </div>
{chips} </InputGroup>
</div> <Modal
</InputGroup> className="awx-c-modal"
<Modal title={i18n._(t`Select ${header}`)}
className="awx-c-modal" isOpen={isModalOpen}
title={`Select ${lookupHeader}`} onClose={this.handleModalToggle}
isOpen={isModalOpen} actions={[
onClose={this.handleModalToggle} <Button key="save" variant="primary" onClick={this.saveModal} style={(results.length === 0) ? { display: 'none' } : {}}>{i18n._(t`Save`)}</Button>,
actions={[ <Button key="cancel" variant="secondary" onClick={this.handleModalToggle}>{(results.length === 0) ? i18n._(t`Close`) : i18n._(t`Cancel`)}</Button>
<Button key="save" variant="primary" onClick={this.saveModal} style={(results.length === 0) ? { display: 'none' } : {}}>{i18n._(t`Save`)}</Button>, ]}
<Button key="cancel" variant="secondary" onClick={this.handleModalToggle}>{(results.length === 0) ? i18n._(t`Close`) : i18n._(t`Cancel`)}</Button> >
]} <PaginatedDataList
> items={results}
<PaginatedDataList itemCount={count}
items={results} itemName={lookupHeader}
itemCount={count} qsConfig={this.qsConfig}
itemName={lookupHeader} toolbarColumns={columns}
qsConfig={this.qsConfig} renderItem={item => (
toolbarColumns={columns} <CheckboxListItem
renderItem={item => ( key={item.id}
<CheckboxListItem itemId={item.id}
key={item.id} name={item.name}
itemId={item.id} isSelected={lookupSelectedItems.some(i => i.id === item.id)}
name={item.name} onSelect={() => this.toggleSelected(item)}
isSelected={lookupSelectedItems.some(i => i.id === item.id)}
onSelect={() => this.toggleSelected(item)}
/>
)}
alignToolbarLeft
showPageSizeOptions={false}
paginationStyling={paginationStyling}
/> />
{lookupSelectedItems.length > 0 && ( )}
<SelectedList alignToolbarLeft
label={i18n._(t`Selected`)} showPageSizeOptions={false}
selected={lookupSelectedItems} paginationStyling={paginationStyling}
showOverflowAfter={5} />
onRemove={this.toggleSelected} {lookupSelectedItems.length > 0 && (
/> <SelectedList
)} label={i18n._(t`Selected`)}
{ error ? <div>error</div> : '' } selected={lookupSelectedItems}
</Modal> showOverflowAfter={5}
</Fragment> onRemove={this.toggleSelected}
)} />
</I18n> )}
{ error ? <div>error</div> : '' }
</Modal>
</Fragment>
); );
} }
} }
@@ -217,9 +215,9 @@ Lookup.propTypes = {
Lookup.defaultProps = { Lookup.defaultProps = {
id: 'lookup-search', id: 'lookup-search',
lookupHeader: 'items', lookupHeader: null,
name: null, name: null,
}; };
export { Lookup as _Lookup }; export { Lookup as _Lookup };
export default withNetwork(withRouter(Lookup)); export default withI18n()(withNetwork(withRouter(Lookup)));

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import { shape, number, string, bool, func } from 'prop-types'; import { shape, number, string, bool, func } from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import { import {
@@ -37,66 +37,63 @@ function NotificationListItem (props) {
detailUrl, detailUrl,
successTurnedOn, successTurnedOn,
errorTurnedOn, errorTurnedOn,
toggleNotification toggleNotification,
i18n
} = props; } = props;
return ( return (
<I18n> <DataListItem
{({ i18n }) => ( aria-labelledby={`items-list-item-${notification.id}`}
<DataListItem key={notification.id}
aria-labelledby={`items-list-item-${notification.id}`} >
key={notification.id} <DataListItemRow>
> <DataListItemCells dataListCells={[
<DataListItemRow> <DataListCell key="name">
<DataListItemCells dataListCells={[ <Link
<DataListCell key="name"> to={{
<Link pathname: detailUrl
to={{ }}
pathname: detailUrl css="margin-right: 1.5em;"
}} >
css="margin-right: 1.5em;" <b id={`items-list-item-${notification.id}`}>{notification.name}</b>
> </Link>
<b id={`items-list-item-${notification.id}`}>{notification.name}</b> <Badge
</Link> css="text-transform: capitalize;"
<Badge isRead
css="text-transform: capitalize;" >
isRead {notification.notification_type}
> </Badge>
{notification.notification_type} </DataListCell>,
</Badge> <DataListCell righthalf="true" key="toggles">
</DataListCell>, <Switch
<DataListCell righthalf="true" key="toggles"> id={`notification-${notification.id}-success-toggle`}
<Switch label={i18n._(t`Successful`)}
id={`notification-${notification.id}-success-toggle`} isChecked={successTurnedOn}
label={i18n._(t`Successful`)} isDisabled={!canToggleNotifications}
isChecked={successTurnedOn} onChange={() => toggleNotification(
isDisabled={!canToggleNotifications} notification.id,
onChange={() => toggleNotification( successTurnedOn,
notification.id, 'success'
successTurnedOn, )}
'success' aria-label={i18n._(t`Toggle notification success`)}
)}
aria-label={i18n._(t`Toggle notification success`)}
/>
<Switch
id={`notification-${notification.id}-error-toggle`}
label={i18n._(t`Failure`)}
isChecked={errorTurnedOn}
isDisabled={!canToggleNotifications}
onChange={() => toggleNotification(
notification.id,
errorTurnedOn,
'error'
)}
aria-label={i18n._(t`Toggle notification failure`)}
/>
</DataListCell>
]}
/> />
</DataListItemRow> <Switch
</DataListItem> id={`notification-${notification.id}-error-toggle`}
)} label={i18n._(t`Failure`)}
</I18n> isChecked={errorTurnedOn}
isDisabled={!canToggleNotifications}
onChange={() => toggleNotification(
notification.id,
errorTurnedOn,
'error'
)}
aria-label={i18n._(t`Toggle notification failure`)}
/>
</DataListCell>
]}
/>
</DataListItemRow>
</DataListItem>
); );
} }
@@ -118,4 +115,4 @@ NotificationListItem.defaultProps = {
successTurnedOn: false, successTurnedOn: false,
}; };
export default NotificationListItem; export default withI18n()(NotificationListItem);

View File

@@ -1,8 +1,8 @@
import React from 'react'; import React, { Fragment } from 'react';
import { Redirect, withRouter } from 'react-router-dom'; import { Redirect, withRouter } from 'react-router-dom';
import { withI18n } from '@lingui/react';
import { Trans } from '@lingui/macro'; import { t } from '@lingui/macro';
import { withRootDialog } from '../contexts/RootDialog'; import { withRootDialog } from '../contexts/RootDialog';
@@ -14,16 +14,13 @@ const NotifyAndRedirect = ({
strict, strict,
sensitive, sensitive,
setRootDialogMessage, setRootDialogMessage,
location location,
i18n
}) => { }) => {
setRootDialogMessage({ setRootDialogMessage({
title: '404', title: '404',
bodyText: ( bodyText: (
<Trans> <Fragment>{i18n._(t`Cannot find route ${(<strong>{location.pathname}</strong>)}.`)}</Fragment>
Cannot find route
<strong>{` ${location.pathname}`}</strong>
.
</Trans>
), ),
variant: 'warning' variant: 'warning'
}); });
@@ -41,4 +38,4 @@ const NotifyAndRedirect = ({
}; };
export { NotifyAndRedirect as _NotifyAndRedirect }; export { NotifyAndRedirect as _NotifyAndRedirect };
export default withRootDialog(withRouter(NotifyAndRedirect)); export default withI18n()(withRootDialog(withRouter(NotifyAndRedirect)));

View File

@@ -1,8 +1,7 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { I18n } from '@lingui/react';
import { import {
Dropdown, Dropdown,
DropdownItem, DropdownItem,
@@ -57,78 +56,75 @@ class PageHeaderToolbar extends Component {
isAboutDisabled, isAboutDisabled,
onAboutClick, onAboutClick,
onLogoutClick, onLogoutClick,
loggedInUser loggedInUser,
i18n
} = this.props; } = this.props;
return ( return (
<I18n> <Toolbar>
{({ i18n }) => ( <ToolbarGroup>
<Toolbar> <Tooltip position="left" content={<div>{i18n._(t`Info`)}</div>}>
<ToolbarGroup> <ToolbarItem>
<Tooltip position="left" content={<div>{i18n._(t`Info`)}</div>}> <Dropdown
<ToolbarItem> isPlain
<Dropdown isOpen={isHelpOpen}
isPlain position={DropdownPosition.right}
isOpen={isHelpOpen} onSelect={this.handleHelpSelect}
position={DropdownPosition.right} toggle={(
onSelect={this.handleHelpSelect} <DropdownToggle onToggle={this.handleHelpToggle}>
toggle={( <QuestionCircleIcon />
<DropdownToggle onToggle={this.handleHelpToggle}> </DropdownToggle>
<QuestionCircleIcon /> )}
</DropdownToggle> dropdownItems={[
<DropdownItem key="help" target="_blank" href={DOCLINK}>
{i18n._(t`Help`)}
</DropdownItem>,
<DropdownItem
key="about"
component="button"
isDisabled={isAboutDisabled}
onClick={onAboutClick}
>
{i18n._(t`About`)}
</DropdownItem>
]}
/>
</ToolbarItem>
</Tooltip>
<Tooltip position="left" content={<div>User</div>}>
<ToolbarItem>
<Dropdown
isPlain
isOpen={isUserOpen}
position={DropdownPosition.right}
onSelect={this.handleUserSelect}
toggle={(
<DropdownToggle onToggle={this.handleUserToggle}>
<UserIcon />
{loggedInUser && (
<span style={{ marginLeft: '10px' }}>
{loggedInUser.username}
</span>
)} )}
dropdownItems={[ </DropdownToggle>
<DropdownItem key="help" target="_blank" href={DOCLINK}> )}
{i18n._(t`Help`)} dropdownItems={[
</DropdownItem>, <DropdownItem key="user" href="#/home">
<DropdownItem {i18n._(t`User Details`)}
key="about" </DropdownItem>,
component="button" <DropdownItem
isDisabled={isAboutDisabled} key="logout"
onClick={onAboutClick} component="button"
> onClick={onLogoutClick}
{i18n._(t`About`)} >
</DropdownItem> {i18n._(t`Logout`)}
]} </DropdownItem>
/> ]}
</ToolbarItem> />
</Tooltip> </ToolbarItem>
<Tooltip position="left" content={<div>User</div>}> </Tooltip>
<ToolbarItem> </ToolbarGroup>
<Dropdown </Toolbar>
isPlain
isOpen={isUserOpen}
position={DropdownPosition.right}
onSelect={this.handleUserSelect}
toggle={(
<DropdownToggle onToggle={this.handleUserToggle}>
<UserIcon />
{loggedInUser && (
<span style={{ marginLeft: '10px' }}>
{loggedInUser.username}
</span>
)}
</DropdownToggle>
)}
dropdownItems={[
<DropdownItem key="user" href="#/home">
{i18n._(t`User Details`)}
</DropdownItem>,
<DropdownItem
key="logout"
component="button"
onClick={onLogoutClick}
>
{i18n._(t`Logout`)}
</DropdownItem>
]}
/>
</ToolbarItem>
</Tooltip>
</ToolbarGroup>
</Toolbar>
)}
</I18n>
); );
} }
} }
@@ -143,4 +139,4 @@ PageHeaderToolbar.defaultProps = {
isAboutDisabled: false isAboutDisabled: false
}; };
export default PageHeaderToolbar; export default withI18n()(PageHeaderToolbar);

View File

@@ -14,8 +14,8 @@ import {
EmptyStateBody, EmptyStateBody,
} from '@patternfly/react-core'; } from '@patternfly/react-core';
import { CubesIcon } from '@patternfly/react-icons'; import { CubesIcon } from '@patternfly/react-icons';
import { I18n, i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { Trans, t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { withRouter, Link } from 'react-router-dom'; import { withRouter, Link } from 'react-router-dom';
import Pagination from '../Pagination'; import Pagination from '../Pagination';
@@ -108,13 +108,15 @@ class PaginatedDataList extends React.Component {
showPageSizeOptions, showPageSizeOptions,
paginationStyling, paginationStyling,
location, location,
i18n
} = this.props; } = this.props;
const { error } = this.state; const { error } = this.state;
const [orderBy, sortOrder] = this.getSortOrder(); const [orderBy, sortOrder] = this.getSortOrder();
const queryParams = parseNamespacedQueryString(qsConfig, location.search); const queryParams = parseNamespacedQueryString(qsConfig, location.search);
const columns = toolbarColumns.length ? toolbarColumns : [{ name: i18n._(t`Name`), key: 'name', isSortable: true }];
return ( return (
<I18n> <Fragment>
{({ i18n }) => ( {error && (
<Fragment> <Fragment>
{error && ( {error && (
<Fragment> <Fragment>
@@ -128,24 +130,10 @@ class PaginatedDataList extends React.Component {
<EmptyState> <EmptyState>
<EmptyStateIcon icon={CubesIcon} /> <EmptyStateIcon icon={CubesIcon} />
<Title size="lg"> <Title size="lg">
<Trans> {i18n._(t`No ${ucFirst(itemNamePlural || pluralize(itemName))} Found`)}
No
{' '}
{ucFirst(itemNamePlural || pluralize(itemName))}
{' '}
Found
</Trans>
</Title> </Title>
<EmptyStateBody> <EmptyStateBody>
<Trans> {i18n._(t`Please add ${getArticle(itemName)} ${itemName} to populate this list`)}
Please add
{' '}
{getArticle(itemName)}
{' '}
{itemName}
{' '}
to populate this list
</Trans>
</EmptyStateBody> </EmptyStateBody>
</EmptyState> </EmptyState>
) : ( ) : (
@@ -153,7 +141,7 @@ class PaginatedDataList extends React.Component {
<DataListToolbar <DataListToolbar
sortedColumnKey={orderBy} sortedColumnKey={orderBy}
sortOrder={sortOrder} sortOrder={sortOrder}
columns={toolbarColumns} columns={columns}
onSearch={() => { }} onSearch={() => { }}
onSort={this.handleSort} onSort={this.handleSort}
showSelectAll={showSelectAll} showSelectAll={showSelectAll}
@@ -199,9 +187,67 @@ class PaginatedDataList extends React.Component {
/> />
</Fragment> </Fragment>
)} )}
</Fragment> // TODO: replace with proper error handling
)}
{items.length === 0 ? (
<EmptyState>
<EmptyStateIcon icon={CubesIcon} />
<Title size="lg">
{i18n._(t`No ${ucFirst(itemNamePlural || pluralize(itemName))} Found`)}
</Title>
<EmptyStateBody>
{i18n._(t`Please add ${getArticle(itemName)} ${itemName} to populate this list`)}
</EmptyStateBody>
</EmptyState>
) : (
<Fragment>
<DataListToolbar
sortedColumnKey={orderBy}
sortOrder={sortOrder}
columns={columns}
onSearch={() => { }}
onSort={this.handleSort}
showSelectAll={showSelectAll}
isAllSelected={isAllSelected}
onSelectAll={onSelectAll}
additionalControls={additionalControls}
/>
<DataList aria-label={i18n._(t`${ucFirst(pluralize(itemName))} List`)}>
{items.map(item => (renderItem ? renderItem(item) : (
<DataListItem
aria-labelledby={`items-list-item-${item.id}`}
key={item.id}
>
<DataListItemRow>
<DataListItemCells dataListCells={[
<DataListCell key="team-name">
<TextContent style={detailWrapperStyle}>
<Link to={{ pathname: item.url }}>
<Text
id={`items-list-item-${item.id}`}
style={detailLabelStyle}
>
{item.name}
</Text>
</Link>
</TextContent>
</DataListCell>
]}
/>
</DataListItemRow>
</DataListItem>
)))}
</DataList>
<Pagination
count={itemCount}
page={queryParams.page}
pageCount={this.getPageCount()}
page_size={queryParams.page_size}
onSetPage={this.handleSetPage}
/>
</Fragment> </Fragment>
)} )}
</I18n> </Fragment>
); );
} }
} }
@@ -235,9 +281,7 @@ PaginatedDataList.propTypes = {
PaginatedDataList.defaultProps = { PaginatedDataList.defaultProps = {
renderItem: null, renderItem: null,
toolbarColumns: [ toolbarColumns: [],
{ name: i18nMark('Name'), key: 'name', isSortable: true },
],
additionalControls: [], additionalControls: [],
itemName: 'item', itemName: 'item',
itemNamePlural: '', itemNamePlural: '',
@@ -250,4 +294,4 @@ PaginatedDataList.defaultProps = {
}; };
export { PaginatedDataList as _PaginatedDataList }; export { PaginatedDataList as _PaginatedDataList };
export default withRouter(PaginatedDataList); export default withI18n()(withRouter(PaginatedDataList));

View File

@@ -3,7 +3,7 @@ import { string, func } from 'prop-types';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import { Button as PFButton } from '@patternfly/react-core'; import { Button as PFButton } from '@patternfly/react-core';
import { PlusIcon } from '@patternfly/react-icons'; import { PlusIcon } from '@patternfly/react-icons';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import styled from 'styled-components'; import styled from 'styled-components';
@@ -20,39 +20,32 @@ const Button = styled(PFButton)`
} }
`; `;
function ToolbarAddButton ({ linkTo, onClick }) { function ToolbarAddButton ({ linkTo, onClick, i18n }) {
if (!linkTo && !onClick) { if (!linkTo && !onClick) {
throw new Error('ToolbarAddButton requires either `linkTo` or `onClick` prop'); throw new Error('ToolbarAddButton requires either `linkTo` or `onClick` prop');
} }
if (linkTo) { if (linkTo) {
// TODO: This should only be a <Link> (no <Button>) but CSS is off // TODO: This should only be a <Link> (no <Button>) but CSS is off
return ( return (
<I18n> <Link to={linkTo}>
{({ i18n }) => (
<Link to={linkTo}>
<Button
variant="primary"
aria-label={i18n._(t`Add`)}
>
<PlusIcon />
</Button>
</Link>
)}
</I18n>
);
}
return (
<I18n>
{({ i18n }) => (
<Button <Button
variant="primary" variant="primary"
aria-label={i18n._(t`Add`)} aria-label={i18n._(t`Add`)}
onClick={onClick}
> >
<PlusIcon /> <PlusIcon />
</Button> </Button>
)} </Link>
</I18n> );
}
return (
<Button
variant="primary"
aria-label={i18n._(t`Add`)}
onClick={onClick}
>
<PlusIcon />
</Button>
); );
} }
ToolbarAddButton.propTypes = { ToolbarAddButton.propTypes = {
@@ -64,4 +57,4 @@ ToolbarAddButton.defaultProps = {
onClick: null onClick: null
}; };
export default ToolbarAddButton; export default withI18n()(ToolbarAddButton);

View File

@@ -2,9 +2,9 @@ import React, { Fragment } from 'react';
import { func, bool, number, string, arrayOf, shape } from 'prop-types'; import { func, bool, number, string, arrayOf, shape } from 'prop-types';
import { Button as PFButton, Tooltip } from '@patternfly/react-core'; import { Button as PFButton, Tooltip } from '@patternfly/react-core';
import { TrashAltIcon } from '@patternfly/react-icons'; import { TrashAltIcon } from '@patternfly/react-icons';
import { I18n, i18nMark } from '@lingui/react';
import { Trans, t } from '@lingui/macro';
import styled from 'styled-components'; import styled from 'styled-components';
import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import AlertModal from '../AlertModal'; import AlertModal from '../AlertModal';
import { pluralize } from '../../util/strings'; import { pluralize } from '../../util/strings';
@@ -89,103 +89,95 @@ class ToolbarDeleteButton extends React.Component {
} }
renderTooltip () { renderTooltip () {
const { itemsToDelete, itemName } = this.props; const { itemsToDelete, itemName, i18n } = this.props;
const itemsUnableToDelete = itemsToDelete
.filter(cannotDelete)
.map(item => (
<div key={item.id}>
{item.name}
</div>
));
if (itemsToDelete.some(cannotDelete)) { if (itemsToDelete.some(cannotDelete)) {
return ( return (
<div> <div>
<Trans> {i18n._(t`You do not have permission to delete the following ${pluralize(itemName)}: ${itemsUnableToDelete}`)}
You dont have permission to delete the following
{' '}
{pluralize(itemName)}
:
</Trans>
{itemsToDelete
.filter(cannotDelete)
.map(item => (
<div key={item.id}>
{item.name}
</div>
))
}
</div> </div>
); );
} }
if (itemsToDelete.length) { if (itemsToDelete.length) {
return i18nMark('Delete'); return i18n._(t`Delete`);
} }
return i18nMark('Select a row to delete'); return i18n._(t`Select a row to delete`);
} }
render () { render () {
const { itemsToDelete, itemName } = this.props; const { itemsToDelete, itemName, i18n } = this.props;
const { isModalOpen } = this.state; const { isModalOpen } = this.state;
const isDisabled = itemsToDelete.length === 0 const isDisabled = itemsToDelete.length === 0
|| itemsToDelete.some(cannotDelete); || itemsToDelete.some(cannotDelete);
return ( return (
<I18n> <Fragment>
{({ i18n }) => ( <Tooltip
<Fragment> content={this.renderTooltip()}
<Tooltip position="left"
content={this.renderTooltip()} >
position="left" <Button
> className="awx-ToolBarBtn"
variant="plain"
aria-label={i18n._(t`Delete`)}
onClick={this.handleConfirmDelete}
isDisabled={isDisabled}
>
<TrashAltIcon className="awx-ToolBarTrashCanIcon" />
</Button>
</Tooltip>
{ isModalOpen && (
<AlertModal
variant="danger"
title={itemsToDelete === 1
? i18n._(t`Delete ${itemName}`)
: i18n._(t`Delete ${pluralize(itemName)}`)
}
isOpen={isModalOpen}
onClose={this.handleCancelDelete}
actions={[
<Button <Button
variant="plain" key="delete"
aria-label={i18n._(t`Delete`)}
onClick={this.handleConfirmDelete}
isDisabled={isDisabled}
>
<TrashAltIcon />
</Button>
</Tooltip>
{ isModalOpen && (
<AlertModal
variant="danger" variant="danger"
title={itemsToDelete === 1 aria-label={i18n._(t`confirm delete`)}
? i18n._(t`Delete ${itemName}`) onClick={this.handleDelete}
: i18n._(t`Delete ${pluralize(itemName)}`)
}
isOpen={isModalOpen}
onClose={this.handleCancelDelete}
actions={[
<Button
key="delete"
variant="danger"
aria-label={i18n._(t`confirm delete`)}
onClick={this.handleDelete}
>
{i18n._(t`Delete`)}
</Button>,
<Button
key="cancel"
variant="secondary"
aria-label={i18n._(t`cancel delete`)}
onClick={this.handleCancelDelete}
>
{i18n._(t`Cancel`)}
</Button>
]}
> >
{i18n._(t`Are you sure you want to delete:`)} {i18n._(t`Delete`)}
</Button>,
<Button
key="cancel"
variant="secondary"
aria-label={i18n._(t`cancel delete`)}
onClick={this.handleCancelDelete}
>
{i18n._(t`Cancel`)}
</Button>
]}
>
{i18n._(t`Are you sure you want to delete:`)}
<br />
{itemsToDelete.map((item) => (
<span key={item.id}>
<strong>
{item.name}
</strong>
<br /> <br />
{itemsToDelete.map((item) => ( </span>
<span key={item.id}> ))}
<strong> <br />
{item.name} </AlertModal>
</strong>
<br />
</span>
))}
<br />
</AlertModal>
)}
</Fragment>
)} )}
</I18n> </Fragment>
); );
} }
} }
export default ToolbarDeleteButton; export default withI18n()(ToolbarDeleteButton);

View File

@@ -1,7 +1,7 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { Trans, t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Button, Button,
Dropdown, Dropdown,
@@ -106,7 +106,8 @@ class Pagination extends Component {
page_size, page_size,
pageSizeOptions, pageSizeOptions,
showPageSizeOptions, showPageSizeOptions,
style style,
i18n
} = this.props; } = this.props;
const { value, isOpen } = this.state; const { value, isOpen } = this.state;
let opts = []; let opts = [];
@@ -135,98 +136,91 @@ class Pagination extends Component {
)); ));
return ( return (
<I18n> <div className="awx-pagination" style={style}>
{({ i18n }) => ( {showPageSizeOptions && (
<div className="awx-pagination" style={style}> <div className="awx-pagination__page-size-selection">
{showPageSizeOptions && ( {i18n._(t`Items Per Page`)}
<div className="awx-pagination__page-size-selection"> <Dropdown
<Trans>Items Per Page</Trans> onToggle={this.onTogglePageSize}
<Dropdown onSelect={this.onSelectPageSize}
direction={up}
isOpen={isOpen}
toggle={(
<DropdownToggle
className="togglePageSize"
onToggle={this.onTogglePageSize} onToggle={this.onTogglePageSize}
onSelect={this.onSelectPageSize} >
direction={up} {page_size}
isOpen={isOpen} </DropdownToggle>
toggle={(
<DropdownToggle
className="togglePageSize"
onToggle={this.onTogglePageSize}
>
{page_size}
</DropdownToggle>
)}
dropdownItems={dropdownItems}
/>
</div>
)}
<div className="awx-pagination__counts">
<div className="awx-pagination__item-count">
<Trans>{`Items ${itemMin} ${itemMax} of ${count}`}</Trans>
</div>
{pageCount !== 1 && (
<div className="awx-pagination__page-count">
<div className="pf-c-input-group pf-m-previous">
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`First`)}
isDisabled={isOnFirst}
onClick={this.onFirst}
>
<i className="fas fa-angle-double-left" />
</Button>
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Previous`)}
isDisabled={isOnFirst}
onClick={this.onPrevious}
>
<i className="fas fa-angle-left" />
</Button>
</div>
<form
className="awx-pagination__page-input-form"
onSubmit={this.onSubmit}
>
<Trans>
{'Page '}
<TextInput
className="awx-pagination__page-input"
aria-label={i18n._(t`Page Number`)}
value={value}
type="text"
onChange={this.onPageChange}
/>
{' of '}
{pageCount}
</Trans>
</form>
<div className="pf-c-input-group">
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Next`)}
isDisabled={isOnLast}
onClick={this.onNext}
>
<i className="fas fa-angle-right" />
</Button>
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Last`)}
isDisabled={isOnLast}
onClick={this.onLast}
>
<i className="fas fa-angle-double-right" />
</Button>
</div>
</div>
)} )}
</div> dropdownItems={dropdownItems}
/>
</div> </div>
)} )}
</I18n> <div className="awx-pagination__counts">
<div className="awx-pagination__item-count">
{i18n._(t`Items ${itemMin} ${itemMax} of ${count}`)}
</div>
{pageCount !== 1 && (
<div className="awx-pagination__page-count">
<div className="pf-c-input-group pf-m-previous">
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`First`)}
isDisabled={isOnFirst}
onClick={this.onFirst}
>
<i className="fas fa-angle-double-left" />
</Button>
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Previous`)}
isDisabled={isOnFirst}
onClick={this.onPrevious}
>
<i className="fas fa-angle-left" />
</Button>
</div>
<form
className="awx-pagination__page-input-form"
onSubmit={this.onSubmit}
>
{i18n._(t`Page `)}
<TextInput
className="awx-pagination__page-input"
aria-label={i18n._(t`Page Number`)}
value={value}
type="text"
onChange={this.onPageChange}
/>
{i18n._(t` of ${pageCount}`)}
</form>
<div className="pf-c-input-group">
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Next`)}
isDisabled={isOnLast}
onClick={this.onNext}
>
<i className="fas fa-angle-right" />
</Button>
<Button
className="awx-pagination__page-button"
variant="tertiary"
aria-label={i18n._(t`Last`)}
isDisabled={isOnLast}
onClick={this.onLast}
>
<i className="fas fa-angle-double-right" />
</Button>
</div>
</div>
)}
</div>
</div>
); );
} }
} }
@@ -248,4 +242,4 @@ Pagination.defaultProps = {
showPageSizeOptions: true showPageSizeOptions: true
}; };
export default Pagination; export default withI18n()(Pagination);

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Button as PFButton, Button as PFButton,
@@ -90,7 +90,8 @@ class Search extends React.Component {
render () { render () {
const { up } = DropdownPosition; const { up } = DropdownPosition;
const { const {
columns columns,
i18n
} = this.props; } = this.props;
const { const {
isSearchDropdownOpen, isSearchDropdownOpen,
@@ -109,41 +110,37 @@ class Search extends React.Component {
)); ));
return ( return (
<I18n> <div className="pf-c-input-group">
{({ i18n }) => ( <Dropdown
<div className="pf-c-input-group"> onToggle={this.handleDropdownToggle}
<Dropdown onSelect={this.handleDropdownSelect}
direction={up}
isOpen={isSearchDropdownOpen}
toggle={(
<DropdownToggle
id="awx-search"
onToggle={this.handleDropdownToggle} onToggle={this.handleDropdownToggle}
onSelect={this.handleDropdownSelect}
direction={up}
isOpen={isSearchDropdownOpen}
toggle={(
<DropdownToggle
id="awx-search"
onToggle={this.handleDropdownToggle}
>
{searchColumnName}
</DropdownToggle>
)}
dropdownItems={searchDropdownItems}
/>
<TextInput
type="search"
aria-label="Search text input"
value={searchValue}
onChange={this.handleSearchInputChange}
style={{ height: '30px' }}
/>
<Button
variant="tertiary"
aria-label={i18n._(t`Search`)}
onClick={this.handleSearch}
> >
<SearchIcon /> {searchColumnName}
</Button> </DropdownToggle>
</div> )}
)} dropdownItems={searchDropdownItems}
</I18n> />
<TextInput
type="search"
aria-label={i18n._(t`Search text input`)}
value={searchValue}
onChange={this.handleSearchInputChange}
style={{ height: '30px' }}
/>
<Button
variant="tertiary"
aria-label={i18n._(t`Search`)}
onClick={this.handleSearch}
>
<SearchIcon />
</Button>
</div>
); );
} }
} }
@@ -159,4 +156,4 @@ Search.defaultProps = {
sortedColumnKey: 'name' sortedColumnKey: 'name'
}; };
export default Search; export default withI18n()(Search);

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Button, Button,
@@ -85,7 +85,8 @@ class Sort extends React.Component {
const { const {
columns, columns,
sortedColumnKey, sortedColumnKey,
sortOrder sortOrder,
i18n
} = this.props; } = this.props;
const { const {
isSortDropdownOpen isSortDropdownOpen
@@ -109,40 +110,36 @@ class Sort extends React.Component {
} }
return ( return (
<I18n> <React.Fragment>
{({ i18n }) => ( { sortDropdownItems.length > 1 && (
<React.Fragment> <Dropdown
{ sortDropdownItems.length > 1 && ( style={{ marginRight: '20px' }}
<Dropdown onToggle={this.handleDropdownToggle}
style={{ marginRight: '20px' }} onSelect={this.handleDropdownSelect}
direction={up}
isOpen={isSortDropdownOpen}
toggle={(
<DropdownToggle
id="awx-sort"
onToggle={this.handleDropdownToggle} onToggle={this.handleDropdownToggle}
onSelect={this.handleDropdownSelect} >
direction={up} {sortedColumnName}
isOpen={isSortDropdownOpen} </DropdownToggle>
toggle={(
<DropdownToggle
id="awx-sort"
onToggle={this.handleDropdownToggle}
>
{sortedColumnName}
</DropdownToggle>
)}
dropdownItems={sortDropdownItems}
/>
)} )}
<Button dropdownItems={sortDropdownItems}
onClick={this.handleSort} />
variant="plain"
aria-label={i18n._(t`Sort`)}
css="padding: 0;"
>
<IconWrapper>
<SortIcon />
</IconWrapper>
</Button>
</React.Fragment>
)} )}
</I18n> <Button
onClick={this.handleSort}
variant="plain"
aria-label={i18n._(t`Sort`)}
css="padding: 0;"
>
<IconWrapper>
<SortIcon />
</IconWrapper>
</Button>
</React.Fragment>
); );
} }
} }
@@ -160,4 +157,4 @@ Sort.defaultProps = {
sortedColumnKey: 'name' sortedColumnKey: 'name'
}; };
export default Sort; export default withI18n()(Sort);

View File

@@ -1,7 +1,7 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { Brand } from '@patternfly/react-core'; import { Brand } from '@patternfly/react-core';
@@ -34,6 +34,7 @@ class TowerLogo extends Component {
render () { render () {
const { hover } = this.state; const { hover } = this.state;
const { i18n } = this.props;
let src = TowerLogoHeader; let src = TowerLogoHeader;
@@ -42,19 +43,15 @@ class TowerLogo extends Component {
} }
return ( return (
<I18n> <Brand
{({ i18n }) => ( src={src}
<Brand alt={i18n._(t`Tower Brand Image`)}
src={src} onMouseOut={this.onHover}
alt={i18n._(t`Tower Brand Image`)} onMouseOver={this.onHover}
onMouseOut={this.onHover} onBlur={this.onHover}
onMouseOver={this.onHover} onFocus={this.onHover}
onBlur={this.onHover} onClick={this.onClick}
onFocus={this.onHover} />
onClick={this.onClick}
/>
)}
</I18n>
); );
} }
} }
@@ -67,4 +64,4 @@ TowerLogo.defaultProps = {
linkTo: null, linkTo: null,
}; };
export default withRouter(TowerLogo); export default withI18n()(withRouter(TowerLogo));

View File

@@ -4,7 +4,8 @@ import React, { Component } from 'react';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { withRootDialog } from './RootDialog'; import { withRootDialog } from './RootDialog';
@@ -33,27 +34,27 @@ class Provider extends Component {
} }
handle401 () { handle401 () {
const { handle401, history, setRootDialogMessage } = this.props; const { handle401, history, setRootDialogMessage, i18n } = this.props;
if (handle401) { if (handle401) {
handle401(); handle401();
return; return;
} }
history.replace('/login'); history.replace('/login');
setRootDialogMessage({ setRootDialogMessage({
bodyText: i18nMark('You have been logged out.') bodyText: i18n._(t`You have been logged out.`)
}); });
} }
handle404 () { handle404 () {
const { handle404, history, setRootDialogMessage } = this.props; const { handle404, history, setRootDialogMessage, i18n } = this.props;
if (handle404) { if (handle404) {
handle404(); handle404();
return; return;
} }
history.replace('/home'); history.replace('/home');
setRootDialogMessage({ setRootDialogMessage({
title: i18nMark('404'), title: i18n._(t`404`),
bodyText: i18nMark('Cannot find resource.'), bodyText: i18n._(t`Cannot find resource.`),
variant: 'warning' variant: 'warning'
}); });
} }
@@ -72,7 +73,7 @@ class Provider extends Component {
} }
export { Provider as _NetworkProvider }; export { Provider as _NetworkProvider };
export const NetworkProvider = withRootDialog(withRouter(Provider)); export const NetworkProvider = withI18n()(withRootDialog(withRouter(Provider)));
export function withNetwork (Child) { export function withNetwork (Child) {
return (props) => ( return (props) => (

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Applications extends Component { class Applications extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Applications</Trans> {i18n._(t`Applications`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Applications extends Component {
} }
} }
export default Applications; export default withI18n()(Applications);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class AuthSettings extends Component { class AuthSettings extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Authentication Settings</Trans> {i18n._(t`Authentication Settings`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class AuthSettings extends Component {
} }
} }
export default AuthSettings; export default withI18n()(AuthSettings);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class CredentialTypes extends Component { class CredentialTypes extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Credential Types</Trans> {i18n._(t`Credential Types`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class CredentialTypes extends Component {
} }
} }
export default CredentialTypes; export default withI18n()(CredentialTypes);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Credentials extends Component { class Credentials extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Credentials</Trans> {i18n._(t`Credentials`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Credentials extends Component {
} }
} }
export default Credentials; export default withI18n()(Credentials);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Dashboard extends Component { class Dashboard extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Dashboard</Trans> {i18n._(t`Dashboard`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Dashboard extends Component {
} }
} }
export default Dashboard; export default withI18n()(Dashboard);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class InstanceGroups extends Component { class InstanceGroups extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Instance Groups</Trans> {i18n._(t`Instance Groups`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class InstanceGroups extends Component {
} }
} }
export default InstanceGroups; export default withI18n()(InstanceGroups);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Inventories extends Component { class Inventories extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Inventories</Trans> {i18n._(t`Inventories`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Inventories extends Component {
} }
} }
export default Inventories; export default withI18n()(Inventories);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class InventoryScripts extends Component { class InventoryScripts extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Inventory Scripts</Trans> {i18n._(t`Inventory Scripts`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class InventoryScripts extends Component {
} }
} }
export default InventoryScripts; export default withI18n()(InventoryScripts);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Jobs extends Component { class Jobs extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Jobs</Trans> {i18n._(t`Jobs`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Jobs extends Component {
} }
} }
export default Jobs; export default withI18n()(Jobs);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class JobsSettings extends Component { class JobsSettings extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Jobs Settings</Trans> {i18n._(t`Jobs Settings`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class JobsSettings extends Component {
} }
} }
export default JobsSettings; export default withI18n()(JobsSettings);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class License extends Component { class License extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>License</Trans> {i18n._(t`License`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class License extends Component {
} }
} }
export default License; export default withI18n()(License);

View File

@@ -1,6 +1,6 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import { Redirect, withRouter } from 'react-router-dom'; import { Redirect, withRouter } from 'react-router-dom';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
LoginForm, LoginForm,
@@ -62,7 +62,7 @@ class AWXLogin extends Component {
render () { render () {
const { username, password, isInputValid, isAuthenticated } = this.state; const { username, password, isInputValid, isAuthenticated } = this.state;
const { alt, loginInfo, logo, bodyText: errorMessage } = this.props; const { alt, loginInfo, logo, bodyText: errorMessage, i18n } = this.props;
const logoSrc = logo ? `data:image/jpeg;${logo}` : towerLogo; const logoSrc = logo ? `data:image/jpeg;${logo}` : towerLogo;
if (isAuthenticated) { if (isAuthenticated) {
@@ -70,34 +70,30 @@ class AWXLogin extends Component {
} }
return ( return (
<I18n> <LoginPage
{({ i18n }) => ( brandImgSrc={logoSrc}
<LoginPage brandImgAlt={alt || 'Ansible Tower'}
brandImgSrc={logoSrc} loginTitle={i18n._(t`Welcome to Ansible Tower! Please Sign In.`)}
brandImgAlt={alt || 'Ansible Tower'} textContent={loginInfo}
loginTitle={i18n._(t`Welcome to Ansible Tower! Please Sign In.`)} >
textContent={loginInfo} <LoginForm
> className={errorMessage && 'pf-m-error'}
<LoginForm usernameLabel={i18n._(t`Username`)}
className={errorMessage && 'pf-m-error'} passwordLabel={i18n._(t`Password`)}
usernameLabel={i18n._(t`Username`)} showHelperText={!isInputValid || !!errorMessage}
passwordLabel={i18n._(t`Password`)} helperText={errorMessage || i18n._(t`Invalid username or password. Please try again.`)}
showHelperText={!isInputValid || !!errorMessage} usernameValue={username}
helperText={errorMessage || i18n._(t`Invalid username or password. Please try again.`)} passwordValue={password}
usernameValue={username} isValidUsername={isInputValid}
passwordValue={password} isValidPassword={isInputValid}
isValidUsername={isInputValid} onChangeUsername={this.onChangeUsername}
isValidPassword={isInputValid} onChangePassword={this.onChangePassword}
onChangeUsername={this.onChangeUsername} onLoginButtonClick={this.onLoginButtonClick}
onChangePassword={this.onChangePassword} />
onLoginButtonClick={this.onLoginButtonClick} </LoginPage>
/>
</LoginPage>
)}
</I18n>
); );
} }
} }
export { AWXLogin as _AWXLogin }; export { AWXLogin as _AWXLogin };
export default withNetwork(withRootDialog(withRouter(AWXLogin))); export default withI18n()(withNetwork(withRootDialog(withRouter(AWXLogin))));

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class ManagementJobs extends Component { class ManagementJobs extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Management Jobs</Trans> {i18n._(t`Management Jobs`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class ManagementJobs extends Component {
} }
} }
export default ManagementJobs; export default withI18n()(ManagementJobs);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class NotificationTemplates extends Component { class NotificationTemplates extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Notification Templates</Trans> {i18n._(t`Notification Templates`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class NotificationTemplates extends Component {
} }
} }
export default NotificationTemplates; export default withI18n()(NotificationTemplates);

View File

@@ -1,7 +1,7 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Route, withRouter, Switch } from 'react-router-dom'; import { Route, withRouter, Switch } from 'react-router-dom';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { Trans } from '@lingui/macro'; import { t } from '@lingui/macro';
import { Config } from '../../contexts/Config'; import { Config } from '../../contexts/Config';
import { NetworkProvider } from '../../contexts/Network'; import { NetworkProvider } from '../../contexts/Network';
@@ -14,34 +14,42 @@ import OrganizationAdd from './screens/OrganizationAdd';
import Organization from './screens/Organization/Organization'; import Organization from './screens/Organization/Organization';
class Organizations extends Component { class Organizations extends Component {
state = { constructor (props) {
breadcrumbConfig: { super(props);
'/organizations': i18nMark('Organizations'),
'/organizations/add': i18nMark('Create New Organization') const { i18n } = props;
}
}; this.state = {
breadcrumbConfig: {
'/organizations': i18n._(t`Organizations`),
'/organizations/add': i18n._(t`Create New Organization`)
}
};
}
setBreadcrumbConfig = (organization) => { setBreadcrumbConfig = (organization) => {
const { i18n } = this.props;
if (!organization) { if (!organization) {
return; return;
} }
const breadcrumbConfig = { const breadcrumbConfig = {
'/organizations': i18nMark('Organizations'), '/organizations': i18n._(t`Organizations`),
'/organizations/add': i18nMark('Create New Organization'), '/organizations/add': i18n._(t`Create New Organization`),
[`/organizations/${organization.id}`]: `${organization.name}`, [`/organizations/${organization.id}`]: `${organization.name}`,
[`/organizations/${organization.id}/edit`]: i18nMark('Edit Details'), [`/organizations/${organization.id}/edit`]: i18n._(t`Edit Details`),
[`/organizations/${organization.id}/details`]: i18nMark('Details'), [`/organizations/${organization.id}/details`]: i18n._(t`Details`),
[`/organizations/${organization.id}/access`]: i18nMark('Access'), [`/organizations/${organization.id}/access`]: i18n._(t`Access`),
[`/organizations/${organization.id}/teams`]: i18nMark('Teams'), [`/organizations/${organization.id}/teams`]: i18n._(t`Teams`),
[`/organizations/${organization.id}/notifications`]: i18nMark('Notifications'), [`/organizations/${organization.id}/notifications`]: i18n._(t`Notifications`),
}; };
this.setState({ breadcrumbConfig }); this.setState({ breadcrumbConfig });
} }
render () { render () {
const { match, history, location, setRootDialogMessage } = this.props; const { match, history, location, setRootDialogMessage, i18n } = this.props;
const { breadcrumbConfig } = this.state; const { breadcrumbConfig } = this.state;
return ( return (
@@ -65,11 +73,11 @@ class Organizations extends Component {
setRootDialogMessage({ setRootDialogMessage({
title: '404', title: '404',
bodyText: ( bodyText: (
<Trans> <Fragment>
Cannot find organization with ID {i18n._(t`Cannot find organization with ID`)}
<strong>{` ${newRouteMatch.params.id}`}</strong> <strong>{` ${newRouteMatch.params.id}`}</strong>
. .
</Trans> </Fragment>
), ),
variant: 'warning' variant: 'warning'
}); });
@@ -101,4 +109,4 @@ class Organizations extends Component {
} }
export { Organizations as _Organizations }; export { Organizations as _Organizations };
export default withRootDialog(withRouter(Organizations)); export default withI18n()(withRootDialog(withRouter(Organizations)));

View File

@@ -1,8 +1,8 @@
import React from 'react'; import React, { Fragment } from 'react';
import { func, string } from 'prop-types'; import { func, string } from 'prop-types';
import { Button } from '@patternfly/react-core'; import { Button } from '@patternfly/react-core';
import { I18n, i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t, Trans } from '@lingui/macro'; import { t } from '@lingui/macro';
import AlertModal from '../../../components/AlertModal'; import AlertModal from '../../../components/AlertModal';
import { Role } from '../../../types'; import { Role } from '../../../types';
@@ -24,57 +24,43 @@ class DeleteRoleConfirmationModal extends React.Component {
} }
render () { render () {
const { role, username, onCancel, onConfirm } = this.props; const { role, username, onCancel, onConfirm, i18n } = this.props;
const title = `Remove ${this.isTeamRole() ? 'Team' : 'User'} Access`; const title = i18n._(t`Remove ${this.isTeamRole() ? i18n._(t`Team`) : i18n._(t`User`)} Access`);
return ( return (
<I18n> <AlertModal
{({ i18n }) => ( variant="danger"
<AlertModal title={title}
isOpen
onClose={onCancel}
actions={[
<Button
key="delete"
variant="danger" variant="danger"
title={i18nMark(title)} aria-label="Confirm delete"
isOpen onClick={onConfirm}
onClose={onCancel}
actions={[
<Button
key="delete"
variant="danger"
aria-label="Confirm delete"
onClick={onConfirm}
>
{i18n._(t`Delete`)}
</Button>,
<Button key="cancel" variant="secondary" onClick={onCancel}>
{i18n._(t`Cancel`)}
</Button>
]}
> >
{this.isTeamRole() ? ( {i18n._(t`Delete`)}
<Trans> </Button>,
Are you sure you want to remove <Button key="cancel" variant="secondary" onClick={onCancel}>
<b>{` ${role.name} `}</b> {i18n._(t`Cancel`)}
access from </Button>
<b>{` ${role.team_name}`}</b> ]}
? Doing so affects all members of the team. >
<br /> {this.isTeamRole() ? (
<br /> <Fragment>
If you {i18n._(t`Are you sure you want to remove ${role.name} access from ${role.team_name}? Doing so affects all members of the team.`)}
<b><i> only </i></b> <br />
want to remove access for this particular user, please remove them from the team. <br />
</Trans> {i18n._(t`If you ${(<b><i>only</i></b>)} want to remove access for this particular user, please remove them from the team.`)}
) : ( </Fragment>
<Trans> ) : (
Are you sure you want to remove <Fragment>
<b>{` ${role.name} `}</b> {i18n._(t`Are you sure you want to remove ${role.name} access from ${username}?`)}
access from </Fragment>
<b>{` ${username}`}</b>
?
</Trans>
)}
</AlertModal>
)} )}
</I18n> </AlertModal>
); );
} }
} }
export default DeleteRoleConfirmationModal; export default withI18n()(DeleteRoleConfirmationModal);

View File

@@ -1,20 +1,14 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { I18n, i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { FormGroup, Tooltip } from '@patternfly/react-core'; import { FormGroup, Tooltip } from '@patternfly/react-core';
import { QuestionCircleIcon } from '@patternfly/react-icons'; import { QuestionCircleIcon } from '@patternfly/react-icons';
import { t } from '@lingui/macro';
import Lookup from '../../../components/Lookup'; import Lookup from '../../../components/Lookup';
import { withNetwork } from '../../../contexts/Network'; import { withNetwork } from '../../../contexts/Network';
const INSTANCE_GROUPS_LOOKUP_COLUMNS = [
{ name: i18nMark('Name'), key: 'name', isSortable: true },
{ name: i18nMark('Modified'), key: 'modified', isSortable: false, isNumeric: true },
{ name: i18nMark('Created'), key: 'created', isSortable: false, isNumeric: true }
];
class InstanceGroupsLookup extends React.Component { class InstanceGroupsLookup extends React.Component {
constructor (props) { constructor (props) {
super(props); super(props);
@@ -29,43 +23,43 @@ class InstanceGroupsLookup extends React.Component {
} }
render () { render () {
const { value, tooltip, onChange } = this.props; const { value, tooltip, onChange, i18n } = this.props;
return ( return (
<I18n> <FormGroup
{({ i18n }) => ( label={(
<FormGroup <Fragment>
label={( {i18n._(t`Instance Groups`)}
<Fragment> {' '}
{i18n._(t`Instance Groups`)} {
{' '} tooltip && (
{ <Tooltip
tooltip && ( position="right"
<Tooltip content={tooltip}
position="right" >
content={tooltip} <QuestionCircleIcon />
> </Tooltip>
<QuestionCircleIcon /> )
</Tooltip> }
) </Fragment>
}
</Fragment>
)}
fieldId="org-instance-groups"
>
<Lookup
id="org-instance-groups"
lookupHeader={i18n._(t`Instance Groups`)}
name="instanceGroups"
value={value}
onLookupSave={onChange}
getItems={this.getInstanceGroups}
columns={INSTANCE_GROUPS_LOOKUP_COLUMNS}
sortedColumnKey="name"
/>
</FormGroup>
)} )}
</I18n> fieldId="org-instance-groups"
>
<Lookup
id="org-instance-groups"
lookupHeader={i18n._(t`Instance Groups`)}
name="instanceGroups"
value={value}
onLookupSave={onChange}
getItems={this.getInstanceGroups}
columns={[
{ name: i18n._(t`Name`), key: 'name', isSortable: true },
{ name: i18n._(t`Modified`), key: 'modified', isSortable: false, isNumeric: true },
{ name: i18n._(t`Created`), key: 'created', isSortable: false, isNumeric: true }
]}
sortedColumnKey="name"
/>
</FormGroup>
); );
} }
} }
@@ -80,4 +74,4 @@ InstanceGroupsLookup.defaultProps = {
tooltip: '', tooltip: '',
}; };
export default withNetwork(InstanceGroupsLookup); export default withI18n()(withNetwork(InstanceGroupsLookup));

View File

@@ -1,6 +1,6 @@
import React from 'react'; import React from 'react';
import { func } from 'prop-types'; import { func } from 'prop-types';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
DataListItem, DataListItem,
@@ -81,96 +81,92 @@ class OrganizationAccessItem extends React.Component {
} }
render () { render () {
const { accessRecord, onRoleDelete } = this.props; const { accessRecord, onRoleDelete, i18n } = this.props;
const [teamRoles, userRoles] = this.getRoleLists(); const [teamRoles, userRoles] = this.getRoleLists();
return ( return (
<I18n> <DataListItem aria-labelledby="access-list-item" key={accessRecord.id}>
{({ i18n }) => ( <DataListItemRow>
<DataListItem aria-labelledby="access-list-item" key={accessRecord.id}> <DataListItemCells dataListCells={[
<DataListItemRow> <DataListCell key="name">
<DataListItemCells dataListCells={[ {accessRecord.username && (
<DataListCell key="name"> <TextContent style={detailWrapperStyle}>
{accessRecord.username && ( {accessRecord.url ? (
<TextContent style={detailWrapperStyle}> <Link to={{ pathname: accessRecord.url }}>
{accessRecord.url ? ( <Text component={TextVariants.h6} style={detailLabelStyle}>
<Link to={{ pathname: accessRecord.url }}> {accessRecord.username}
<Text component={TextVariants.h6} style={detailLabelStyle}> </Text>
{accessRecord.username} </Link>
</Text>
</Link>
) : (
<Text component={TextVariants.h6} style={detailLabelStyle}>
{accessRecord.username}
</Text>
)}
</TextContent>
)}
{accessRecord.first_name || accessRecord.last_name ? (
<Detail
label={i18n._(t`Name`)}
value={`${accessRecord.first_name} ${accessRecord.last_name}`}
url={null}
customStyles={null}
/>
) : ( ) : (
null <Text component={TextVariants.h6} style={detailLabelStyle}>
{accessRecord.username}
</Text>
)} )}
</DataListCell>, </TextContent>
<DataListCell key="roles"> )}
{userRoles.length > 0 && ( {accessRecord.first_name || accessRecord.last_name ? (
<ul style={userRolesWrapperStyle}> <Detail
<Text component={TextVariants.h6} style={detailLabelStyle}> label={i18n._(t`Name`)}
{i18n._(t`User Roles`)} value={`${accessRecord.first_name} ${accessRecord.last_name}`}
</Text> url={null}
{userRoles.map(role => ( customStyles={null}
role.user_capabilities.unattach ? ( />
<Chip ) : (
key={role.id} null
className="awx-c-chip" )}
onClick={() => { onRoleDelete(role, accessRecord); }} </DataListCell>,
> <DataListCell key="roles">
{role.name} {userRoles.length > 0 && (
</Chip> <ul style={userRolesWrapperStyle}>
) : ( <Text component={TextVariants.h6} style={detailLabelStyle}>
<BasicChip key={role.id}> {i18n._(t`User Roles`)}
{role.name} </Text>
</BasicChip> {userRoles.map(role => (
) role.user_capabilities.unattach ? (
))} <Chip
</ul> key={role.id}
)} className="awx-c-chip"
{teamRoles.length > 0 && ( onClick={() => { onRoleDelete(role, accessRecord); }}
<ul style={userRolesWrapperStyle}> >
<Text component={TextVariants.h6} style={detailLabelStyle}> {role.name}
{i18n._(t`Team Roles`)} </Chip>
</Text> ) : (
{teamRoles.map(role => ( <BasicChip key={role.id}>
role.user_capabilities.unattach ? ( {role.name}
<Chip </BasicChip>
key={role.id} )
className="awx-c-chip" ))}
onClick={() => { onRoleDelete(role, accessRecord); }} </ul>
> )}
{role.name} {teamRoles.length > 0 && (
</Chip> <ul style={userRolesWrapperStyle}>
) : ( <Text component={TextVariants.h6} style={detailLabelStyle}>
<BasicChip key={role.id}> {i18n._(t`Team Roles`)}
{role.name} </Text>
</BasicChip> {teamRoles.map(role => (
) role.user_capabilities.unattach ? (
))} <Chip
</ul> key={role.id}
)} className="awx-c-chip"
</DataListCell> onClick={() => { onRoleDelete(role, accessRecord); }}
]} >
/> {role.name}
</DataListItemRow> </Chip>
</DataListItem> ) : (
)} <BasicChip key={role.id}>
</I18n> {role.name}
</BasicChip>
)
))}
</ul>
)}
</DataListCell>
]}
/>
</DataListItemRow>
</DataListItem>
); );
} }
} }
export default OrganizationAccessItem; export default withI18n()(OrganizationAccessItem);

View File

@@ -2,7 +2,7 @@ import React, { Component } from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { Formik, Field } from 'formik'; import { Formik, Field } from 'formik';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
Form, Form,
@@ -83,76 +83,72 @@ class OrganizationForm extends Component {
} }
render () { render () {
const { organization, handleCancel } = this.props; const { organization, handleCancel, i18n } = this.props;
const { instanceGroups, formIsValid, error } = this.state; const { instanceGroups, formIsValid, error } = this.state;
const defaultVenv = '/venv/ansible/'; const defaultVenv = '/venv/ansible/';
return ( return (
<I18n> <Formik
{({ i18n }) => ( initialValues={{
<Formik name: organization.name,
initialValues={{ description: organization.description,
name: organization.name, custom_virtualenv: organization.custom_virtualenv || '',
description: organization.description, }}
custom_virtualenv: organization.custom_virtualenv || '', onSubmit={this.handleSubmit}
}} render={formik => (
onSubmit={this.handleSubmit} <Form autoComplete="off" onSubmit={formik.handleSubmit}>
render={formik => ( <FormRow>
<Form autoComplete="off" onSubmit={formik.handleSubmit}> <FormField
<FormRow> id="org-name"
<FormField name="name"
id="org-name" type="text"
name="name" label={i18n._(t`Name`)}
type="text" validate={required(null, i18n)}
label={i18n._(t`Name`)} isRequired
validate={required()} />
isRequired <FormField
/> id="org-description"
<FormField name="description"
id="org-description" type="text"
name="description" label={i18n._(t`Description`)}
type="text" />
label={i18n._(t`Description`)} <Config>
/> {({ custom_virtualenvs }) => (
<Config> custom_virtualenvs && custom_virtualenvs.length > 1 && (
{({ custom_virtualenvs }) => ( <Field
custom_virtualenvs && custom_virtualenvs.length > 1 && ( name="custom_virtualenv"
<Field render={({ field }) => (
name="custom_virtualenv" <FormGroup
render={({ field }) => ( fieldId="org-custom-virtualenv"
<FormGroup label={i18n._(t`Ansible Environment`)}
fieldId="org-custom-virtualenv" >
label={i18n._(t`Ansible Environment`)} <AnsibleSelect
> data={custom_virtualenvs}
<AnsibleSelect defaultSelected={defaultVenv}
data={custom_virtualenvs} label={i18n._(t`Ansible Environment`)}
defaultSelected={defaultVenv} {...field}
label={i18n._(t`Ansible Environment`)} />
{...field} </FormGroup>
/> )}
</FormGroup> />
)} )
/> )}
) </Config>
)} </FormRow>
</Config> <InstanceGroupsLookup
</FormRow> value={instanceGroups}
<InstanceGroupsLookup onChange={this.handleInstanceGroupsChange}
value={instanceGroups} tooltip={i18n._(t`Select the Instance Groups for this Organization to run on.`)}
onChange={this.handleInstanceGroupsChange} />
tooltip={i18n._(t`Select the Instance Groups for this Organization to run on.`)} <FormActionGroup
/> onCancel={handleCancel}
<FormActionGroup onSubmit={formik.handleSubmit}
onCancel={handleCancel} submitDisabled={!formIsValid}
onSubmit={formik.handleSubmit} />
submitDisabled={!formIsValid} {error ? <div>error</div> : null}
/> </Form>
{error ? <div>error</div> : null}
</Form>
)}
/>
)} )}
</I18n> />
); );
} }
} }
@@ -176,4 +172,4 @@ OrganizationForm.contextTypes = {
}; };
export { OrganizationForm as _OrganizationForm }; export { OrganizationForm as _OrganizationForm };
export default withNetwork(withRouter(OrganizationForm)); export default withI18n()(withNetwork(withRouter(OrganizationForm)));

View File

@@ -1,6 +1,7 @@
import React from 'react'; import React from 'react';
import { string, bool, func } from 'prop-types'; import { string, bool, func } from 'prop-types';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
Badge as PFBadge, Badge as PFBadge,
DataListItem, DataListItem,
@@ -62,6 +63,7 @@ class OrganizationListItem extends React.Component {
isSelected, isSelected,
onSelect, onSelect,
detailUrl, detailUrl,
i18n
} = this.props; } = this.props;
const labelId = `check-action-${organization.id}`; const labelId = `check-action-${organization.id}`;
return ( return (
@@ -86,13 +88,15 @@ class OrganizationListItem extends React.Component {
</DataListCell>, </DataListCell>,
<DataListCell key="org-members" righthalf="true" width={2}> <DataListCell key="org-members" righthalf="true" width={2}>
<ListGroup> <ListGroup>
<Trans>Members</Trans> {i18n._(t`Members`)}
<Badge isRead> <Badge isRead>
{organization.summary_fields.related_field_counts.users} {organization.summary_fields.related_field_counts.users}
</Badge> </Badge>
</ListGroup> </ListGroup>
</DataListCell>,
<DataListCell key="teams">
<ListGroup> <ListGroup>
<Trans>Teams</Trans> {i18n._(t`Teams`)}
<Badge isRead> <Badge isRead>
{organization.summary_fields.related_field_counts.teams} {organization.summary_fields.related_field_counts.teams}
</Badge> </Badge>
@@ -105,4 +109,4 @@ class OrganizationListItem extends React.Component {
); );
} }
} }
export default OrganizationListItem; export default withI18n()(OrganizationListItem);

View File

@@ -1,5 +1,5 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import { I18n, i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { Switch, Route, withRouter, Redirect } from 'react-router-dom'; import { Switch, Route, withRouter, Redirect } from 'react-router-dom';
import { Card, CardHeader, PageSection } from '@patternfly/react-core'; import { Card, CardHeader, PageSection } from '@patternfly/react-core';
@@ -100,7 +100,8 @@ class Organization extends Component {
location, location,
match, match,
me, me,
history history,
i18n
} = this.props; } = this.props;
const { const {
@@ -126,14 +127,14 @@ class Organization extends Component {
); );
const tabsArray = [ const tabsArray = [
{ name: i18nMark('Details'), link: `${match.url}/details`, id: 0 }, { name: i18n._(t`Details`), link: `${match.url}/details`, id: 0 },
{ name: i18nMark('Access'), link: `${match.url}/access`, id: 1 }, { name: i18n._(t`Access`), link: `${match.url}/access`, id: 1 },
{ name: i18nMark('Teams'), link: `${match.url}/teams`, id: 2 } { name: i18n._(t`Teams`), link: `${match.url}/teams`, id: 2 }
]; ];
if (canSeeNotificationsTab) { if (canSeeNotificationsTab) {
tabsArray.push({ tabsArray.push({
name: i18nMark('Notifications'), name: i18n._(t`Notifications`),
link: `${match.url}/notifications`, link: `${match.url}/notifications`,
id: 3 id: 3
}); });
@@ -145,24 +146,18 @@ class Organization extends Component {
<CardHeader <CardHeader
style={tabsStyle} style={tabsStyle}
> >
<I18n> <div className="awx-orgTabs-container">
{({ i18n }) => ( <RoutedTabs
<React.Fragment> match={match}
<div className="awx-orgTabs-container"> history={history}
<RoutedTabs labeltext={i18n._(t`Organization detail tabs`)}
match={match} tabsArray={tabsArray}
history={history} />
labeltext={i18n._(t`Organization detail tabs`)} <CardCloseButton linkTo="/organizations" />
tabsArray={tabsArray} <div
/> className="awx-orgTabs__bottom-border"
<CardCloseButton linkTo="/organizations" /> />
<div </div>
className="awx-orgTabs__bottom-border"
/>
</div>
</React.Fragment>
)}
</I18n>
</CardHeader> </CardHeader>
)); ));
if (!match) { if (!match) {
@@ -245,5 +240,5 @@ class Organization extends Component {
); );
} }
} }
export default withNetwork(withRouter(Organization)); export default withI18n()(withNetwork(withRouter(Organization)));
export { Organization as _Organization }; export { Organization as _Organization };

View File

@@ -1,6 +1,7 @@
import React, { Fragment } from 'react'; import React, { Fragment } from 'react';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import PaginatedDataList, { ToolbarAddButton } from '../../../../components/PaginatedDataList'; import PaginatedDataList, { ToolbarAddButton } from '../../../../components/PaginatedDataList';
import OrganizationAccessItem from '../../components/OrganizationAccessItem'; import OrganizationAccessItem from '../../components/OrganizationAccessItem';
import DeleteRoleConfirmationModal from '../../components/DeleteRoleConfirmationModal'; import DeleteRoleConfirmationModal from '../../components/DeleteRoleConfirmationModal';
@@ -130,7 +131,7 @@ class OrganizationAccess extends React.Component {
} }
render () { render () {
const { api, organization } = this.props; const { organization, i18n } = this.props;
const { const {
isLoading, isLoading,
isInitialized, isInitialized,
@@ -167,9 +168,9 @@ class OrganizationAccess extends React.Component {
itemName="role" itemName="role"
qsConfig={QS_CONFIG} qsConfig={QS_CONFIG}
toolbarColumns={[ toolbarColumns={[
{ name: i18nMark('Name'), key: 'first_name', isSortable: true }, { name: i18n._(t`Name`), key: 'first_name', isSortable: true },
{ name: i18nMark('Username'), key: 'username', isSortable: true }, { name: i18n._(t`Username`), key: 'username', isSortable: true },
{ name: i18nMark('Last Name'), key: 'last_name', isSortable: true }, { name: i18n._(t`Last Name`), key: 'last_name', isSortable: true },
]} ]}
additionalControls={canEdit ? [ additionalControls={canEdit ? [
<ToolbarAddButton key="add" onClick={this.toggleAddModal} /> <ToolbarAddButton key="add" onClick={this.toggleAddModal} />
@@ -187,7 +188,6 @@ class OrganizationAccess extends React.Component {
<AddResourceRole <AddResourceRole
onClose={this.toggleAddModal} onClose={this.toggleAddModal}
onSave={this.handleSuccessfulRoleAdd} onSave={this.handleSuccessfulRoleAdd}
api={api}
roles={organization.summary_fields.object_roles} roles={organization.summary_fields.object_roles}
/> />
)} )}
@@ -197,4 +197,4 @@ class OrganizationAccess extends React.Component {
} }
export { OrganizationAccess as _OrganizationAccess }; export { OrganizationAccess as _OrganizationAccess };
export default withNetwork(withRouter(OrganizationAccess)); export default withI18n()(withNetwork(withRouter(OrganizationAccess)));

View File

@@ -1,7 +1,7 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import { Link, withRouter } from 'react-router-dom'; import { Link, withRouter } from 'react-router-dom';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { Trans, t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
CardBody, CardBody,
@@ -103,7 +103,8 @@ class OrganizationDetail extends Component {
modified, modified,
summary_fields summary_fields
}, },
match match,
i18n
} = this.props; } = this.props;
const showOverflowChipAfter = 5; const showOverflowChipAfter = 5;
@@ -127,58 +128,54 @@ class OrganizationDetail extends Component {
) : null; ) : null;
return ( return (
<I18n> <CardBody>
{({ i18n }) => ( <div className="pf-l-grid pf-m-gutter pf-m-all-12-col-on-md pf-m-all-6-col-on-lg pf-m-all-4-col-on-xl">
<CardBody> <Detail
<div className="pf-l-grid pf-m-gutter pf-m-all-12-col-on-md pf-m-all-6-col-on-lg pf-m-all-4-col-on-xl"> label={i18n._(t`Name`)}
<Detail value={name}
label={i18n._(t`Name`)} />
value={name} <Detail
/> label={i18n._(t`Description`)}
<Detail value={description}
label={i18n._(t`Description`)} />
value={description} <Detail
/> label={i18n._(t`Ansible Environment`)}
<Detail value={custom_virtualenv}
label={i18n._(t`Ansible Environment`)} />
value={custom_virtualenv} <Detail
/> label={i18n._(t`Created`)}
<Detail value={created}
label={i18n._(t`Created`)} />
value={created} <Detail
/> label={i18n._(t`Last Modified`)}
<Detail value={modified}
label={i18n._(t`Last Modified`)} />
value={modified} {(instanceGroups && instanceGroups.length > 0) && (
/> <TextContent style={{ display: 'flex', gridColumn: '1 / -1' }}>
{(instanceGroups && instanceGroups.length > 0) && ( <Text
<TextContent style={{ display: 'flex', gridColumn: '1 / -1' }}> component={TextVariants.h6}
<Text style={detailLabelStyle}
component={TextVariants.h6} >
style={detailLabelStyle} {i18n._(t`Instance Groups`)}
> </Text>
<Trans>Instance Groups</Trans> <div style={detailValueStyle}>
</Text> {instanceGroupChips}
<div style={detailValueStyle}> {overflowChip}
{instanceGroupChips}
{overflowChip}
</div>
</TextContent>
)}
</div>
{summary_fields.user_capabilities.edit && (
<div style={{ display: 'flex', flexDirection: 'row-reverse', marginTop: '20px' }}>
<Link to={`/organizations/${match.params.id}/edit`}>
<Button><Trans>Edit</Trans></Button>
</Link>
</div> </div>
)} </TextContent>
{error ? 'error!' : ''} )}
</CardBody> </div>
{summary_fields.user_capabilities.edit && (
<div style={{ display: 'flex', flexDirection: 'row-reverse', marginTop: '20px' }}>
<Link to={`/organizations/${match.params.id}/edit`}>
<Button>{i18n._(t`Edit`)}</Button>
</Link>
</div>
)} )}
</I18n> {error ? 'error!' : ''}
</CardBody>
); );
} }
} }
export default withRouter(withNetwork(OrganizationDetail)); export default withI18n()(withRouter(withNetwork(OrganizationDetail)));

View File

@@ -1,7 +1,7 @@
import React from 'react'; import React from 'react';
import PropTypes from 'prop-types'; import PropTypes from 'prop-types';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { I18n } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro'; import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
@@ -57,30 +57,27 @@ class OrganizationAdd extends React.Component {
render () { render () {
const { error } = this.state; const { error } = this.state;
const { i18n } = this.props;
return ( return (
<PageSection> <PageSection>
<I18n> <Card>
{({ i18n }) => ( <CardHeader className="at-u-textRight">
<Card> <Tooltip
<CardHeader className="at-u-textRight"> content={i18n._(t`Close`)}
<Tooltip position="top"
content={i18n._(t`Close`)} >
position="top" <CardCloseButton onClick={this.handleCancel} />
> </Tooltip>
<CardCloseButton onClick={this.handleCancel} /> </CardHeader>
</Tooltip> <CardBody>
</CardHeader> <OrganizationForm
<CardBody> handleSubmit={this.handleSubmit}
<OrganizationForm handleCancel={this.handleCancel}
handleSubmit={this.handleSubmit} />
handleCancel={this.handleCancel} {error ? <div>error</div> : ''}
/> </CardBody>
{error ? <div>error</div> : ''} </Card>
</CardBody>
</Card>
)}
</I18n>
</PageSection> </PageSection>
); );
} }
@@ -95,4 +92,4 @@ OrganizationAdd.contextTypes = {
}; };
export { OrganizationAdd as _OrganizationAdd }; export { OrganizationAdd as _OrganizationAdd };
export default withNetwork(withRouter(OrganizationAdd)); export default withI18n()(withNetwork(withRouter(OrganizationAdd)));

View File

@@ -1,6 +1,7 @@
import React, { Component } from 'react'; import React, { Component } from 'react';
import { withRouter } from 'react-router-dom'; import { withRouter } from 'react-router-dom';
import { i18nMark } from '@lingui/react'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
Card, Card,
PageSection, PageSection,
@@ -15,12 +16,6 @@ import PaginatedDataList, {
import OrganizationListItem from '../components/OrganizationListItem'; import OrganizationListItem from '../components/OrganizationListItem';
import { getQSConfig, parseNamespacedQueryString } from '../../../util/qs'; import { getQSConfig, parseNamespacedQueryString } from '../../../util/qs';
const COLUMNS = [
{ name: i18nMark('Name'), key: 'name', isSortable: true },
{ name: i18nMark('Modified'), key: 'modified', isSortable: true, isNumeric: true },
{ name: i18nMark('Created'), key: 'created', isSortable: true, isNumeric: true },
];
const QS_CONFIG = getQSConfig('organization', { const QS_CONFIG = getQSConfig('organization', {
page: 1, page: 1,
page_size: 5, page_size: 5,
@@ -36,7 +31,7 @@ class OrganizationsList extends Component {
isLoading: true, isLoading: true,
isInitialized: false, isInitialized: false,
organizations: [], organizations: [],
selected: [], selected: []
}; };
this.handleSelectAll = this.handleSelectAll.bind(this); this.handleSelectAll = this.handleSelectAll.bind(this);
@@ -148,9 +143,9 @@ class OrganizationsList extends Component {
isLoading, isLoading,
isInitialized, isInitialized,
selected, selected,
organizations, organizations
} = this.state; } = this.state;
const { match } = this.props; const { match, i18n } = this.props;
const isAllSelected = selected.length === organizations.length; const isAllSelected = selected.length === organizations.length;
@@ -163,7 +158,11 @@ class OrganizationsList extends Component {
itemCount={itemCount} itemCount={itemCount}
itemName="organization" itemName="organization"
qsConfig={QS_CONFIG} qsConfig={QS_CONFIG}
toolbarColumns={COLUMNS} toolbarColumns={[
{ name: i18n._(t`Name`), key: 'name', isSortable: true },
{ name: i18n._(t`Modified`), key: 'modified', isSortable: true, isNumeric: true },
{ name: i18n._(t`Created`), key: 'created', isSortable: true, isNumeric: true },
]}
showSelectAll showSelectAll
isAllSelected={isAllSelected} isAllSelected={isAllSelected}
onSelectAll={this.handleSelectAll} onSelectAll={this.handleSelectAll}
@@ -198,4 +197,4 @@ class OrganizationsList extends Component {
} }
export { OrganizationsList as _OrganizationsList }; export { OrganizationsList as _OrganizationsList };
export default withNetwork(withRouter(OrganizationsList)); export default withI18n()(withNetwork(withRouter(OrganizationsList)));

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Portal extends Component { class Portal extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>My View</Trans> {i18n._(t`My View`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Portal extends Component {
} }
} }
export default Portal; export default withI18n()(Portal);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Projects extends Component { class Projects extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Projects</Trans> {i18n._(t`Projects`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Projects extends Component {
} }
} }
export default Projects; export default withI18n()(Projects);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Schedules extends Component { class Schedules extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Schedules</Trans> {i18n._(t`Schedules`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Schedules extends Component {
} }
} }
export default Schedules; export default withI18n()(Schedules);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class SystemSettings extends Component { class SystemSettings extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>System Settings</Trans> {i18n._(t`System Settings`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class SystemSettings extends Component {
} }
} }
export default SystemSettings; export default withI18n()(SystemSettings);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Teams extends Component { class Teams extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Teams</Trans> {i18n._(t`Teams`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Teams extends Component {
} }
} }
export default Teams; export default withI18n()(Teams);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Templates extends Component { class Templates extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Templates</Trans> {i18n._(t`Templates`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Templates extends Component {
} }
} }
export default Templates; export default withI18n()(Templates);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class UISettings extends Component { class UISettings extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>User Interface Settings</Trans> {i18n._(t`User Interface Settings`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class UISettings extends Component {
} }
} }
export default UISettings; export default withI18n()(UISettings);

View File

@@ -1,5 +1,6 @@
import React, { Component, Fragment } from 'react'; import React, { Component, Fragment } from 'react';
import { Trans } from '@lingui/macro'; import { withI18n } from '@lingui/react';
import { t } from '@lingui/macro';
import { import {
PageSection, PageSection,
PageSectionVariants, PageSectionVariants,
@@ -8,13 +9,14 @@ import {
class Users extends Component { class Users extends Component {
render () { render () {
const { i18n } = this.props;
const { light, medium } = PageSectionVariants; const { light, medium } = PageSectionVariants;
return ( return (
<Fragment> <Fragment>
<PageSection variant={light} className="pf-m-condensed"> <PageSection variant={light} className="pf-m-condensed">
<Title size="2xl"> <Title size="2xl">
<Trans>Users</Trans> {i18n._(t`Users`)}
</Title> </Title>
</PageSection> </PageSection>
<PageSection variant={medium} /> <PageSection variant={medium} />
@@ -23,4 +25,4 @@ class Users extends Component {
} }
} }
export default Users; export default withI18n()(Users);

View File

@@ -1,20 +0,0 @@
import { i18nMark } from '@lingui/react';
export function required (message) {
return value => {
if (!value.trim()) {
return message || i18nMark('This field must not be blank');
}
return undefined;
};
}
export function maxLength (max) {
return value => {
if (value.trim().length
> max) {
return i18nMark(`This field must not exceed ${max} characters`);
}
return undefined;
};
}

20
src/util/validators.jsx Normal file
View File

@@ -0,0 +1,20 @@
import { t } from '@lingui/macro';
export function required (message, i18n) {
return value => {
if (!value.trim()) {
return message || i18n._(t`This field must not be blank`);
}
return undefined;
};
}
export function maxLength (max, i18n) {
return value => {
if (value.trim().length
> max) {
return i18n._(t`This field must not exceed ${max} characters`);
}
return undefined;
};
}