mirror of
https://github.com/ansible/awx.git
synced 2026-02-25 23:16:01 -03:30
Flush out useDeleteItems hook
refactor TemplateList to use useRequest, useDeleteItems hooks refactor CredentialList and OrganizationList to use useDeleteItems hook
This commit is contained in:
@@ -54,6 +54,8 @@ function CredentialList({ i18n }) {
|
|||||||
fetchCredentials();
|
fetchCredentials();
|
||||||
}, [fetchCredentials]);
|
}, [fetchCredentials]);
|
||||||
|
|
||||||
|
const isAllSelected =
|
||||||
|
selected.length > 0 && selected.length === credentials.length;
|
||||||
const {
|
const {
|
||||||
isLoading: isDeleteLoading,
|
isLoading: isDeleteLoading,
|
||||||
deleteItems: deleteCredentials,
|
deleteItems: deleteCredentials,
|
||||||
@@ -65,8 +67,7 @@ function CredentialList({ i18n }) {
|
|||||||
}, [selected]),
|
}, [selected]),
|
||||||
{
|
{
|
||||||
qsConfig: QS_CONFIG,
|
qsConfig: QS_CONFIG,
|
||||||
items: credentials,
|
allItemsSelected: isAllSelected,
|
||||||
selected,
|
|
||||||
fetchItems: fetchCredentials,
|
fetchItems: fetchCredentials,
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
@@ -90,8 +91,6 @@ function CredentialList({ i18n }) {
|
|||||||
|
|
||||||
const canAdd =
|
const canAdd =
|
||||||
actions && Object.prototype.hasOwnProperty.call(actions, 'POST');
|
actions && Object.prototype.hasOwnProperty.call(actions, 'POST');
|
||||||
const isAllSelected =
|
|
||||||
selected.length > 0 && selected.length === credentials.length;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<PageSection>
|
<PageSection>
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import { t } from '@lingui/macro';
|
|||||||
import { Card, PageSection } from '@patternfly/react-core';
|
import { Card, PageSection } from '@patternfly/react-core';
|
||||||
|
|
||||||
import { OrganizationsAPI } from '@api';
|
import { OrganizationsAPI } from '@api';
|
||||||
import useRequest from '@util/useRequest';
|
import useRequest, { useDeleteItems } from '@util/useRequest';
|
||||||
import AlertModal from '@components/AlertModal';
|
import AlertModal from '@components/AlertModal';
|
||||||
import DataListToolbar from '@components/DataListToolbar';
|
import DataListToolbar from '@components/DataListToolbar';
|
||||||
import ErrorDetail from '@components/ErrorDetail';
|
import ErrorDetail from '@components/ErrorDetail';
|
||||||
@@ -14,7 +14,6 @@ import PaginatedDataList, {
|
|||||||
ToolbarDeleteButton,
|
ToolbarDeleteButton,
|
||||||
} from '@components/PaginatedDataList';
|
} from '@components/PaginatedDataList';
|
||||||
import { getQSConfig, parseQueryString } from '@util/qs';
|
import { getQSConfig, parseQueryString } from '@util/qs';
|
||||||
import updateUrlAfterDelete from '@util/updateUrlAfterDelete';
|
|
||||||
import OrganizationListItem from './OrganizationListItem';
|
import OrganizationListItem from './OrganizationListItem';
|
||||||
|
|
||||||
const QS_CONFIG = getQSConfig('organization', {
|
const QS_CONFIG = getQSConfig('organization', {
|
||||||
@@ -29,7 +28,6 @@ function OrganizationsList({ i18n }) {
|
|||||||
const match = useRouteMatch();
|
const match = useRouteMatch();
|
||||||
|
|
||||||
const [selected, setSelected] = useState([]);
|
const [selected, setSelected] = useState([]);
|
||||||
const [deletionError, setDeletionError] = useState(null);
|
|
||||||
|
|
||||||
const addUrl = `${match.url}/add`;
|
const addUrl = `${match.url}/add`;
|
||||||
|
|
||||||
@@ -58,48 +56,37 @@ function OrganizationsList({ i18n }) {
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const {
|
|
||||||
isLoading: isDeleteLoading,
|
|
||||||
error: dError,
|
|
||||||
request: deleteOrganizations,
|
|
||||||
} = useRequest(
|
|
||||||
useCallback(async () => {
|
|
||||||
return Promise.all(
|
|
||||||
selected.map(({ id }) => OrganizationsAPI.destroy(id))
|
|
||||||
);
|
|
||||||
}, [selected])
|
|
||||||
);
|
|
||||||
|
|
||||||
useEffect(() => {
|
|
||||||
if (dError) {
|
|
||||||
setDeletionError(dError);
|
|
||||||
}
|
|
||||||
}, [dError]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
fetchOrganizations();
|
fetchOrganizations();
|
||||||
}, [fetchOrganizations]);
|
}, [fetchOrganizations]);
|
||||||
|
|
||||||
|
const isAllSelected =
|
||||||
|
selected.length === organizations.length && selected.length > 0;
|
||||||
|
const {
|
||||||
|
isLoading: isDeleteLoading,
|
||||||
|
deleteItems: deleteOrganizations,
|
||||||
|
deletionError,
|
||||||
|
clearDeletionError,
|
||||||
|
} = useDeleteItems(
|
||||||
|
useCallback(async () => {
|
||||||
|
return Promise.all(
|
||||||
|
selected.map(({ id }) => OrganizationsAPI.destroy(id))
|
||||||
|
);
|
||||||
|
}, [selected]),
|
||||||
|
{
|
||||||
|
qsConfig: QS_CONFIG,
|
||||||
|
allItemsSelected: isAllSelected,
|
||||||
|
fetchItems: fetchOrganizations,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
const handleOrgDelete = async () => {
|
const handleOrgDelete = async () => {
|
||||||
await deleteOrganizations();
|
await deleteOrganizations();
|
||||||
const url = updateUrlAfterDelete(
|
|
||||||
QS_CONFIG,
|
|
||||||
location,
|
|
||||||
organizations,
|
|
||||||
selected
|
|
||||||
);
|
|
||||||
if (url) {
|
|
||||||
history.push(url);
|
|
||||||
} else {
|
|
||||||
fetchOrganizations();
|
|
||||||
}
|
|
||||||
setSelected([]);
|
setSelected([]);
|
||||||
};
|
};
|
||||||
|
|
||||||
const hasContentLoading = isDeleteLoading || isOrgsLoading;
|
const hasContentLoading = isDeleteLoading || isOrgsLoading;
|
||||||
const canAdd = actions && actions.POST;
|
const canAdd = actions && actions.POST;
|
||||||
const isAllSelected =
|
|
||||||
selected.length === organizations.length && selected.length > 0;
|
|
||||||
|
|
||||||
const handleSelectAll = isSelected => {
|
const handleSelectAll = isSelected => {
|
||||||
if (isSelected) {
|
if (isSelected) {
|
||||||
@@ -189,7 +176,7 @@ function OrganizationsList({ i18n }) {
|
|||||||
isOpen={deletionError}
|
isOpen={deletionError}
|
||||||
variant="danger"
|
variant="danger"
|
||||||
title={i18n._(t`Error!`)}
|
title={i18n._(t`Error!`)}
|
||||||
onClose={() => setDeletionError(null)}
|
onClose={clearDeletionError}
|
||||||
>
|
>
|
||||||
{i18n._(t`Failed to delete one or more organizations.`)}
|
{i18n._(t`Failed to delete one or more organizations.`)}
|
||||||
<ErrorDetail error={deletionError} />
|
<ErrorDetail error={deletionError} />
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState, useCallback } from 'react';
|
||||||
import { useParams, useLocation } from 'react-router-dom';
|
import { useParams, useLocation } from 'react-router-dom';
|
||||||
import { withI18n } from '@lingui/react';
|
import { withI18n } from '@lingui/react';
|
||||||
import { t } from '@lingui/macro';
|
import { t } from '@lingui/macro';
|
||||||
@@ -15,6 +15,7 @@ import ErrorDetail from '@components/ErrorDetail';
|
|||||||
import PaginatedDataList, {
|
import PaginatedDataList, {
|
||||||
ToolbarDeleteButton,
|
ToolbarDeleteButton,
|
||||||
} from '@components/PaginatedDataList';
|
} from '@components/PaginatedDataList';
|
||||||
|
import useRequest, { useDeleteItems } from '@util/useRequest';
|
||||||
import { getQSConfig, parseQueryString } from '@util/qs';
|
import { getQSConfig, parseQueryString } from '@util/qs';
|
||||||
|
|
||||||
import AddDropDownButton from '@components/AddDropDownButton';
|
import AddDropDownButton from '@components/AddDropDownButton';
|
||||||
@@ -31,79 +32,78 @@ const QS_CONFIG = getQSConfig('template', {
|
|||||||
|
|
||||||
function TemplateList({ i18n }) {
|
function TemplateList({ i18n }) {
|
||||||
const { id: projectId } = useParams();
|
const { id: projectId } = useParams();
|
||||||
const { pathname, search } = useLocation();
|
const location = useLocation();
|
||||||
|
|
||||||
const [deletionError, setDeletionError] = useState(null);
|
|
||||||
const [contentError, setContentError] = useState(null);
|
|
||||||
const [hasContentLoading, setHasContentLoading] = useState(true);
|
|
||||||
const [jtActions, setJTActions] = useState(null);
|
|
||||||
const [wfjtActions, setWFJTActions] = useState(null);
|
|
||||||
const [count, setCount] = useState(0);
|
|
||||||
const [templates, setTemplates] = useState([]);
|
|
||||||
const [selected, setSelected] = useState([]);
|
const [selected, setSelected] = useState([]);
|
||||||
|
|
||||||
useEffect(
|
const {
|
||||||
() => {
|
result: { templates, count, jtActions, wfjtActions },
|
||||||
const loadTemplates = async () => {
|
error: contentError,
|
||||||
const params = {
|
isLoading,
|
||||||
...parseQueryString(QS_CONFIG, search),
|
request: fetchTemplates,
|
||||||
};
|
} = useRequest(
|
||||||
|
useCallback(async () => {
|
||||||
let jtOptionsPromise;
|
const params = parseQueryString(QS_CONFIG, location.search);
|
||||||
if (jtActions) {
|
if (location.pathname.startsWith('/projects') && projectId) {
|
||||||
jtOptionsPromise = Promise.resolve({
|
params.jobtemplate__project = projectId;
|
||||||
data: { actions: jtActions },
|
}
|
||||||
});
|
const results = await Promise.all([
|
||||||
} else {
|
UnifiedJobTemplatesAPI.read(params),
|
||||||
jtOptionsPromise = JobTemplatesAPI.readOptions();
|
JobTemplatesAPI.readOptions(),
|
||||||
}
|
WorkflowJobTemplatesAPI.readOptions(),
|
||||||
|
]);
|
||||||
let wfjtOptionsPromise;
|
return {
|
||||||
if (wfjtActions) {
|
templates: results[0].data.results,
|
||||||
wfjtOptionsPromise = Promise.resolve({
|
count: results[0].data.count,
|
||||||
data: { actions: wfjtActions },
|
jtActions: results[1].data.actions,
|
||||||
});
|
wfjtActions: results[2].data.actions,
|
||||||
} else {
|
|
||||||
wfjtOptionsPromise = WorkflowJobTemplatesAPI.readOptions();
|
|
||||||
}
|
|
||||||
if (pathname.startsWith('/projects') && projectId) {
|
|
||||||
params.jobtemplate__project = projectId;
|
|
||||||
}
|
|
||||||
|
|
||||||
const promises = Promise.all([
|
|
||||||
UnifiedJobTemplatesAPI.read(params),
|
|
||||||
jtOptionsPromise,
|
|
||||||
wfjtOptionsPromise,
|
|
||||||
]);
|
|
||||||
setDeletionError(null);
|
|
||||||
|
|
||||||
try {
|
|
||||||
const [
|
|
||||||
{
|
|
||||||
data: { count: itemCount, results },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: { actions: jobTemplateActions },
|
|
||||||
},
|
|
||||||
{
|
|
||||||
data: { actions: workFlowJobTemplateActions },
|
|
||||||
},
|
|
||||||
] = await promises;
|
|
||||||
setJTActions(jobTemplateActions);
|
|
||||||
setWFJTActions(workFlowJobTemplateActions);
|
|
||||||
setCount(itemCount);
|
|
||||||
setTemplates(results);
|
|
||||||
setHasContentLoading(false);
|
|
||||||
} catch (err) {
|
|
||||||
setContentError(err);
|
|
||||||
}
|
|
||||||
};
|
};
|
||||||
loadTemplates();
|
}, [location, projectId]),
|
||||||
},
|
{
|
||||||
// eslint-disable-next-line react-hooks/exhaustive-deps
|
templates: [],
|
||||||
[pathname, search, count, projectId]
|
count: 0,
|
||||||
|
jtActions: {},
|
||||||
|
wfjtActions: {},
|
||||||
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
fetchTemplates();
|
||||||
|
}, [fetchTemplates]);
|
||||||
|
|
||||||
|
const isAllSelected =
|
||||||
|
selected.length === templates.length && selected.length > 0;
|
||||||
|
const {
|
||||||
|
isLoading: isDeleteLoading,
|
||||||
|
deleteItems: deleteTemplates,
|
||||||
|
deletionError,
|
||||||
|
clearDeletionError,
|
||||||
|
} = useDeleteItems(
|
||||||
|
useCallback(async () => {
|
||||||
|
return Promise.all(
|
||||||
|
selected.map(({ type, id }) => {
|
||||||
|
if (type === 'job_template') {
|
||||||
|
return JobTemplatesAPI.destroy(id);
|
||||||
|
}
|
||||||
|
if (type === 'workflow_job_template') {
|
||||||
|
return WorkflowJobTemplatesAPI.destroy(id);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}, [selected]),
|
||||||
|
{
|
||||||
|
qsConfig: QS_CONFIG,
|
||||||
|
allItemsSelected: isAllSelected,
|
||||||
|
fetchItems: fetchTemplates,
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
const handleTemplateDelete = async () => {
|
||||||
|
await deleteTemplates();
|
||||||
|
setSelected([]);
|
||||||
|
};
|
||||||
|
|
||||||
const handleSelectAll = isSelected => {
|
const handleSelectAll = isSelected => {
|
||||||
const selectedItems = isSelected ? [...templates] : [];
|
const selectedItems = isSelected ? [...templates] : [];
|
||||||
setSelected(selectedItems);
|
setSelected(selectedItems);
|
||||||
@@ -117,26 +117,6 @@ function TemplateList({ i18n }) {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleTemplateDelete = async () => {
|
|
||||||
setHasContentLoading(true);
|
|
||||||
try {
|
|
||||||
await Promise.all(
|
|
||||||
selected.map(({ type, id }) => {
|
|
||||||
let deletePromise;
|
|
||||||
if (type === 'job_template') {
|
|
||||||
deletePromise = JobTemplatesAPI.destroy(id);
|
|
||||||
} else if (type === 'workflow_job_template') {
|
|
||||||
deletePromise = WorkflowJobTemplatesAPI.destroy(id);
|
|
||||||
}
|
|
||||||
return deletePromise;
|
|
||||||
})
|
|
||||||
);
|
|
||||||
setCount(count - selected.length);
|
|
||||||
} catch (err) {
|
|
||||||
setDeletionError(err);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const canAddJT =
|
const canAddJT =
|
||||||
jtActions && Object.prototype.hasOwnProperty.call(jtActions, 'POST');
|
jtActions && Object.prototype.hasOwnProperty.call(jtActions, 'POST');
|
||||||
const canAddWFJT =
|
const canAddWFJT =
|
||||||
@@ -154,8 +134,6 @@ function TemplateList({ i18n }) {
|
|||||||
url: `/templates/workflow_job_template/add/`,
|
url: `/templates/workflow_job_template/add/`,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
const isAllSelected =
|
|
||||||
selected.length === templates.length && selected.length > 0;
|
|
||||||
const addButton = (
|
const addButton = (
|
||||||
<AddDropDownButton key="add" dropdownItems={addButtonOptions} />
|
<AddDropDownButton key="add" dropdownItems={addButtonOptions} />
|
||||||
);
|
);
|
||||||
@@ -164,7 +142,7 @@ function TemplateList({ i18n }) {
|
|||||||
<Card>
|
<Card>
|
||||||
<PaginatedDataList
|
<PaginatedDataList
|
||||||
contentError={contentError}
|
contentError={contentError}
|
||||||
hasContentLoading={hasContentLoading}
|
hasContentLoading={isLoading || isDeleteLoading}
|
||||||
items={templates}
|
items={templates}
|
||||||
itemCount={count}
|
itemCount={count}
|
||||||
pluralizedItemName={i18n._(t`Templates`)}
|
pluralizedItemName={i18n._(t`Templates`)}
|
||||||
@@ -247,7 +225,7 @@ function TemplateList({ i18n }) {
|
|||||||
key={template.id}
|
key={template.id}
|
||||||
value={template.name}
|
value={template.name}
|
||||||
template={template}
|
template={template}
|
||||||
detailUrl={`${pathname}/${template.type}/${template.id}`}
|
detailUrl={`${location.pathname}/${template.type}/${template.id}`}
|
||||||
onSelect={() => handleSelect(template)}
|
onSelect={() => handleSelect(template)}
|
||||||
isSelected={selected.some(row => row.id === template.id)}
|
isSelected={selected.some(row => row.id === template.id)}
|
||||||
/>
|
/>
|
||||||
@@ -259,7 +237,7 @@ function TemplateList({ i18n }) {
|
|||||||
isOpen={deletionError}
|
isOpen={deletionError}
|
||||||
variant="danger"
|
variant="danger"
|
||||||
title={i18n._(t`Error!`)}
|
title={i18n._(t`Error!`)}
|
||||||
onClose={() => setDeletionError(null)}
|
onClose={clearDeletionError}
|
||||||
>
|
>
|
||||||
{i18n._(t`Failed to delete one or more templates.`)}
|
{i18n._(t`Failed to delete one or more templates.`)}
|
||||||
<ErrorDetail error={deletionError} />
|
<ErrorDetail error={deletionError} />
|
||||||
|
|||||||
@@ -1,22 +0,0 @@
|
|||||||
import {
|
|
||||||
parseQueryString,
|
|
||||||
replaceParams,
|
|
||||||
encodeNonDefaultQueryString,
|
|
||||||
} from './qs';
|
|
||||||
|
|
||||||
export default function updateUrlAfterDelete(
|
|
||||||
qsConfig,
|
|
||||||
location,
|
|
||||||
items,
|
|
||||||
selectedItems
|
|
||||||
) {
|
|
||||||
const params = parseQueryString(qsConfig, location.search);
|
|
||||||
if (params.page > 1 && selectedItems.length === items.length) {
|
|
||||||
const newParams = encodeNonDefaultQueryString(
|
|
||||||
qsConfig,
|
|
||||||
replaceParams(params, { page: params.page - 1 })
|
|
||||||
);
|
|
||||||
return `${location.pathname}?${newParams}`;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
@@ -56,18 +56,23 @@ export default function useRequest(makeRequest, initialValue) {
|
|||||||
|
|
||||||
export function useDeleteItems(
|
export function useDeleteItems(
|
||||||
makeRequest,
|
makeRequest,
|
||||||
{ qsConfig, items, selected, fetchItems }
|
{ qsConfig, allItemsSelected, fetchItems }
|
||||||
) {
|
) {
|
||||||
const location = useLocation();
|
const location = useLocation();
|
||||||
const history = useHistory();
|
const history = useHistory();
|
||||||
const [showError, setShowError] = useState(false);
|
const [showError, setShowError] = useState(false);
|
||||||
|
|
||||||
const { error, isLoading, request } = useRequest(makeRequest, null);
|
const { error, isLoading, request } = useRequest(makeRequest, null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (error) {
|
||||||
|
setShowError(true);
|
||||||
|
}
|
||||||
|
}, [error]);
|
||||||
|
|
||||||
const deleteItems = async () => {
|
const deleteItems = async () => {
|
||||||
await request();
|
await request();
|
||||||
const params = parseQueryString(qsConfig, location.search);
|
const params = parseQueryString(qsConfig, location.search);
|
||||||
if (params.page > 1 && selected.length === items.length) {
|
if (params.page > 1 && allItemsSelected) {
|
||||||
const newParams = encodeNonDefaultQueryString(
|
const newParams = encodeNonDefaultQueryString(
|
||||||
qsConfig,
|
qsConfig,
|
||||||
replaceParams(params, { page: params.page - 1 })
|
replaceParams(params, { page: params.page - 1 })
|
||||||
|
|||||||
Reference in New Issue
Block a user