🔧 Refactoring ComicDetail and re-organizing folder structure
This commit is contained in:
@@ -5,12 +5,15 @@ import React, {
|
|||||||
ReactElement,
|
ReactElement,
|
||||||
useContext,
|
useContext,
|
||||||
} from "react";
|
} from "react";
|
||||||
import PropTypes from "prop-types";
|
|
||||||
import { useParams } from "react-router-dom";
|
import { useParams } from "react-router-dom";
|
||||||
import Card from "./Carda";
|
import Card from "./Carda";
|
||||||
import { ComicVineMatchPanel } from "./ComicDetail/ComicVineMatchPanel";
|
import { ComicVineMatchPanel } from "./ComicDetail/ComicVineMatchPanel";
|
||||||
import AcquisitionPanel from "./AcquisitionPanel";
|
import { VolumeInformation } from "./ComicDetail/Tabs/VolumeInformation";
|
||||||
import DownloadsPanel from "./DownloadsPanel";
|
import { ComicVineDetails } from "./ComicDetail/ComicVineDetails";
|
||||||
|
import { RawFileDetails } from "./ComicDetail/RawFileDetails";
|
||||||
|
import AcquisitionPanel from "./ComicDetail/AcquisitionPanel";
|
||||||
|
import DownloadsPanel from "./ComicDetail/DownloadsPanel";
|
||||||
import SlidingPane from "react-sliding-pane";
|
import SlidingPane from "react-sliding-pane";
|
||||||
import Select, { components } from "react-select";
|
import Select, { components } from "react-select";
|
||||||
|
|
||||||
@@ -24,8 +27,7 @@ import {
|
|||||||
getComicBookDetailById,
|
getComicBookDetailById,
|
||||||
extractComicArchive,
|
extractComicArchive,
|
||||||
} from "../actions/comicinfo.actions";
|
} from "../actions/comicinfo.actions";
|
||||||
import { detectIssueTypes } from "../shared/utils/tradepaperback.utils";
|
|
||||||
import dayjs from "dayjs";
|
|
||||||
const prettyBytes = require("pretty-bytes");
|
const prettyBytes = require("pretty-bytes");
|
||||||
import { DnD } from "./DnD";
|
import { DnD } from "./DnD";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
@@ -158,9 +160,7 @@ export const ComicDetail = ({}: ComicDetailProps): ReactElement => {
|
|||||||
}, [dispatch, comicBookDetailData]);
|
}, [dispatch, comicBookDetailData]);
|
||||||
|
|
||||||
const [active, setActive] = useState(1);
|
const [active, setActive] = useState(1);
|
||||||
const createDescriptionMarkup = (html) => {
|
|
||||||
return { __html: html };
|
|
||||||
};
|
|
||||||
const isComicBookMetadataAvailable =
|
const isComicBookMetadataAvailable =
|
||||||
comicBookDetailData.sourcedMetadata &&
|
comicBookDetailData.sourcedMetadata &&
|
||||||
!isUndefined(comicBookDetailData.sourcedMetadata.comicvine) &&
|
!isUndefined(comicBookDetailData.sourcedMetadata.comicvine) &&
|
||||||
@@ -173,59 +173,7 @@ export const ComicDetail = ({}: ComicDetailProps): ReactElement => {
|
|||||||
name: "Volume Information",
|
name: "Volume Information",
|
||||||
icon: <i className="fa-solid fa-layer-group"></i>,
|
icon: <i className="fa-solid fa-layer-group"></i>,
|
||||||
content: isComicBookMetadataAvailable ? (
|
content: isComicBookMetadataAvailable ? (
|
||||||
<div key={1}>
|
<VolumeInformation data={comicBookDetailData} key={1} />
|
||||||
<div className="columns">
|
|
||||||
<div className="column is-narrow">
|
|
||||||
<figure className="card-image">
|
|
||||||
<img
|
|
||||||
src={
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine
|
|
||||||
.volumeInformation.image.thumb_url
|
|
||||||
}
|
|
||||||
/>
|
|
||||||
</figure>
|
|
||||||
</div>
|
|
||||||
<div className="column is-4">
|
|
||||||
<dl>
|
|
||||||
<dt>
|
|
||||||
Is a part of{" "}
|
|
||||||
<span className="has-text-info">
|
|
||||||
{
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine
|
|
||||||
.volumeInformation.name
|
|
||||||
}
|
|
||||||
</span>
|
|
||||||
</dt>
|
|
||||||
<dd>
|
|
||||||
Published by
|
|
||||||
<span className="has-text-weight-semibold">
|
|
||||||
{" "}
|
|
||||||
{
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine
|
|
||||||
.volumeInformation.publisher.name
|
|
||||||
}
|
|
||||||
</span>
|
|
||||||
</dd>
|
|
||||||
<dd>
|
|
||||||
Total issues in this volume:
|
|
||||||
{
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine
|
|
||||||
.volumeInformation.count_of_issues
|
|
||||||
}
|
|
||||||
</dd>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="columns">
|
|
||||||
<div
|
|
||||||
className="column is-three-quarters"
|
|
||||||
dangerouslySetInnerHTML={createDescriptionMarkup(
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine.volumeInformation
|
|
||||||
.description,
|
|
||||||
)}
|
|
||||||
></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
) : null,
|
) : null,
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@@ -339,118 +287,6 @@ export const ComicDetail = ({}: ComicDetailProps): ReactElement => {
|
|||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
const RawFileDetails = (props) => (
|
|
||||||
<div className="content comic-detail">
|
|
||||||
<dl>
|
|
||||||
<dt>Raw File Details</dt>
|
|
||||||
<dd className="is-size-7">
|
|
||||||
{props.data.containedIn +
|
|
||||||
"/" +
|
|
||||||
props.data.name +
|
|
||||||
props.data.extension}
|
|
||||||
</dd>
|
|
||||||
<dd>
|
|
||||||
<div className="field is-grouped">
|
|
||||||
<div className="control">
|
|
||||||
<div className="tags has-addons">
|
|
||||||
<span className="tag">Size</span>
|
|
||||||
<span className="tag is-info is-light">
|
|
||||||
{prettyBytes(props.data.fileSize)}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div className="control">
|
|
||||||
<div className="tags has-addons">
|
|
||||||
<span className="tag">Extension</span>
|
|
||||||
<span className="tag is-primary is-light">
|
|
||||||
{props.data.extension}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</dd>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
const ComicVineDetails = (props) => (
|
|
||||||
<div className="content comic-detail">
|
|
||||||
<dl>
|
|
||||||
<dt>ComicVine Metadata</dt>
|
|
||||||
<dd className="is-size-7">
|
|
||||||
Last scraped on{" "}
|
|
||||||
{dayjs(props.updatedAt).format("MMM D YYYY [at] h:mm a")}
|
|
||||||
</dd>
|
|
||||||
<dd>
|
|
||||||
<h6>{props.data.name}</h6>
|
|
||||||
</dd>
|
|
||||||
<dd>{props.data.number}</dd>
|
|
||||||
<dd>
|
|
||||||
<div className="field is-grouped is-grouped-multiline">
|
|
||||||
{!isUndefined(
|
|
||||||
detectIssueTypes(
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine.volumeInformation
|
|
||||||
.description,
|
|
||||||
),
|
|
||||||
) ? (
|
|
||||||
<div className="control">
|
|
||||||
<div className="tags has-addons">
|
|
||||||
<span className="tag is-light">Detected Type</span>
|
|
||||||
<span className="tag is-warning">
|
|
||||||
{
|
|
||||||
detectIssueTypes(
|
|
||||||
comicBookDetailData.sourcedMetadata.comicvine
|
|
||||||
.volumeInformation.description,
|
|
||||||
).displayName
|
|
||||||
}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
) : (
|
|
||||||
<div className="control">
|
|
||||||
<div className="tags has-addons">
|
|
||||||
<span className="tag is-light">Type</span>
|
|
||||||
<span className="tag is-warning">
|
|
||||||
{props.data.resource_type}
|
|
||||||
</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
<div className="control">
|
|
||||||
<div className="tags has-addons">
|
|
||||||
<span className="tag is-light">ComicVine Issue ID</span>
|
|
||||||
<span className="tag is-success">{props.data.id}</span>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</dd>
|
|
||||||
</dl>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
|
|
||||||
ComicVineDetails.propTypes = {
|
|
||||||
updatedAt: PropTypes.string,
|
|
||||||
data: PropTypes.shape({
|
|
||||||
name: PropTypes.string,
|
|
||||||
number: PropTypes.string,
|
|
||||||
resource_type: PropTypes.string,
|
|
||||||
id: PropTypes.number,
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
RawFileDetails.propTypes = {
|
|
||||||
data: PropTypes.shape({
|
|
||||||
containedIn: PropTypes.string,
|
|
||||||
name: PropTypes.string,
|
|
||||||
fileSize: PropTypes.number,
|
|
||||||
path: PropTypes.string,
|
|
||||||
extension: PropTypes.string,
|
|
||||||
cover: PropTypes.shape({
|
|
||||||
filePath: PropTypes.string,
|
|
||||||
}),
|
|
||||||
}),
|
|
||||||
};
|
|
||||||
|
|
||||||
// Determine which cover image to use:
|
// Determine which cover image to use:
|
||||||
// 1. from the locally imported or
|
// 1. from the locally imported or
|
||||||
// 2. from the CV-scraped version
|
// 2. from the CV-scraped version
|
||||||
@@ -506,7 +342,6 @@ export const ComicDetail = ({}: ComicDetailProps): ReactElement => {
|
|||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<section className="container">
|
<section className="container">
|
||||||
<div className="section">
|
<div className="section">
|
||||||
|
|||||||
@@ -9,12 +9,12 @@ import {
|
|||||||
search,
|
search,
|
||||||
downloadAirDCPPItem,
|
downloadAirDCPPItem,
|
||||||
getBundlesForComic,
|
getBundlesForComic,
|
||||||
} from "../actions/airdcpp.actions";
|
} from "../../actions/airdcpp.actions";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
import { RootState, SearchInstance } from "threetwo-ui-typings";
|
import { RootState, SearchInstance } from "threetwo-ui-typings";
|
||||||
import ellipsize from "ellipsize";
|
import ellipsize from "ellipsize";
|
||||||
import { isEmpty, isNil, map } from "lodash";
|
import { isEmpty, isNil, map } from "lodash";
|
||||||
import { AirDCPPSocketContext } from "../context/AirDCPPSocket";
|
import { AirDCPPSocketContext } from "../../context/AirDCPPSocket";
|
||||||
interface IAcquisitionPanelProps {
|
interface IAcquisitionPanelProps {
|
||||||
comicBookMetadata: any;
|
comicBookMetadata: any;
|
||||||
}
|
}
|
||||||
74
src/client/components/ComicDetail/ComicVineDetails.tsx
Normal file
74
src/client/components/ComicDetail/ComicVineDetails.tsx
Normal file
@@ -0,0 +1,74 @@
|
|||||||
|
import React, { ReactElement } from "react";
|
||||||
|
import PropTypes from "prop-types";
|
||||||
|
import { detectIssueTypes } from "../../shared/utils/tradepaperback.utils";
|
||||||
|
import dayjs from "dayjs";
|
||||||
|
import { isUndefined } from "lodash";
|
||||||
|
|
||||||
|
export const ComicVineDetails = (props): ReactElement => {
|
||||||
|
const { data, updatedAt } = props;
|
||||||
|
return (
|
||||||
|
<div className="content comic-detail">
|
||||||
|
<dl>
|
||||||
|
<dt>ComicVine Metadata</dt>
|
||||||
|
<dd className="is-size-7">
|
||||||
|
Last scraped on {dayjs(updatedAt).format("MMM D YYYY [at] h:mm a")}
|
||||||
|
</dd>
|
||||||
|
<dd>
|
||||||
|
<h6>{data.name}</h6>
|
||||||
|
</dd>
|
||||||
|
{data.issue_number && (
|
||||||
|
<dd className="mb-2">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag is-light">Issue Number</span>
|
||||||
|
<span className="tag is-warning">{data.issue_number}</span>
|
||||||
|
</div>
|
||||||
|
</dd>
|
||||||
|
)}
|
||||||
|
<dd>
|
||||||
|
<div className="field is-grouped is-grouped-multiline">
|
||||||
|
{!isUndefined(
|
||||||
|
detectIssueTypes(data.volumeInformation.description),
|
||||||
|
) ? (
|
||||||
|
<div className="control">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag is-light">Detected Type</span>
|
||||||
|
<span className="tag is-warning">
|
||||||
|
{
|
||||||
|
detectIssueTypes(data.volumeInformation.description)
|
||||||
|
.displayName
|
||||||
|
}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<div className="control">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag is-light">Type</span>
|
||||||
|
<span className="tag is-warning">{data.resource_type}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
<div className="control">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag is-light">ComicVine Issue ID</span>
|
||||||
|
<span className="tag is-success">{data.id}</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ComicVineDetails;
|
||||||
|
|
||||||
|
ComicVineDetails.propTypes = {
|
||||||
|
updatedAt: PropTypes.string,
|
||||||
|
data: PropTypes.shape({
|
||||||
|
name: PropTypes.string,
|
||||||
|
number: PropTypes.string,
|
||||||
|
resource_type: PropTypes.string,
|
||||||
|
id: PropTypes.number,
|
||||||
|
}),
|
||||||
|
};
|
||||||
@@ -2,14 +2,14 @@ import React, { useEffect, useContext, ReactElement } from "react";
|
|||||||
import {
|
import {
|
||||||
getDownloadProgress,
|
getDownloadProgress,
|
||||||
getBundlesForComic,
|
getBundlesForComic,
|
||||||
} from "../actions/airdcpp.actions";
|
} from "../../actions/airdcpp.actions";
|
||||||
import { useDispatch, useSelector } from "react-redux";
|
import { useDispatch, useSelector } from "react-redux";
|
||||||
import { RootState } from "threetwo-ui-typings";
|
import { RootState } from "threetwo-ui-typings";
|
||||||
import { isEmpty, isNil, map } from "lodash";
|
import { isEmpty, isNil, map } from "lodash";
|
||||||
import prettyBytes from "pretty-bytes";
|
import prettyBytes from "pretty-bytes";
|
||||||
import dayjs from "dayjs";
|
import dayjs from "dayjs";
|
||||||
import ellipsize from "ellipsize";
|
import ellipsize from "ellipsize";
|
||||||
import { AirDCPPSocketContext } from "../context/AirDCPPSocket";
|
import { AirDCPPSocketContext } from "../../context/AirDCPPSocket";
|
||||||
|
|
||||||
interface IDownloadsPanelProps {
|
interface IDownloadsPanelProps {
|
||||||
data: any;
|
data: any;
|
||||||
52
src/client/components/ComicDetail/RawFileDetails.tsx
Normal file
52
src/client/components/ComicDetail/RawFileDetails.tsx
Normal file
@@ -0,0 +1,52 @@
|
|||||||
|
import React, { ReactElement } from "react";
|
||||||
|
import PropTypes from "prop-types";
|
||||||
|
import prettyBytes from "pretty-bytes";
|
||||||
|
|
||||||
|
export const RawFileDetails = (props): ReactElement => {
|
||||||
|
const { data } = props;
|
||||||
|
return (
|
||||||
|
<div className="content comic-detail">
|
||||||
|
<dl>
|
||||||
|
<dt>Raw File Details</dt>
|
||||||
|
<dd className="is-size-7">
|
||||||
|
{data.containedIn + "/" + data.name + data.extension}
|
||||||
|
</dd>
|
||||||
|
<dd>
|
||||||
|
<div className="field is-grouped">
|
||||||
|
<div className="control">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag">Size</span>
|
||||||
|
<span className="tag is-info is-light">
|
||||||
|
{prettyBytes(data.fileSize)}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="control">
|
||||||
|
<div className="tags has-addons">
|
||||||
|
<span className="tag">Extension</span>
|
||||||
|
<span className="tag is-primary is-light">
|
||||||
|
{data.extension}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default RawFileDetails;
|
||||||
|
|
||||||
|
RawFileDetails.propTypes = {
|
||||||
|
data: PropTypes.shape({
|
||||||
|
containedIn: PropTypes.string,
|
||||||
|
name: PropTypes.string,
|
||||||
|
fileSize: PropTypes.number,
|
||||||
|
path: PropTypes.string,
|
||||||
|
extension: PropTypes.string,
|
||||||
|
cover: PropTypes.shape({
|
||||||
|
filePath: PropTypes.string,
|
||||||
|
}),
|
||||||
|
}),
|
||||||
|
};
|
||||||
58
src/client/components/ComicDetail/Tabs/VolumeInformation.tsx
Normal file
58
src/client/components/ComicDetail/Tabs/VolumeInformation.tsx
Normal file
@@ -0,0 +1,58 @@
|
|||||||
|
import React, { ReactElement } from "react";
|
||||||
|
|
||||||
|
export const VolumeInformation = (props): ReactElement => {
|
||||||
|
const { data } = props;
|
||||||
|
const createDescriptionMarkup = (html) => {
|
||||||
|
return { __html: html };
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div key={1}>
|
||||||
|
<div className="columns">
|
||||||
|
<div className="column is-narrow">
|
||||||
|
<figure className="card-image">
|
||||||
|
<img
|
||||||
|
src={
|
||||||
|
data.sourcedMetadata.comicvine.volumeInformation.image.thumb_url
|
||||||
|
}
|
||||||
|
/>
|
||||||
|
</figure>
|
||||||
|
</div>
|
||||||
|
<div className="column is-4">
|
||||||
|
<dl>
|
||||||
|
<dt>
|
||||||
|
Is a part of{" "}
|
||||||
|
<span className="has-text-info">
|
||||||
|
{data.sourcedMetadata.comicvine.volumeInformation.name}
|
||||||
|
</span>
|
||||||
|
</dt>
|
||||||
|
<dd>
|
||||||
|
Published by
|
||||||
|
<span className="has-text-weight-semibold">
|
||||||
|
{" "}
|
||||||
|
{
|
||||||
|
data.sourcedMetadata.comicvine.volumeInformation.publisher
|
||||||
|
.name
|
||||||
|
}
|
||||||
|
</span>
|
||||||
|
</dd>
|
||||||
|
<dd>
|
||||||
|
Total issues in this volume:
|
||||||
|
{data.sourcedMetadata.comicvine.volumeInformation.count_of_issues}
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div className="columns">
|
||||||
|
<div
|
||||||
|
className="column is-three-quarters"
|
||||||
|
dangerouslySetInnerHTML={createDescriptionMarkup(
|
||||||
|
data.sourcedMetadata.comicvine.volumeInformation.description,
|
||||||
|
)}
|
||||||
|
></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default VolumeInformation;
|
||||||
Reference in New Issue
Block a user