foreman-tasks 1.1.3 → 1.2.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/app/controllers/foreman_tasks/api/tasks_controller.rb +29 -1
- data/app/controllers/foreman_tasks/tasks_controller.rb +3 -11
- data/app/models/foreman_tasks/task.rb +4 -0
- data/app/models/setting/foreman_tasks.rb +1 -1
- data/app/services/ui_notifications/tasks/task_bulk_stop.rb +36 -0
- data/app/views/foreman_tasks/api/tasks/details.json.rabl +0 -1
- data/config/routes.rb +1 -0
- data/db/migrate/20200517215015_rename_bookmarks_controller.rb +35 -0
- data/db/migrate/20200519093217_drop_dynflow_allow_dangerous_actions_setting.foreman_tasks.rb +5 -0
- data/db/seeds.d/30-notification_blueprints.rb +7 -0
- data/lib/foreman_tasks/engine.rb +1 -1
- data/lib/foreman_tasks/tasks/cleanup.rake +2 -2
- data/lib/foreman_tasks/tasks/export_tasks.rake +1 -1
- data/lib/foreman_tasks/version.rb +1 -1
- data/locale/action_names.rb +1 -1
- data/locale/en/foreman_tasks.po +227 -41
- data/locale/foreman_tasks.pot +579 -288
- data/webpack/ForemanTasks/Components/TaskActions/TaskAction.test.js +60 -0
- data/webpack/ForemanTasks/Components/{TasksTable/TasksTableActionHelpers.js → TaskActions/TaskActionHelpers.js} +21 -6
- data/webpack/ForemanTasks/Components/{TasksTable/__tests__/TasksTableActionHelpers.test.js → TaskActions/TaskActionHelpers.test.js} +2 -2
- data/webpack/ForemanTasks/Components/TaskActions/TaskActionsConstants.js +16 -0
- data/webpack/ForemanTasks/Components/TaskActions/UnlockModals.js +60 -0
- data/webpack/ForemanTasks/Components/TaskActions/UnlockModals.test.js +14 -0
- data/webpack/ForemanTasks/Components/TaskActions/__snapshots__/TaskAction.test.js.snap +233 -0
- data/webpack/ForemanTasks/Components/TaskActions/__snapshots__/UnlockModals.test.js.snap +25 -0
- data/webpack/ForemanTasks/Components/TaskActions/index.js +115 -0
- data/webpack/ForemanTasks/Components/TaskDetails/Components/Task.js +130 -152
- data/webpack/ForemanTasks/Components/TaskDetails/Components/__tests__/Task.test.js +67 -3
- data/webpack/ForemanTasks/Components/TaskDetails/Components/__tests__/TaskInfo.test.js +0 -1
- data/webpack/ForemanTasks/Components/TaskDetails/Components/__tests__/__snapshots__/Task.test.js.snap +101 -70
- data/webpack/ForemanTasks/Components/TaskDetails/TaskDetailsActions.js +0 -15
- data/webpack/ForemanTasks/Components/TaskDetails/TaskDetailsConstants.js +0 -5
- data/webpack/ForemanTasks/Components/TaskDetails/TaskDetailsReducer.js +0 -6
- data/webpack/ForemanTasks/Components/TaskDetails/TaskDetailsSelectors.js +0 -9
- data/webpack/ForemanTasks/Components/TaskDetails/TasksDetailsHelper.js +6 -1
- data/webpack/ForemanTasks/Components/TaskDetails/__tests__/TaskDetails.test.js +5 -0
- data/webpack/ForemanTasks/Components/TaskDetails/__tests__/__snapshots__/TaskDetails.test.js.snap +2 -7
- data/webpack/ForemanTasks/Components/TaskDetails/__tests__/integration.test.js +4 -0
- data/webpack/ForemanTasks/Components/TaskDetails/index.js +4 -8
- data/webpack/ForemanTasks/Components/TasksTable/Components/ActionSelectButton.js +14 -1
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/ConfirmModal.js +83 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/ConfirmModalActions.js +106 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/ConfirmModalReducer.js +38 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/ConfirmModalSelectors.js +45 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/ConfirmModal.test.js +36 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/ConfirmModalActions.test.js +205 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/ConfirmModalReducer.test.js +27 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/ConfirmModalSelectors.test.js +54 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/__snapshots__/ConfirmModal.test.js.snap +41 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/__snapshots__/ConfirmModalReducer.test.js.snap +19 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/__test__/__snapshots__/ConfirmModalSelectors.test.js.snap +30 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmModal/index.js +29 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/ActionSelectButton.test.js +1 -0
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/__snapshots__/ActionSelectButton.test.js.snap +11 -0
- data/webpack/ForemanTasks/Components/TasksTable/TasksBulkActions.js +113 -30
- data/webpack/ForemanTasks/Components/TasksTable/TasksTable.js +13 -9
- data/webpack/ForemanTasks/Components/TasksTable/TasksTableActions.js +23 -63
- data/webpack/ForemanTasks/Components/TasksTable/TasksTableConstants.js +10 -12
- data/webpack/ForemanTasks/Components/TasksTable/TasksTablePage.js +24 -94
- data/webpack/ForemanTasks/Components/TasksTable/TasksTableSchema.js +2 -2
- data/webpack/ForemanTasks/Components/TasksTable/TasksTableSelectors.js +7 -4
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/TasksBulkActions.test.js +35 -0
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/TasksTable.fixtures.js +2 -12
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/TasksTableActions.test.js +22 -26
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/__snapshots__/SubTasksPage.test.js.snap +2 -14
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/__snapshots__/TasksBulkActions.test.js.snap +107 -0
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/__snapshots__/TasksIndexPage.test.js.snap +2 -14
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/__snapshots__/TasksTableActions.test.js.snap +17 -124
- data/webpack/ForemanTasks/Components/TasksTable/__tests__/__snapshots__/TasksTablePage.test.js.snap +24 -128
- data/webpack/ForemanTasks/Components/TasksTable/index.js +2 -2
- data/webpack/ForemanTasks/Components/common/ActionButtons/ActionButton.js +47 -19
- data/webpack/ForemanTasks/Components/common/ActionButtons/ActionButton.test.js +61 -14
- data/webpack/ForemanTasks/Components/common/ActionButtons/__snapshots__/ActionButton.test.js.snap +80 -21
- data/webpack/ForemanTasks/Components/common/ClickConfirmation/ClickConfirmation.scss +9 -0
- data/webpack/ForemanTasks/Components/common/ClickConfirmation/ClickConfirmation.test.js +44 -0
- data/webpack/ForemanTasks/Components/common/ClickConfirmation/__snapshots__/ClickConfirmation.test.js.snap +52 -0
- data/webpack/ForemanTasks/Components/common/ClickConfirmation/index.js +59 -66
- data/webpack/ForemanTasks/Components/common/{ToastTypesConstants.js → ToastsHelpers/ToastTypesConstants.js} +0 -0
- data/webpack/ForemanTasks/Components/common/ToastsHelpers/index.js +15 -0
- data/webpack/ForemanTasks/ForemanTasksReducers.js +2 -0
- data/webpack/ForemanTasks/Routes/ForemanTasksRoutes.test.js +2 -1
- data/webpack/__mocks__/foremanReact/components/ForemanModal/ForemanModalHooks.js +2 -2
- data/webpack/__mocks__/foremanReact/components/ForemanModal/index.js +17 -3
- data/webpack/__mocks__/foremanReact/components/common/ActionButtons/ActionButtons.js +3 -0
- metadata +32 -22
- data/db/migrate/20200611090846_add_task_lock_index_on_resource_type_and_task_id.rb +0 -9
- data/lib/foreman_tasks/tasks/dynflow.rake +0 -6
- data/webpack/ForemanTasks/Components/TasksTable/Components/CancelConfirm.js +0 -53
- data/webpack/ForemanTasks/Components/TasksTable/Components/ConfirmationModals.js +0 -56
- data/webpack/ForemanTasks/Components/TasksTable/Components/ResumeConfirm.js +0 -52
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/CancelConfirm.test.js +0 -26
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/ConfirmationModals.test.js +0 -24
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/ResumeConfirm.test.js +0 -26
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/__snapshots__/CancelConfirm.test.js.snap +0 -65
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/__snapshots__/ConfirmationModals.test.js.snap +0 -30
- data/webpack/ForemanTasks/Components/TasksTable/Components/__test__/__snapshots__/ResumeConfirm.test.js.snap +0 -63
- data/webpack/ForemanTasks/Components/common/ActionButtons/CancelButton.js +0 -23
- data/webpack/ForemanTasks/Components/common/ActionButtons/CancelButton.test.js +0 -26
- data/webpack/ForemanTasks/Components/common/ActionButtons/ResumeButton.js +0 -23
- data/webpack/ForemanTasks/Components/common/ActionButtons/ResumeButton.test.js +0 -27
- data/webpack/ForemanTasks/Components/common/ActionButtons/__snapshots__/CancelButton.test.js.snap +0 -15
- data/webpack/ForemanTasks/Components/common/ActionButtons/__snapshots__/ResumeButton.test.js.snap +0 -15
@@ -13,9 +13,9 @@ import {
|
|
13
13
|
selectSort,
|
14
14
|
selectActionName,
|
15
15
|
selectSelectedRows,
|
16
|
-
selectClicked,
|
17
16
|
selectAllRowsSelected,
|
18
17
|
selectShowSelectAll,
|
18
|
+
selectModalID,
|
19
19
|
} from './TasksTableSelectors';
|
20
20
|
|
21
21
|
const mapStateToProps = state => ({
|
@@ -27,9 +27,9 @@ const mapStateToProps = state => ({
|
|
27
27
|
itemCount: selectItemCount(state),
|
28
28
|
actionName: selectActionName(state),
|
29
29
|
selectedRows: selectSelectedRows(state),
|
30
|
-
clicked: selectClicked(state),
|
31
30
|
allRowsSelected: selectAllRowsSelected(state),
|
32
31
|
showSelectAll: selectShowSelectAll(state),
|
32
|
+
modalID: selectModalID(state),
|
33
33
|
});
|
34
34
|
|
35
35
|
const mapDispatchToProps = dispatch =>
|
@@ -1,27 +1,53 @@
|
|
1
1
|
import React from 'react';
|
2
2
|
import PropTypes from 'prop-types';
|
3
|
-
import {
|
4
|
-
import {
|
3
|
+
import { translate as __ } from 'foremanReact/common/I18n';
|
4
|
+
import { ActionButtons } from 'foremanReact/components/common/ActionButtons/ActionButtons';
|
5
5
|
|
6
|
-
export const ActionButton = ({
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
6
|
+
export const ActionButton = ({
|
7
|
+
id,
|
8
|
+
name,
|
9
|
+
availableActions: { resumable, cancellable, stoppable },
|
10
|
+
taskActions,
|
11
|
+
}) => {
|
12
|
+
const buttons = [];
|
13
|
+
const isTitle = !(resumable || cancellable || stoppable);
|
14
|
+
const title = isTitle ? __('Task cannot be canceled') : undefined;
|
15
|
+
if (resumable) {
|
16
|
+
buttons.push({
|
17
|
+
title: __('Resume'),
|
18
|
+
action: {
|
19
|
+
disabled: !resumable,
|
20
|
+
onClick: () => taskActions.resumeTask(id, name),
|
21
|
+
id: `task-resume-button-${id}`,
|
22
|
+
},
|
23
|
+
});
|
24
|
+
}
|
25
|
+
if (cancellable || (!stoppable && !resumable)) {
|
26
|
+
// Cancel is the default button that should be shown if no task action can be done
|
27
|
+
buttons.push({
|
28
|
+
title: __('Cancel'),
|
29
|
+
action: {
|
30
|
+
disabled: !cancellable,
|
31
|
+
onClick: () => taskActions.cancelTask(id, name),
|
32
|
+
id: `task-cancel-button-${id}`,
|
33
|
+
},
|
34
|
+
});
|
35
|
+
}
|
36
|
+
|
37
|
+
if (stoppable) {
|
38
|
+
buttons.push({
|
39
|
+
title: __('Force Cancel'),
|
40
|
+
action: {
|
41
|
+
disabled: !stoppable,
|
42
|
+
onClick: () => taskActions.forceCancelTask(id, name),
|
43
|
+
id: `task-force-cancel-button-${id}`,
|
44
|
+
},
|
45
|
+
});
|
17
46
|
}
|
18
47
|
return (
|
19
|
-
<
|
20
|
-
|
21
|
-
|
22
|
-
disabled={!availableActions.cancellable}
|
23
|
-
onClick={taskActions.cancelTask}
|
24
|
-
/>
|
48
|
+
<span title={title}>
|
49
|
+
<ActionButtons buttons={buttons} />
|
50
|
+
</span>
|
25
51
|
);
|
26
52
|
};
|
27
53
|
|
@@ -31,9 +57,11 @@ ActionButton.propTypes = {
|
|
31
57
|
availableActions: PropTypes.shape({
|
32
58
|
cancellable: PropTypes.bool,
|
33
59
|
resumable: PropTypes.bool,
|
60
|
+
stoppable: PropTypes.bool,
|
34
61
|
}).isRequired,
|
35
62
|
taskActions: PropTypes.shape({
|
36
63
|
cancelTask: PropTypes.func,
|
37
64
|
resumeTask: PropTypes.func,
|
65
|
+
forceCancelTask: PropTypes.func,
|
38
66
|
}).isRequired,
|
39
67
|
};
|
@@ -1,17 +1,19 @@
|
|
1
|
-
import
|
1
|
+
import React from 'react';
|
2
|
+
import { testComponentSnapshotsWithFixtures, shallow } from '@theforeman/test';
|
2
3
|
|
3
4
|
import { ActionButton } from './ActionButton';
|
4
5
|
|
6
|
+
const resumeTask = jest.fn();
|
7
|
+
const cancelTask = jest.fn();
|
8
|
+
const forceCancelTask = jest.fn();
|
9
|
+
const taskActions = { resumeTask, cancelTask, forceCancelTask };
|
5
10
|
const fixtures = {
|
6
11
|
'render with cancellable true props': {
|
7
12
|
availableActions: {
|
8
13
|
cancellable: true,
|
9
14
|
resumable: false,
|
10
15
|
},
|
11
|
-
taskActions
|
12
|
-
cancel: jest.fn(),
|
13
|
-
resume: jest.fn(),
|
14
|
-
},
|
16
|
+
taskActions,
|
15
17
|
id: 'id',
|
16
18
|
name: 'some-name',
|
17
19
|
},
|
@@ -20,10 +22,16 @@ const fixtures = {
|
|
20
22
|
cancellable: false,
|
21
23
|
resumable: true,
|
22
24
|
},
|
23
|
-
taskActions
|
24
|
-
|
25
|
-
|
25
|
+
taskActions,
|
26
|
+
id: 'id',
|
27
|
+
name: 'some-name',
|
28
|
+
},
|
29
|
+
'render with stoppable and cancellable true props': {
|
30
|
+
availableActions: {
|
31
|
+
cancellable: true,
|
32
|
+
stoppable: true,
|
26
33
|
},
|
34
|
+
taskActions,
|
27
35
|
id: 'id',
|
28
36
|
name: 'some-name',
|
29
37
|
},
|
@@ -32,14 +40,53 @@ const fixtures = {
|
|
32
40
|
cancellable: false,
|
33
41
|
resumable: false,
|
34
42
|
},
|
35
|
-
taskActions
|
36
|
-
cancel: jest.fn(),
|
37
|
-
resume: jest.fn(),
|
38
|
-
},
|
43
|
+
taskActions,
|
39
44
|
id: 'id',
|
40
45
|
name: 'some-name',
|
41
46
|
},
|
42
47
|
};
|
43
48
|
|
44
|
-
describe('ActionButton', () =>
|
45
|
-
|
49
|
+
describe('ActionButton', () => {
|
50
|
+
describe('snapshot test', () =>
|
51
|
+
testComponentSnapshotsWithFixtures(ActionButton, fixtures));
|
52
|
+
describe('click test', () => {
|
53
|
+
const id = 'some-id';
|
54
|
+
const name = 'some-name';
|
55
|
+
it('cancel', () => {
|
56
|
+
const component = shallow(
|
57
|
+
<ActionButton
|
58
|
+
id={id}
|
59
|
+
name={name}
|
60
|
+
availableActions={{ cancellable: true }}
|
61
|
+
taskActions={taskActions}
|
62
|
+
/>
|
63
|
+
).children();
|
64
|
+
component.props().buttons[0].action.onClick();
|
65
|
+
expect(cancelTask).toHaveBeenCalledWith(id, name);
|
66
|
+
});
|
67
|
+
it('resume', () => {
|
68
|
+
const component = shallow(
|
69
|
+
<ActionButton
|
70
|
+
id={id}
|
71
|
+
name={name}
|
72
|
+
availableActions={{ resumable: true }}
|
73
|
+
taskActions={taskActions}
|
74
|
+
/>
|
75
|
+
).children();
|
76
|
+
component.props().buttons[0].action.onClick();
|
77
|
+
expect(resumeTask).toHaveBeenCalledWith(id, name);
|
78
|
+
});
|
79
|
+
it('force cancel', () => {
|
80
|
+
const component = shallow(
|
81
|
+
<ActionButton
|
82
|
+
id={id}
|
83
|
+
name={name}
|
84
|
+
availableActions={{ stoppable: true }}
|
85
|
+
taskActions={taskActions}
|
86
|
+
/>
|
87
|
+
).children();
|
88
|
+
component.props().buttons[0].action.onClick();
|
89
|
+
expect(cancelTask).toHaveBeenCalledWith(id, name);
|
90
|
+
});
|
91
|
+
});
|
92
|
+
});
|
data/webpack/ForemanTasks/Components/common/ActionButtons/__snapshots__/ActionButton.test.js.snap
CHANGED
@@ -1,28 +1,87 @@
|
|
1
1
|
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
2
2
|
|
3
|
-
exports[`ActionButton render with cancellable false props 1`] = `
|
4
|
-
<
|
5
|
-
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
3
|
+
exports[`ActionButton snapshot test render with cancellable false props 1`] = `
|
4
|
+
<span
|
5
|
+
title="Task cannot be canceled"
|
6
|
+
>
|
7
|
+
<ActionButtons
|
8
|
+
buttons={
|
9
|
+
Array [
|
10
|
+
Object {
|
11
|
+
"action": Object {
|
12
|
+
"disabled": true,
|
13
|
+
"id": "task-cancel-button-id",
|
14
|
+
"onClick": [Function],
|
15
|
+
},
|
16
|
+
"title": "Cancel",
|
17
|
+
},
|
18
|
+
]
|
19
|
+
}
|
20
|
+
/>
|
21
|
+
</span>
|
10
22
|
`;
|
11
23
|
|
12
|
-
exports[`ActionButton render with cancellable true props 1`] = `
|
13
|
-
<
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
24
|
+
exports[`ActionButton snapshot test render with cancellable true props 1`] = `
|
25
|
+
<span>
|
26
|
+
<ActionButtons
|
27
|
+
buttons={
|
28
|
+
Array [
|
29
|
+
Object {
|
30
|
+
"action": Object {
|
31
|
+
"disabled": false,
|
32
|
+
"id": "task-cancel-button-id",
|
33
|
+
"onClick": [Function],
|
34
|
+
},
|
35
|
+
"title": "Cancel",
|
36
|
+
},
|
37
|
+
]
|
38
|
+
}
|
39
|
+
/>
|
40
|
+
</span>
|
19
41
|
`;
|
20
42
|
|
21
|
-
exports[`ActionButton render with resumable true props 1`] = `
|
22
|
-
<
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
43
|
+
exports[`ActionButton snapshot test render with resumable true props 1`] = `
|
44
|
+
<span>
|
45
|
+
<ActionButtons
|
46
|
+
buttons={
|
47
|
+
Array [
|
48
|
+
Object {
|
49
|
+
"action": Object {
|
50
|
+
"disabled": false,
|
51
|
+
"id": "task-resume-button-id",
|
52
|
+
"onClick": [Function],
|
53
|
+
},
|
54
|
+
"title": "Resume",
|
55
|
+
},
|
56
|
+
]
|
57
|
+
}
|
58
|
+
/>
|
59
|
+
</span>
|
60
|
+
`;
|
61
|
+
|
62
|
+
exports[`ActionButton snapshot test render with stoppable and cancellable true props 1`] = `
|
63
|
+
<span>
|
64
|
+
<ActionButtons
|
65
|
+
buttons={
|
66
|
+
Array [
|
67
|
+
Object {
|
68
|
+
"action": Object {
|
69
|
+
"disabled": false,
|
70
|
+
"id": "task-cancel-button-id",
|
71
|
+
"onClick": [Function],
|
72
|
+
},
|
73
|
+
"title": "Cancel",
|
74
|
+
},
|
75
|
+
Object {
|
76
|
+
"action": Object {
|
77
|
+
"disabled": false,
|
78
|
+
"id": "task-force-cancel-button-id",
|
79
|
+
"onClick": [Function],
|
80
|
+
},
|
81
|
+
"title": "Force Cancel",
|
82
|
+
},
|
83
|
+
]
|
84
|
+
}
|
85
|
+
/>
|
86
|
+
</span>
|
28
87
|
`;
|
@@ -0,0 +1,44 @@
|
|
1
|
+
import React from 'react';
|
2
|
+
import { testComponentSnapshotsWithFixtures, mount } from '@theforeman/test';
|
3
|
+
import { useForemanModal } from 'foremanReact/components/ForemanModal/ForemanModalHooks';
|
4
|
+
import { ClickConfirmation } from './';
|
5
|
+
|
6
|
+
const fixtures = {
|
7
|
+
render: {
|
8
|
+
title: 'some-title',
|
9
|
+
confirmType: 'danger',
|
10
|
+
body: 'some-body',
|
11
|
+
confirmationMessage: 'some-message',
|
12
|
+
id: 'some-id',
|
13
|
+
confirmAction: 'some-confirm',
|
14
|
+
onClick: jest.fn(),
|
15
|
+
},
|
16
|
+
};
|
17
|
+
|
18
|
+
describe('ClickConfirmation', () => {
|
19
|
+
testComponentSnapshotsWithFixtures(ClickConfirmation, fixtures);
|
20
|
+
it('enable button on checkbox click', () => {
|
21
|
+
const component = mount(<ClickConfirmation {...fixtures.render} />);
|
22
|
+
const getButton = () => component.find('.confirm-button').at(0);
|
23
|
+
expect(getButton().props().disabled).toBeTruthy();
|
24
|
+
const checkbox = component.find('input').at(0);
|
25
|
+
checkbox.simulate('change', { target: { checked: true } });
|
26
|
+
expect(getButton().props().disabled).toBeFalsy();
|
27
|
+
});
|
28
|
+
|
29
|
+
it('click test', () => {
|
30
|
+
const setModalClosed = jest.fn();
|
31
|
+
useForemanModal.mockImplementation(id => ({
|
32
|
+
setModalClosed: () => setModalClosed(id),
|
33
|
+
}));
|
34
|
+
const onClick = jest.fn();
|
35
|
+
const props = { ...fixtures.render, onClick };
|
36
|
+
const component = mount(<ClickConfirmation {...props} />);
|
37
|
+
const getButton = () => component.find('.confirm-button').at(0);
|
38
|
+
const checkbox = component.find('input').at(0);
|
39
|
+
checkbox.simulate('change', { target: { checked: true } });
|
40
|
+
getButton().simulate('click');
|
41
|
+
expect(onClick).toBeCalled();
|
42
|
+
expect(setModalClosed).toBeCalledWith({ id: fixtures.render.id });
|
43
|
+
});
|
44
|
+
});
|
@@ -0,0 +1,52 @@
|
|
1
|
+
// Jest Snapshot v1, https://goo.gl/fbAQLP
|
2
|
+
|
3
|
+
exports[`ClickConfirmation render 1`] = `
|
4
|
+
<ForemanModal
|
5
|
+
id="some-id"
|
6
|
+
>
|
7
|
+
<Component>
|
8
|
+
<span
|
9
|
+
className="glyphicon glyphicon-exclamation-sign"
|
10
|
+
/>
|
11
|
+
some-title
|
12
|
+
</Component>
|
13
|
+
<span>
|
14
|
+
some-body
|
15
|
+
</span>
|
16
|
+
<div
|
17
|
+
className="confirmation-check"
|
18
|
+
>
|
19
|
+
<input
|
20
|
+
checked={false}
|
21
|
+
onChange={[Function]}
|
22
|
+
type="checkbox"
|
23
|
+
/>
|
24
|
+
<span>
|
25
|
+
some-message
|
26
|
+
</span>
|
27
|
+
</div>
|
28
|
+
<Component>
|
29
|
+
<Button
|
30
|
+
active={false}
|
31
|
+
block={false}
|
32
|
+
bsClass="btn"
|
33
|
+
bsStyle="danger"
|
34
|
+
className="confirm-button"
|
35
|
+
disabled={true}
|
36
|
+
onClick={[Function]}
|
37
|
+
>
|
38
|
+
some-confirm
|
39
|
+
</Button>
|
40
|
+
<Button
|
41
|
+
active={false}
|
42
|
+
block={false}
|
43
|
+
bsClass="btn"
|
44
|
+
bsStyle="default"
|
45
|
+
disabled={false}
|
46
|
+
onClick={[MockFunction]}
|
47
|
+
>
|
48
|
+
Cancel
|
49
|
+
</Button>
|
50
|
+
</Component>
|
51
|
+
</ForemanModal>
|
52
|
+
`;
|
@@ -1,82 +1,75 @@
|
|
1
|
-
import {
|
1
|
+
import React, { useState, useEffect } from 'react';
|
2
2
|
import PropTypes from 'prop-types';
|
3
|
-
import
|
3
|
+
import { Button } from 'patternfly-react';
|
4
|
+
import ForemanModal from 'foremanReact/components/ForemanModal';
|
4
5
|
import { translate as __ } from 'foremanReact/common/I18n';
|
6
|
+
import { useForemanModal } from 'foremanReact/components/ForemanModal/ForemanModalHooks';
|
7
|
+
import './ClickConfirmation.scss';
|
5
8
|
|
6
|
-
export
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
52
|
-
|
53
|
-
|
54
|
-
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
>
|
59
|
-
{confirmAction}
|
60
|
-
</Button>
|
61
|
-
</Modal.Footer>
|
62
|
-
</Modal>
|
63
|
-
);
|
64
|
-
}
|
65
|
-
}
|
9
|
+
export const ClickConfirmation = ({
|
10
|
+
title,
|
11
|
+
confirmType,
|
12
|
+
body,
|
13
|
+
confirmationMessage,
|
14
|
+
id,
|
15
|
+
confirmAction,
|
16
|
+
onClick,
|
17
|
+
}) => {
|
18
|
+
const [isConfirmed, setConfirm] = useState(false);
|
19
|
+
const { setModalClosed, modalOpen } = useForemanModal({
|
20
|
+
id,
|
21
|
+
});
|
22
|
+
useEffect(() => {
|
23
|
+
setConfirm(false);
|
24
|
+
}, [modalOpen]);
|
25
|
+
const icon = confirmType === 'warning' ? confirmType : 'exclamation';
|
26
|
+
|
27
|
+
return (
|
28
|
+
<ForemanModal id={id}>
|
29
|
+
<ForemanModal.Header>
|
30
|
+
<span className={`glyphicon glyphicon-${icon}-sign`} />
|
31
|
+
{` ${title}`}
|
32
|
+
</ForemanModal.Header>
|
33
|
+
<span>{body}</span>
|
34
|
+
<div className="confirmation-check">
|
35
|
+
<input
|
36
|
+
onChange={e => {
|
37
|
+
setConfirm(e.target.checked);
|
38
|
+
}}
|
39
|
+
checked={isConfirmed}
|
40
|
+
type="checkbox"
|
41
|
+
/>
|
42
|
+
<span>{` ${confirmationMessage}`}</span>
|
43
|
+
</div>
|
44
|
+
<ForemanModal.Footer>
|
45
|
+
<Button
|
46
|
+
className="confirm-button"
|
47
|
+
onClick={() => {
|
48
|
+
onClick();
|
49
|
+
setModalClosed();
|
50
|
+
}}
|
51
|
+
bsStyle={confirmType}
|
52
|
+
disabled={!isConfirmed}
|
53
|
+
>
|
54
|
+
{confirmAction}
|
55
|
+
</Button>
|
56
|
+
<Button onClick={setModalClosed}>{__('Cancel')}</Button>
|
57
|
+
</ForemanModal.Footer>
|
58
|
+
</ForemanModal>
|
59
|
+
);
|
60
|
+
};
|
66
61
|
|
67
62
|
ClickConfirmation.propTypes = {
|
68
|
-
showModal: PropTypes.bool,
|
69
63
|
title: PropTypes.string.isRequired,
|
70
64
|
body: PropTypes.string.isRequired,
|
71
65
|
confirmationMessage: PropTypes.string.isRequired,
|
72
66
|
confirmAction: PropTypes.string.isRequired,
|
73
|
-
|
67
|
+
onClick: PropTypes.func.isRequired,
|
74
68
|
confirmType: PropTypes.oneOf(['warning', 'danger']),
|
75
|
-
|
69
|
+
id: PropTypes.string.isRequired,
|
76
70
|
};
|
77
71
|
|
78
72
|
ClickConfirmation.defaultProps = {
|
79
|
-
showModal: false,
|
80
73
|
confirmType: 'warning',
|
81
74
|
};
|
82
75
|
|