saleor-dashboard/src/discounts/views/SaleDetails.tsx

589 lines
26 KiB
TypeScript
Raw Normal View History

2019-06-19 14:40:52 +00:00
import Button from "@material-ui/core/Button";
import DialogContentText from "@material-ui/core/DialogContentText";
2019-08-09 10:26:22 +00:00
import React from "react";
import { FormattedMessage, useIntl } from "react-intl";
2019-06-19 14:40:52 +00:00
import ActionDialog from "@saleor/components/ActionDialog";
import AssignCategoriesDialog from "@saleor/components/AssignCategoryDialog";
import AssignCollectionDialog from "@saleor/components/AssignCollectionDialog";
import AssignProductDialog from "@saleor/components/AssignProductDialog";
import { WindowTitle } from "@saleor/components/WindowTitle";
import useBulkActions from "@saleor/hooks/useBulkActions";
import useNavigator from "@saleor/hooks/useNavigator";
import useNotifier from "@saleor/hooks/useNotifier";
import usePaginator, {
createPaginationState
} from "@saleor/hooks/usePaginator";
import useShop from "@saleor/hooks/useShop";
import { commonMessages, sectionNames } from "@saleor/intl";
2019-06-19 14:40:52 +00:00
import { categoryUrl } from "../../categories/urls";
import { collectionUrl } from "../../collections/urls";
import { DEFAULT_INITIAL_SEARCH_DATA, PAGINATE_BY } from "../../config";
import SearchCategories from "../../containers/SearchCategories";
import SearchCollections from "../../containers/SearchCollections";
import SearchProducts from "../../containers/SearchProducts";
2019-10-11 09:33:56 +00:00
import { decimal, getMutationState, joinDateTime, maybe } from "../../misc";
2019-06-19 14:40:52 +00:00
import { productUrl } from "../../products/urls";
import { DiscountValueTypeEnum, SaleType } from "../../types/globalTypes";
import SaleDetailsPage, {
SaleDetailsPageTab
} from "../components/SaleDetailsPage";
import {
TypedSaleCataloguesAdd,
TypedSaleCataloguesRemove,
TypedSaleDelete,
TypedSaleUpdate
} from "../mutations";
import { TypedSaleDetails } from "../queries";
import { SaleCataloguesAdd } from "../types/SaleCataloguesAdd";
import { SaleCataloguesRemove } from "../types/SaleCataloguesRemove";
import { SaleDelete } from "../types/SaleDelete";
import { SaleUpdate } from "../types/SaleUpdate";
import {
saleListUrl,
saleUrl,
SaleUrlDialog,
SaleUrlQueryParams
} from "../urls";
interface SaleDetailsProps {
id: string;
params: SaleUrlQueryParams;
}
function discountValueTypeEnum(type: SaleType): DiscountValueTypeEnum {
return type.toString() === DiscountValueTypeEnum.FIXED
? DiscountValueTypeEnum.FIXED
: DiscountValueTypeEnum.PERCENTAGE;
}
export const SaleDetails: React.FC<SaleDetailsProps> = ({ id, params }) => {
2019-06-19 14:40:52 +00:00
const navigate = useNavigator();
const paginate = usePaginator();
const notify = useNotifier();
const shop = useShop();
const { isSelected, listElements, reset, toggle, toggleAll } = useBulkActions(
params.ids
);
const intl = useIntl();
2019-06-19 14:40:52 +00:00
const paginationState = createPaginationState(PAGINATE_BY, params);
const changeTab = (tab: SaleDetailsPageTab) => {
reset();
navigate(
saleUrl(id, {
activeTab: tab
})
);
};
const handleSaleDelete = (data: SaleDelete) => {
if (data.saleDelete.errors.length === 0) {
notify({
text: intl.formatMessage({
defaultMessage: "Removed sale"
2019-06-19 14:40:52 +00:00
})
});
navigate(saleListUrl(), true);
}
};
const handleSaleUpdate = (data: SaleUpdate) => {
if (data.saleUpdate.errors.length === 0) {
notify({
text: intl.formatMessage(commonMessages.savedChanges)
2019-06-19 14:40:52 +00:00
});
}
};
const closeModal = () =>
navigate(
saleUrl(id, {
...params,
action: undefined,
ids: undefined
}),
true
);
const openModal = (action: SaleUrlDialog, ids?: string[]) =>
navigate(
saleUrl(id, {
...params,
action,
ids
})
);
const handleCatalogueAdd = (data: SaleCataloguesAdd) => {
if (data.saleCataloguesAdd.errors.length === 0) {
closeModal();
}
};
const handleCatalogueRemove = (data: SaleCataloguesRemove) => {
if (data.saleCataloguesRemove.errors.length === 0) {
closeModal();
reset();
}
};
const canOpenBulkActionDialog = maybe(() => params.ids.length > 0);
2019-06-19 14:40:52 +00:00
return (
<TypedSaleCataloguesRemove onCompleted={handleCatalogueRemove}>
{(saleCataloguesRemove, saleCataloguesRemoveOpts) => (
<TypedSaleCataloguesAdd onCompleted={handleCatalogueAdd}>
{(saleCataloguesAdd, saleCataloguesAddOpts) => (
<TypedSaleUpdate onCompleted={handleSaleUpdate}>
{(saleUpdate, saleUpdateOpts) => (
<TypedSaleDelete onCompleted={handleSaleDelete}>
{(saleDelete, saleDeleteOpts) => (
<TypedSaleDetails
displayLoader
variables={{ id, ...paginationState }}
>
{({ data, loading }) => {
const tabPageInfo =
params.activeTab === SaleDetailsPageTab.categories
? maybe(() => data.sale.categories.pageInfo)
: params.activeTab ===
SaleDetailsPageTab.collections
? maybe(() => data.sale.collections.pageInfo)
: maybe(() => data.sale.products.pageInfo);
const formTransitionState = getMutationState(
saleUpdateOpts.called,
saleUpdateOpts.loading,
maybe(() => saleUpdateOpts.data.saleUpdate.errors)
);
const assignTransitionState = getMutationState(
saleCataloguesAddOpts.called,
saleCataloguesAddOpts.loading,
maybe(
() =>
saleCataloguesAddOpts.data.saleCataloguesAdd
.errors
)
);
const unassignTransitionState = getMutationState(
saleCataloguesRemoveOpts.called,
saleCataloguesRemoveOpts.loading,
maybe(
() =>
saleCataloguesRemoveOpts.data.saleCataloguesRemove
.errors
)
);
const removeTransitionState = getMutationState(
saleDeleteOpts.called,
saleDeleteOpts.loading,
maybe(() => saleDeleteOpts.data.saleDelete.errors)
);
const handleCategoriesUnassign = (ids: string[]) =>
saleCataloguesRemove({
variables: {
...paginationState,
id,
input: {
categories: ids
}
}
});
const handleCollectionsUnassign = (ids: string[]) =>
saleCataloguesRemove({
variables: {
...paginationState,
id,
input: {
collections: ids
}
}
});
const handleProductsUnassign = (ids: string[]) =>
saleCataloguesRemove({
variables: {
...paginationState,
id,
input: {
products: ids
}
}
});
const {
loadNextPage,
loadPreviousPage,
pageInfo
} = paginate(tabPageInfo, paginationState, params);
return (
<>
<WindowTitle
title={intl.formatMessage(sectionNames.sales)}
/>
2019-06-19 14:40:52 +00:00
<SaleDetailsPage
defaultCurrency={maybe(
() => shop.defaultCurrency
)}
sale={maybe(() => data.sale)}
disabled={
loading || saleCataloguesRemoveOpts.loading
}
errors={maybe(
() => saleUpdateOpts.data.saleUpdate.errors
)}
pageInfo={pageInfo}
onNextPage={loadNextPage}
onPreviousPage={loadPreviousPage}
onCategoryAssign={() =>
openModal("assign-category")
}
onCategoryClick={id => () =>
navigate(categoryUrl(id))}
onCollectionAssign={() =>
openModal("assign-collection")
}
onCollectionUnassign={collectionId =>
handleCollectionsUnassign([collectionId])
}
onCategoryUnassign={categoryId =>
handleCategoriesUnassign([categoryId])
}
onCollectionClick={id => () =>
navigate(collectionUrl(id))}
onProductAssign={() =>
openModal("assign-product")
}
onProductUnassign={productId =>
handleProductsUnassign([productId])
}
onProductClick={id => () =>
navigate(productUrl(id))}
activeTab={params.activeTab}
onBack={() => navigate(saleListUrl())}
onTabClick={changeTab}
onSubmit={formData =>
saleUpdate({
variables: {
id,
input: {
2019-10-10 13:33:50 +00:00
endDate: formData.hasEndDate
? joinDateTime(
formData.endDate,
formData.endTime
)
: null,
2019-06-19 14:40:52 +00:00
name: formData.name,
2019-10-10 13:33:50 +00:00
startDate: joinDateTime(
formData.startDate,
formData.startTime
),
2019-06-19 14:40:52 +00:00
type: discountValueTypeEnum(
formData.type
),
value: decimal(formData.value)
}
}
})
}
onRemove={() => openModal("remove")}
saveButtonBarState={formTransitionState}
categoryListToolbar={
<Button
color="primary"
onClick={() =>
openModal("unassign-category", listElements)
}
>
<FormattedMessage
defaultMessage="Unassign"
description="unassign category from sale, button"
id="saleDetailsUnassignCategory"
/>
2019-06-19 14:40:52 +00:00
</Button>
}
collectionListToolbar={
<Button
color="primary"
onClick={() =>
openModal(
"unassign-collection",
listElements
)
}
>
<FormattedMessage
defaultMessage="Unassign"
description="unassign collection from sale, button"
id="saleDetailsUnassignCollection"
/>
2019-06-19 14:40:52 +00:00
</Button>
}
productListToolbar={
<Button
color="primary"
onClick={() =>
openModal("unassign-product", listElements)
}
>
<FormattedMessage
defaultMessage="Unassign"
description="unassign product from sale, button"
id="saleDetailsUnassignProduct"
/>
2019-06-19 14:40:52 +00:00
</Button>
}
isChecked={isSelected}
selected={listElements.length}
toggle={toggle}
toggleAll={toggleAll}
/>
<SearchProducts
variables={DEFAULT_INITIAL_SEARCH_DATA}
>
{({
search: searchProducts,
result: searchProductsOpts
}) => (
<AssignProductDialog
confirmButtonState={assignTransitionState}
open={params.action === "assign-product"}
onFetch={searchProducts}
loading={searchProductsOpts.loading}
onClose={closeModal}
2019-08-09 11:14:35 +00:00
onSubmit={products =>
2019-06-19 14:40:52 +00:00
saleCataloguesAdd({
variables: {
...paginationState,
id,
input: {
2019-08-09 11:14:35 +00:00
products: products.map(
2019-06-19 14:40:52 +00:00
product => product.id
)
}
}
})
}
products={maybe(() =>
2019-10-15 12:17:35 +00:00
searchProductsOpts.data.search.edges
2019-06-19 14:40:52 +00:00
.map(edge => edge.node)
.filter(
suggestedProduct => suggestedProduct.id
)
)}
/>
)}
</SearchProducts>
<SearchCategories
variables={DEFAULT_INITIAL_SEARCH_DATA}
>
{({
search: searchCategories,
result: searchCategoriesOpts
}) => (
<AssignCategoriesDialog
categories={maybe(() =>
2019-10-15 12:17:35 +00:00
searchCategoriesOpts.data.search.edges
2019-06-19 14:40:52 +00:00
.map(edge => edge.node)
.filter(
suggestedCategory =>
suggestedCategory.id
)
)}
confirmButtonState={assignTransitionState}
open={params.action === "assign-category"}
onFetch={searchCategories}
loading={searchCategoriesOpts.loading}
onClose={closeModal}
2019-08-09 11:14:35 +00:00
onSubmit={categories =>
2019-06-19 14:40:52 +00:00
saleCataloguesAdd({
variables: {
...paginationState,
id,
input: {
2019-08-09 11:14:35 +00:00
categories: categories.map(
2019-06-19 14:40:52 +00:00
product => product.id
)
}
}
})
}
/>
)}
</SearchCategories>
<SearchCollections
variables={DEFAULT_INITIAL_SEARCH_DATA}
>
{({
search: searchCollections,
result: searchCollectionsOpts
}) => (
<AssignCollectionDialog
collections={maybe(() =>
2019-10-15 12:17:35 +00:00
searchCollectionsOpts.data.search.edges
2019-06-19 14:40:52 +00:00
.map(edge => edge.node)
.filter(
suggestedCategory =>
suggestedCategory.id
)
)}
confirmButtonState={assignTransitionState}
open={params.action === "assign-collection"}
onFetch={searchCollections}
loading={searchCollectionsOpts.loading}
onClose={closeModal}
2019-08-09 11:14:35 +00:00
onSubmit={collections =>
2019-06-19 14:40:52 +00:00
saleCataloguesAdd({
variables: {
...paginationState,
id,
input: {
2019-08-09 11:14:35 +00:00
collections: collections.map(
2019-06-19 14:40:52 +00:00
product => product.id
)
}
}
})
}
/>
)}
</SearchCollections>
<ActionDialog
open={
params.action === "unassign-category" &&
canOpenBulkActionDialog
}
title={intl.formatMessage({
defaultMessage: "Unassign Categories From Sale",
description: "dialog header"
})}
2019-06-19 14:40:52 +00:00
confirmButtonState={unassignTransitionState}
onClose={closeModal}
onConfirm={() =>
handleCategoriesUnassign(params.ids)
}
>
{canOpenBulkActionDialog && (
<DialogContentText>
<FormattedMessage
2019-11-05 12:57:30 +00:00
defaultMessage="Are you sure you want to unassign {counter,plural,one{this category} other{{displayQuantity} categories}}?"
description="dialog content"
values={{
counter: params.ids.length,
displayQuantity: (
<strong>{params.ids.length}</strong>
2019-06-19 14:40:52 +00:00
)
}}
/>
</DialogContentText>
)}
2019-06-19 14:40:52 +00:00
</ActionDialog>
<ActionDialog
open={
params.action === "unassign-collection" &&
canOpenBulkActionDialog
}
title={intl.formatMessage({
defaultMessage:
"Unassign Collections From Sale",
description: "dialog header"
})}
2019-06-19 14:40:52 +00:00
confirmButtonState={unassignTransitionState}
onClose={closeModal}
onConfirm={() =>
handleCollectionsUnassign(params.ids)
}
>
{canOpenBulkActionDialog && (
<DialogContentText>
<FormattedMessage
2019-11-05 12:57:30 +00:00
defaultMessage="Are you sure you want to unassign {counter,plural,one{this collection} other{{displayQuantity} collections}}?"
description="dialog content"
values={{
counter: params.ids.length,
displayQuantity: (
<strong>{params.ids.length}</strong>
2019-06-19 14:40:52 +00:00
)
}}
/>
</DialogContentText>
)}
2019-06-19 14:40:52 +00:00
</ActionDialog>
<ActionDialog
open={
params.action === "unassign-product" &&
canOpenBulkActionDialog
}
title={intl.formatMessage({
defaultMessage: "Unassign Products From Sale",
description: "dialog header"
})}
2019-06-19 14:40:52 +00:00
confirmButtonState={unassignTransitionState}
onClose={closeModal}
onConfirm={() =>
handleProductsUnassign(params.ids)
}
>
{canOpenBulkActionDialog && (
<DialogContentText>
<FormattedMessage
2019-11-05 12:57:30 +00:00
defaultMessage="Are you sure you want to unassign {counter,plural,one{this product} other{{displayQuantity} products}}?"
description="dialog content"
values={{
counter: params.ids.length,
displayQuantity: (
<strong>{params.ids.length}</strong>
2019-06-19 14:40:52 +00:00
)
}}
/>
</DialogContentText>
)}
2019-06-19 14:40:52 +00:00
</ActionDialog>
<ActionDialog
open={params.action === "remove"}
title={intl.formatMessage({
defaultMessage: "Delete Sale",
description: "dialog header"
})}
2019-06-19 14:40:52 +00:00
confirmButtonState={removeTransitionState}
onClose={closeModal}
variant="delete"
onConfirm={() =>
saleDelete({
variables: { id }
})
}
>
<DialogContentText>
<FormattedMessage
defaultMessage="Are you sure you want to delete {saleName}?"
description="dialog content"
values={{
saleName: (
<strong>
{maybe(() => data.sale.name, "...")}
</strong>
)
}}
/>
</DialogContentText>
2019-06-19 14:40:52 +00:00
</ActionDialog>
</>
);
}}
</TypedSaleDetails>
)}
</TypedSaleDelete>
)}
</TypedSaleUpdate>
)}
</TypedSaleCataloguesAdd>
)}
</TypedSaleCataloguesRemove>
);
};
export default SaleDetails;