267 lines
9.3 KiB
JavaScript
267 lines
9.3 KiB
JavaScript
import { EditFilled, FileExcelFilled, SyncOutlined } from "@ant-design/icons";
|
|
import { Button, Card, Col, Row, Space } from "antd";
|
|
import axios from "axios";
|
|
import i18n from "i18next";
|
|
import { isFunction } from "lodash";
|
|
import { useCallback, useEffect, useState } from "react";
|
|
import { Gallery } from "react-grid-gallery";
|
|
import { useTranslation } from "react-i18next";
|
|
import Lightbox from "react-image-lightbox";
|
|
import "react-image-lightbox/style.css";
|
|
import { connect } from "react-redux";
|
|
import { createStructuredSelector } from "reselect";
|
|
import { selectBodyshop } from "../../redux/user/user.selectors";
|
|
import DocumentsUploadImgproxyComponent from "../documents-upload-imgproxy/documents-upload-imgproxy.component";
|
|
import { HasFeatureAccess } from "../feature-wrapper/feature-wrapper.component";
|
|
import UpsellComponent, { upsellEnum } from "../upsell/upsell.component";
|
|
import JobsDocumentsDownloadButton from "./jobs-document-imgproxy-gallery.download.component";
|
|
import JobsDocumentsGalleryReassign from "./jobs-document-imgproxy-gallery.reassign.component";
|
|
import JobsDocumentsDeleteButton from "./jobs-documents-imgproxy-gallery.delete.component";
|
|
import JobsDocumentsGallerySelectAllComponent from "./jobs-documents-imgproxy-gallery.selectall.component";
|
|
|
|
const mapStateToProps = createStructuredSelector({
|
|
bodyshop: selectBodyshop
|
|
});
|
|
const mapDispatchToProps = (dispatch) => ({});
|
|
/*
|
|
################################################################################################
|
|
Developer Note:
|
|
Known Technical Debt Item
|
|
Modifications to this code requires complementary changes to the Cloudinary code. Cloudinary code will be removed upon completed migration.
|
|
################################################################################################
|
|
*/
|
|
function JobsDocumentsImgproxyComponent({
|
|
bodyshop,
|
|
data,
|
|
jobId,
|
|
refetch,
|
|
billId,
|
|
billsCallback,
|
|
totalSize,
|
|
downloadIdentifier,
|
|
ignoreSizeLimit
|
|
}) {
|
|
const [galleryImages, setGalleryImages] = useState({ images: [], other: [] });
|
|
const { t } = useTranslation();
|
|
const [modalState, setModalState] = useState({ open: false, index: 0 });
|
|
|
|
const fetchThumbnails = useCallback(() => {
|
|
fetchImgproxyThumbnails({ setStateCallback: setGalleryImages, jobId });
|
|
}, [jobId, setGalleryImages]);
|
|
|
|
useEffect(() => {
|
|
if (data) {
|
|
fetchThumbnails();
|
|
}
|
|
}, [data, fetchThumbnails]);
|
|
|
|
const hasMediaAccess = HasFeatureAccess({ bodyshop, featureName: "media" });
|
|
const hasMobileAccess = HasFeatureAccess({ bodyshop, featureName: "mobile" });
|
|
return (
|
|
<div>
|
|
<Row gutter={[16, 16]}>
|
|
<Col span={24}>
|
|
<Space wrap>
|
|
<Button
|
|
onClick={() => {
|
|
//Handle any doc refresh.
|
|
|
|
isFunction(refetch) && refetch();
|
|
|
|
//Do the imgproxy refresh too
|
|
fetchThumbnails();
|
|
}}
|
|
>
|
|
<SyncOutlined />
|
|
</Button>
|
|
<JobsDocumentsGallerySelectAllComponent galleryImages={galleryImages} setGalleryImages={setGalleryImages} />
|
|
<JobsDocumentsDownloadButton galleryImages={galleryImages} identifier={downloadIdentifier} />
|
|
<JobsDocumentsDeleteButton
|
|
galleryImages={galleryImages}
|
|
deletionCallback={billsCallback || fetchThumbnails || refetch}
|
|
/>
|
|
{!billId && (
|
|
<JobsDocumentsGalleryReassign galleryImages={galleryImages} callback={fetchThumbnails || refetch} />
|
|
)}
|
|
</Space>
|
|
</Col>
|
|
{!hasMediaAccess && (
|
|
<Col span={24}>
|
|
<Card>
|
|
<UpsellComponent disableMask upsell={upsellEnum().media.general} />
|
|
</Card>
|
|
</Col>
|
|
)}
|
|
<Col span={24}>
|
|
<Card>
|
|
<DocumentsUploadImgproxyComponent
|
|
jobId={jobId}
|
|
totalSize={totalSize}
|
|
billId={billId}
|
|
callbackAfterUpload={billsCallback || fetchThumbnails || refetch}
|
|
ignoreSizeLimit={ignoreSizeLimit}
|
|
/>
|
|
</Card>
|
|
</Col>
|
|
{hasMediaAccess && !hasMobileAccess && (
|
|
<Col span={24}>
|
|
<Card>
|
|
<UpsellComponent upsell={upsellEnum().media.mobile} />
|
|
</Card>
|
|
</Col>
|
|
)}
|
|
<Col span={24}>
|
|
<Card title={t("jobs.labels.documents-images")}>
|
|
<Gallery
|
|
images={galleryImages.images}
|
|
onClick={(index, item) => {
|
|
setModalState({ open: true, index: index });
|
|
// window.open(
|
|
// item.fullsize,
|
|
// "_blank",
|
|
// "toolbar=0,location=0,menubar=0"
|
|
// );
|
|
}}
|
|
onSelect={(index, image) => {
|
|
setGalleryImages({
|
|
...galleryImages,
|
|
images: galleryImages.images.map((g, idx) =>
|
|
index === idx ? { ...g, isSelected: !g.isSelected } : g
|
|
)
|
|
});
|
|
}}
|
|
/>
|
|
</Card>
|
|
</Col>
|
|
<Col span={24}>
|
|
<Card title={t("jobs.labels.documents-other")}>
|
|
<Gallery
|
|
images={galleryImages.other}
|
|
thumbnailStyle={() => {
|
|
return {
|
|
backgroundImage: <FileExcelFilled />,
|
|
height: "100%",
|
|
width: "100%",
|
|
cursor: "pointer"
|
|
};
|
|
}}
|
|
onClick={(index) => {
|
|
window.open(galleryImages.other[index].source, "_blank", "toolbar=0,location=0,menubar=0");
|
|
}}
|
|
onSelect={(index) => {
|
|
setGalleryImages({
|
|
...galleryImages,
|
|
other: galleryImages.other.map((g, idx) => (index === idx ? { ...g, isSelected: !g.isSelected } : g))
|
|
});
|
|
}}
|
|
/>
|
|
</Card>
|
|
</Col>
|
|
{modalState.open && (
|
|
<Lightbox
|
|
toolbarButtons={[
|
|
<EditFilled
|
|
key="edit"
|
|
onClick={() => {
|
|
const newWindow = window.open(
|
|
`${window.location.protocol}//${window.location.host}/edit?documentId=${
|
|
galleryImages.images[modalState.index].id
|
|
}`,
|
|
"_blank",
|
|
"noopener,noreferrer"
|
|
);
|
|
if (newWindow) newWindow.opener = null;
|
|
}}
|
|
/>
|
|
]}
|
|
mainSrc={galleryImages.images[modalState.index].fullsize}
|
|
nextSrc={galleryImages.images[(modalState.index + 1) % galleryImages.images.length].fullsize}
|
|
prevSrc={
|
|
galleryImages.images[(modalState.index + galleryImages.images.length - 1) % galleryImages.images.length]
|
|
.fullsize
|
|
}
|
|
onCloseRequest={() => setModalState({ open: false, index: 0 })}
|
|
onMovePrevRequest={() =>
|
|
setModalState({
|
|
...modalState,
|
|
index: (modalState.index + galleryImages.images.length - 1) % galleryImages.images.length
|
|
})
|
|
}
|
|
onMoveNextRequest={() =>
|
|
setModalState({
|
|
...modalState,
|
|
index: (modalState.index + 1) % galleryImages.images.length
|
|
})
|
|
}
|
|
/>
|
|
)}
|
|
</Row>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default connect(mapStateToProps, mapDispatchToProps)(JobsDocumentsImgproxyComponent);
|
|
|
|
export const fetchImgproxyThumbnails = async ({ setStateCallback, jobId, imagesOnly }) => {
|
|
const result = await axios.post("/media/imgproxy/thumbnails", { jobid: jobId });
|
|
const documents = result.data.reduce(
|
|
(acc, value) => {
|
|
if (value.type.startsWith("image")) {
|
|
acc.images.push({
|
|
src: value.thumbnailUrl,
|
|
fullsize: value.originalUrl,
|
|
height: 225,
|
|
width: 225,
|
|
isSelected: false,
|
|
key: value.key,
|
|
extension: value.extension,
|
|
id: value.id,
|
|
type: value.type,
|
|
size: value.size,
|
|
tags: [{ value: value.type, title: value.type }]
|
|
});
|
|
} else {
|
|
const fileName = value.key.split("/").pop();
|
|
acc.other.push({
|
|
source: value.originalUrlViaProxyPath,
|
|
src: value.thumbnailUrl,
|
|
fullsize: value.presignedGetUrl,
|
|
tags: [
|
|
{
|
|
value: fileName,
|
|
title: fileName
|
|
},
|
|
|
|
{ value: value.type, title: value.type },
|
|
...(value.bill
|
|
? [
|
|
{
|
|
value: value.bill.vendor.name,
|
|
title: i18n.t("vendors.fields.name")
|
|
},
|
|
{ value: value.bill.date, title: i18n.t("bills.fields.date") },
|
|
{
|
|
value: value.bill.invoice_number,
|
|
title: i18n.t("bills.fields.invoice_number")
|
|
}
|
|
]
|
|
: [])
|
|
],
|
|
height: 225,
|
|
width: 225,
|
|
isSelected: false,
|
|
extension: value.extension,
|
|
key: value.key,
|
|
id: value.id,
|
|
type: value.type,
|
|
size: value.size
|
|
});
|
|
}
|
|
return acc;
|
|
},
|
|
{ images: [], other: [] }
|
|
);
|
|
|
|
setStateCallback(imagesOnly ? documents.images : documents);
|
|
};
|