Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactor AppDetailsPage to Macaw Next #3818

Merged
merged 4 commits into from
Jun 29, 2023
Merged
Show file tree
Hide file tree
Changes from 2 commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions .changeset/orange-falcons-judge.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"saleor-dashboard": patch
---

Refactored Manage App screen to use Macaw/next. Added missing empty-state messages, like missing permissions or app description.
17 changes: 13 additions & 4 deletions locale/defaultMessages.json
Original file line number Diff line number Diff line change
Expand Up @@ -1927,6 +1927,10 @@
"context": "dialog content",
"string": "Which address would you like to use as shipping address for selected customer:"
},
"CHoZ8S": {
"context": "app privacy policy link",
"string": "View this app’s privacy policy."
},
"CJEIRC": {
"string": "Product export has finished and was sent to your email address."
},
Expand Down Expand Up @@ -2597,10 +2601,6 @@
"context": "button",
"string": "Unassign"
},
"Go50v2": {
"context": "app privacy policy link",
"string": "View this app’s privacy policy"
},
"GpqEl5": {
"context": "shipping method description",
"string": "Description"
Expand Down Expand Up @@ -5444,6 +5444,9 @@
"b+jcaN": {
"string": "There are still fulfillments created for this order. Cancel the fulfillments first before you cancel the order."
},
"b088Xv": {
"string": "App doesn't provide a description."
},
"b1t9bM": {
"context": "empty headers text",
"string": "No custom request headers created for this webhook. Use the button below to add new custom request header."
Expand Down Expand Up @@ -5946,6 +5949,9 @@
"context": "attributes, section header",
"string": "Variant Attributes"
},
"f3hf+w": {
"string": "App doesn't provide a privacy policy."
},
"f91E8b": {
"context": "app repository",
"string": "Repository"
Expand Down Expand Up @@ -8231,6 +8237,9 @@
"context": "order refund amount",
"string": "Proposed refund amount"
},
"wDYozn": {
"string": "App doesn't have any permissions granted."
},
"wHdMAX": {
"context": "sale value, header",
"string": "Value"
Expand Down
38 changes: 26 additions & 12 deletions src/apps/components/AppDetailsPage/AboutCard.tsx
Original file line number Diff line number Diff line change
@@ -1,27 +1,41 @@
import CardTitle from "@dashboard/components/CardTitle";
import Skeleton from "@dashboard/components/Skeleton";
import { Card, CardContent } from "@material-ui/core";
import { Box, BoxProps, Text } from "@saleor/macaw-ui/next";
import React from "react";
import { useIntl } from "react-intl";
import ReactMarkdown from "react-markdown";

import messages from "./messages";

interface AboutCardProps {
type AboutCardProps = {
aboutApp?: string | null;
loading: boolean;
}
} & BoxProps;

const AboutCard: React.FC<AboutCardProps> = ({ aboutApp, loading }) => {
export const AboutCard: React.FC<AboutCardProps> = ({
aboutApp,
loading,
...boxProps
}) => {
const intl = useIntl();

const renderContent = () => {
if (loading) {
return <Skeleton />;
}

if (aboutApp) {
return <ReactMarkdown source={aboutApp ?? ""} />;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Suggested change
return <ReactMarkdown source={aboutApp ?? ""} />;
return <ReactMarkdown source={aboutApp} />;

} else {
return <Text>{intl.formatMessage(messages.noAboutApp)}</Text>;
}
};

return (
<Card>
<CardTitle title={intl.formatMessage(messages.aboutAppTitle)} />
<CardContent>
{!loading ? <ReactMarkdown source={aboutApp ?? ""} /> : <Skeleton />}
</CardContent>
</Card>
<Box {...boxProps}>
<Text variant={"heading"} as={"h2"} marginBottom={4}>
{intl.formatMessage(messages.aboutAppTitle)}
</Text>
<Box>{renderContent()}</Box>
</Box>
);
};
export default AboutCard;
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import { Meta, StoryObj } from "@storybook/react";

import { appDetails } from "../../fixtures";
import AppDetailsPage, { AppDetailsPageProps } from "./AppDetailsPage";
import { AppDetailsPage, AppDetailsPageProps } from "./AppDetailsPage";

const props: AppDetailsPageProps = {
data: appDetails,
Expand Down
66 changes: 41 additions & 25 deletions src/apps/components/AppDetailsPage/AppDetailsPage.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,7 @@ import { render } from "@testing-library/react";
import React from "react";

import { appDetails } from "../../fixtures";
import AppDetailsPage from "./AppDetailsPage";
import { AppDetailsPage } from "./AppDetailsPage";

const mockHeader = jest.fn();
jest.mock("./Header", () => props => {
Expand All @@ -12,22 +12,29 @@ jest.mock("./Header", () => props => {
});

const mockAboutCard = jest.fn();
jest.mock("./AboutCard", () => props => {
mockAboutCard(props);
return <></>;
});
jest.mock("./AboutCard", () => ({
AboutCard: props => {
mockAboutCard(props);
return <></>;
},
}));

const mockPermissionsCard = jest.fn();
jest.mock("./PermissionsCard", () => props => {
mockPermissionsCard(props);
return <></>;
});

jest.mock("./PermissionsCard", () => ({
PermissionsCard: props => {
mockPermissionsCard(props);
return <></>;
},
}));

const mockDataPrivacyCard = jest.fn();
jest.mock("./DataPrivacyCard", () => props => {
mockDataPrivacyCard(props);
return <></>;
});
jest.mock("./DataPrivacyCard", () => ({
DataPrivacyCard: props => {
mockDataPrivacyCard(props);
return <></>;
},
}));

beforeEach(() => {
mockHeader.mockClear();
Expand All @@ -36,6 +43,9 @@ beforeEach(() => {
mockDataPrivacyCard.mockClear();
});

/**
* TODO Rewrite tests to actually render the tree
*/
describe("Apps AppDetailsPage", () => {
it("displays app details when app data passed", () => {
// Arrange
Expand All @@ -61,17 +71,23 @@ describe("Apps AppDetailsPage", () => {
onAppDeactivateOpen,
onAppDeleteOpen,
});
expect(mockAboutCard).toHaveBeenCalledWith({
aboutApp: appDetails.aboutApp,
loading: false,
});
expect(mockPermissionsCard).toHaveBeenCalledWith({
permissions: appDetails.permissions,
loading: false,
});
expect(mockDataPrivacyCard).toHaveBeenCalledWith({
dataPrivacyUrl: appDetails.dataPrivacyUrl,
loading: false,
});
expect(mockAboutCard).toHaveBeenCalledWith(
expect.objectContaining({
aboutApp: appDetails.aboutApp,
loading: false,
}),
);
expect(mockPermissionsCard).toHaveBeenCalledWith(
expect.objectContaining({
permissions: appDetails.permissions,
loading: false,
}),
);
expect(mockDataPrivacyCard).toHaveBeenCalledWith(
expect.objectContaining({
dataPrivacyUrl: appDetails.dataPrivacyUrl,
loading: false,
}),
);
});
});
20 changes: 10 additions & 10 deletions src/apps/components/AppDetailsPage/AppDetailsPage.tsx
Original file line number Diff line number Diff line change
@@ -1,11 +1,10 @@
import CardSpacer from "@dashboard/components/CardSpacer";
import { AppQuery } from "@dashboard/graphql";
import React from "react";

import AboutCard from "./AboutCard";
import DataPrivacyCard from "./DataPrivacyCard";
import { AboutCard } from "./AboutCard";
import { DataPrivacyCard } from "./DataPrivacyCard";
import Header from "./Header";
import PermissionsCard from "./PermissionsCard";
import { PermissionsCard } from "./PermissionsCard";

export interface AppDetailsPageProps {
loading: boolean;
Expand Down Expand Up @@ -34,18 +33,19 @@ export const AppDetailsPage: React.FC<AppDetailsPageProps> = ({
onAppDeactivateOpen={onAppDeactivateOpen}
onAppDeleteOpen={onAppDeleteOpen}
/>
<AboutCard aboutApp={data?.aboutApp} loading={loading} />
<CardSpacer />
<PermissionsCard permissions={data?.permissions} loading={loading} />
<CardSpacer />
<AboutCard margin={6} aboutApp={data?.aboutApp} loading={loading} />
<PermissionsCard
margin={6}
permissions={data?.permissions}
loading={loading}
/>
<DataPrivacyCard
margin={6}
dataPrivacyUrl={data?.dataPrivacyUrl}
loading={loading}
/>
<CardSpacer />
</>
);
};

AppDetailsPage.displayName = "AppDetailsPage";
export default AppDetailsPage;
54 changes: 29 additions & 25 deletions src/apps/components/AppDetailsPage/DataPrivacyCard.tsx
Original file line number Diff line number Diff line change
@@ -1,47 +1,51 @@
// @ts-strict-ignore
import CardTitle from "@dashboard/components/CardTitle";
import ExternalLink from "@dashboard/components/ExternalLink";
import Skeleton from "@dashboard/components/Skeleton";
import { Card, CardContent } from "@material-ui/core";
import { Box, BoxProps, Text } from "@saleor/macaw-ui/next";
import React from "react";
import { FormattedMessage, useIntl } from "react-intl";

import messages from "./messages";
import { useStyles } from "./styles";

interface DataPrivacyCardProps {
type DataPrivacyCardProps = {
dataPrivacyUrl?: string | null;
loading: boolean;
}
} & BoxProps;

const DataPrivacyCard: React.FC<DataPrivacyCardProps> = ({
export const DataPrivacyCard: React.FC<DataPrivacyCardProps> = ({
dataPrivacyUrl,
loading,
...boxProps
}) => {
const classes = useStyles();
const intl = useIntl();

if (!dataPrivacyUrl && !loading) {
return null;
}

const renderContent = () => {
if (loading) {
return <Skeleton />;
}

if (dataPrivacyUrl) {
return (
<ExternalLink href={dataPrivacyUrl} target="_blank">
<FormattedMessage {...messages.dataPrivacyDescription} />
</ExternalLink>
);
} else {
return <Text>{intl.formatMessage(messages.noDataPrivacyPage)}</Text>;
}

throw new Error("Leaking switch statement");
Copy link
Member

@krzysztofwolski krzysztofwolski Jun 29, 2023

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Implementing return early is optional (non-blocking).
Throw instruction can be removed since it's unreachable.

Suggested change
if (dataPrivacyUrl) {
return (
<ExternalLink href={dataPrivacyUrl} target="_blank">
<FormattedMessage {...messages.dataPrivacyDescription} />
</ExternalLink>
);
} else {
return <Text>{intl.formatMessage(messages.noDataPrivacyPage)}</Text>;
}
throw new Error("Leaking switch statement");
if (!dataPrivacyUrl) {
return <Text>{intl.formatMessage(messages.noDataPrivacyPage)}</Text>
}
return (
<ExternalLink href={dataPrivacyUrl} target="_blank">
<FormattedMessage {...messages.dataPrivacyDescription} />
</ExternalLink>
);

Was this error necessary? Seems strange 🤔

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes. Component should render only for cases I want it to. If none of conditions are met, this is a bug in the code. Should not happen. Just like leaky switch/cases. Hence throwing error to quickly see that something wrong happened. If I return "null" here, we will never now if there is a bug

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That's if/else with returning statements inside. Exception throw is unreachable

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

suggestion: shouldn't it be logged into sentry?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Won't sentry automatically do that? Its unhandled on purpose

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I added error boundary and explicitely called Sentry

};

return (
<Card>
<CardTitle title={intl.formatMessage(messages.dataPrivacyTitle)} />
<CardContent>
{!loading ? (
<ExternalLink
className={classes.linkContainer}
href={dataPrivacyUrl}
target="_blank"
>
<FormattedMessage {...messages.dataPrivacyDescription} />
</ExternalLink>
) : (
<Skeleton />
)}
</CardContent>
</Card>
<Box {...boxProps}>
<Text variant={"heading"} marginBottom={4} as={"h2"}>
{intl.formatMessage(messages.dataPrivacyTitle)}
</Text>
<Box>{renderContent()}</Box>
</Box>
);
};
export default DataPrivacyCard;
Loading