Switch from ReactJSS to styled-components in ui-components

This commit is contained in:
Florian Scholdei
2019-10-08 16:42:08 +02:00
parent 7ec1a8dd01
commit 1b6392defc
19 changed files with 422 additions and 566 deletions

View File

@@ -1,5 +1,8 @@
//@flow
import React from "react";
import { translate } from "react-i18next";
import classNames from "classnames";
import styled from "styled-components";
import {
Change,
Diff as DiffComponent,
@@ -8,71 +11,14 @@ import {
getChangeKey,
Hunk
} from "react-diff-view";
import injectSheets from "react-jss";
import classNames from "classnames";
import { translate } from "react-i18next";
import { Button, ButtonGroup } from "../buttons";
import Tag from "../Tag";
const styles = {
panel: {
fontSize: "1rem"
},
/* breaks into a second row
when buttons and title become too long */
level: {
flexWrap: "wrap"
},
titleHeader: {
display: "flex",
maxWidth: "100%",
cursor: "pointer"
},
title: {
marginLeft: ".25rem",
fontSize: "1rem"
},
/* align child to right */
buttonHeader: {
display: "flex",
marginLeft: "auto"
},
hunkDivider: {
margin: ".5rem 0"
},
changeType: {
marginLeft: ".75rem"
},
diff: {
/* column sizing */
"& > colgroup .diff-gutter-col": {
width: "3.25rem"
},
/* prevent following content from moving down */
"& > .diff-gutter:empty:hover::after": {
fontSize: "0.7rem"
},
/* smaller font size for code */
"& .diff-line": {
fontSize: "0.75rem"
},
/* comment padding for sideBySide view */
"&.split .diff-widget-content .is-indented-line": {
paddingLeft: "3.25rem"
},
/* comment padding for combined view */
"&.unified .diff-widget-content .is-indented-line": {
paddingLeft: "6.5rem"
}
}
};
type Props = DiffObjectProps & {
file: File,
collapsible: true,
// context props
classes: any,
t: string => string
};
@@ -81,6 +27,64 @@ type State = {
sideBySide: boolean
};
const DiffFilePanel = styled.div`
${props =>
props.file &&
props.file.isBinary && {
borderBottom: "none"
}};
`;
const FlexWrapLevel = styled.div`
/* breaks into a second row
when buttons and title become too long */
flex-wrap: wrap;
`;
const FullWidthTitleHeader = styled.div`
max-width: 100%;
`;
const TitleWrapper = styled.span`
margin-left: 0.25rem;
`;
const ButtonWrapper = styled.div`
/* align child to right */
margin-left: auto;
`;
const HunkDivider = styled.hr`
margin: 0.5rem 0;
`;
const ChangeTypeTag = styled(Tag)`
marginleft: ".75rem";
`;
const ModifiedDiffComponent = styled(DiffComponent)`
/* column sizing */
> colgroup .diff-gutter-col: {
width: 3.25rem;
}
/* prevent following content from moving down */
> .diff-gutter:empty:hover::after: {
font-size: 0.7rem;
}
/* smaller font size for code */
& .diff-line: {
font-size: 0.75rem;
}
/* comment padding for sidebyside view */
&.split .diff-widget-content .is-indented-line: {
padding-left: 3.25rem;
}
/* comment padding for combined view */
&.unified .diff-widget-content .is-indented-line: {
padding-left: 6.5rem;
}
`;
class DiffFile extends React.Component<Props, State> {
constructor(props: Props) {
super(props);
@@ -111,9 +115,8 @@ class DiffFile extends React.Component<Props, State> {
};
createHunkHeader = (hunk: Hunk, i: number) => {
const { classes } = this.props;
if (i > 0) {
return <hr className={classes.hunkDivider} />;
return <HunkDivider />;
}
return null;
};
@@ -199,7 +202,7 @@ class DiffFile extends React.Component<Props, State> {
};
renderChangeTag = (file: any) => {
const { t, classes } = this.props;
const { t } = this.props;
if (!file.type) {
return;
}
@@ -216,12 +219,8 @@ class DiffFile extends React.Component<Props, State> {
: "info is-outlined";
return (
<Tag
className={classNames(
"is-rounded",
"has-text-weight-normal",
classes.changeType
)}
<ChangeTypeTag
className={classNames("is-rounded", "has-text-weight-normal")}
color={color}
label={value}
/>
@@ -234,7 +233,6 @@ class DiffFile extends React.Component<Props, State> {
fileControlFactory,
fileAnnotationFactory,
collapsible,
classes,
t
} = this.props;
const { collapsed, sideBySide } = this.state;
@@ -250,12 +248,9 @@ class DiffFile extends React.Component<Props, State> {
body = (
<div className="panel-block is-paddingless">
{fileAnnotations}
<DiffComponent
className={classNames(viewType, classes.diff)}
viewType={viewType}
>
<ModifiedDiffComponent className={viewType} viewType={viewType}>
{file.hunks.map(this.renderHunk)}
</DiffComponent>
</ModifiedDiffComponent>
</div>
);
}
@@ -265,23 +260,27 @@ class DiffFile extends React.Component<Props, State> {
? fileControlFactory(file, this.setCollapse)
: null;
return (
<div className={classNames("panel", classes.panel)}>
<DiffFilePanel className={classNames("panel", "is-size-6")}>
<div className="panel-heading">
<div className={classNames("level", classes.level)}>
<div
className={classNames("level-left", classes.titleHeader)}
<FlexWrapLevel className="level">
<FullWidthTitleHeader
className={classNames(
"level-left",
"is-flex",
"has-cursor-pointer"
)}
onClick={this.toggleCollapse}
title={this.hoverFileTitle(file)}
>
{collapseIcon}
<span
className={classNames("is-ellipsis-overflow", classes.title)}
<TitleWrapper
className={classNames("is-ellipsis-overflow", "is-size-6")}
>
{this.renderFileTitle(file)}
</span>
</TitleWrapper>
{this.renderChangeTag(file)}
</div>
<div className={classNames("level-right", classes.buttonHeader)}>
</FullWidthTitleHeader>
<ButtonWrapper className={classNames("level-right", "is-flex")}>
<ButtonGroup>
<Button
action={this.toggleSideBySide}
@@ -301,13 +300,13 @@ class DiffFile extends React.Component<Props, State> {
</Button>
{fileControls}
</ButtonGroup>
</div>
</div>
</ButtonWrapper>
</FlexWrapLevel>
</div>
{body}
</div>
</DiffFilePanel>
);
}
}
export default injectSheets(styles)(translate("repos")(DiffFile));
export default translate("repos")(DiffFile);

View File

@@ -1,60 +1,63 @@
//@flow
import React from "react";
import type { Changeset, Repository } from "@scm-manager/ui-types";
import classNames from "classnames";
import { Interpolate, translate } from "react-i18next";
import ChangesetId from "./ChangesetId";
import injectSheet from "react-jss";
import { DateFromNow } from "../..";
import ChangesetAuthor from "./ChangesetAuthor";
import { parseDescription } from "./changesets";
import { AvatarWrapper, AvatarImage } from "../../avatar";
import classNames from "classnames";
import styled from "styled-components";
import { ExtensionPoint } from "@scm-manager/ui-extensions";
import type { Changeset, Repository } from "@scm-manager/ui-types";
import DateFromNow from "../../DateFromNow";
import { AvatarWrapper, AvatarImage } from "../../avatar";
import { parseDescription } from "./changesets";
import ChangesetId from "./ChangesetId";
import ChangesetAuthor from "./ChangesetAuthor";
import ChangesetTags from "./ChangesetTags";
import ChangesetButtonGroup from "./ChangesetButtonGroup";
const styles = {
changeset: {
// & references parent rule
// have a look at https://cssinjs.org/jss-plugin-nested?v=v10.0.0-alpha.9
"& + &": {
borderTop: "1px solid rgba(219, 219, 219, 0.5)",
marginTop: "1rem",
paddingTop: "1rem"
}
},
avatarFigure: {
marginTop: ".5rem",
marginRight: ".5rem"
},
avatarImage: {
height: "35px",
width: "35px"
},
metadata: {
marginLeft: 0
},
authorMargin: {
marginTop: "0.5rem"
},
isVcentered: {
alignSelf: "center"
},
flexVcenter: {
display: "flex",
alignItems: "center",
justifyContent: "flex-end"
}
};
type Props = {
repository: Repository,
changeset: Changeset,
t: any,
classes: any
// context props
t: string => string
};
const Wrapper = styled.div`
// & references parent rule
// have a look at https://cssinjs.org/jss-plugin-nested?v=v10.0.0-alpha.9
& + &: {
margin-top: 1rem;
padding-top: 1rem;
border-top: 1px solid rgba(219, 219, 219, 0.5);
}
`;
const AvatarFigure = styled.figure`
margin-top: 0.5rem;
margin-right: 0.5rem;
`;
const FixedSizedAvatar = styled.div`
width: 35px;
height: 35px;
`;
const Metadata = styled.div`
margin-left: 0;
`;
const AuthorWrapper = styled.p`
margin-top: 0.5rem;
`;
const VCenteredColumn = styled.div`
align-self: center;
`;
const VCenteredChildColumn = styled.div`
align-items: center;
justify-content: flex-end;
`;
class ChangesetRow extends React.Component<Props> {
createChangesetId = (changeset: Changeset) => {
const { repository } = this.props;
@@ -62,28 +65,26 @@ class ChangesetRow extends React.Component<Props> {
};
render() {
const { repository, changeset, classes } = this.props;
const { repository, changeset } = this.props;
const description = parseDescription(changeset.description);
const changesetId = this.createChangesetId(changeset);
const dateFromNow = <DateFromNow date={changeset.date} />;
return (
<div className={classes.changeset}>
<Wrapper>
<div className="columns is-gapless is-mobile">
<div className="column is-three-fifths">
<div className="columns is-gapless">
<div className="column is-four-fifths">
<div className="media">
<AvatarWrapper>
<figure
className={classNames(classes.avatarFigure, "media-left")}
>
<div className={classNames("image", classes.avatarImage)}>
<AvatarFigure className="media-left">
<FixedSizedAvatar className="image">
<AvatarImage person={changeset.author} />
</div>
</figure>
</FixedSizedAvatar>
</AvatarFigure>
</AvatarWrapper>
<div className={classNames(classes.metadata, "media-right")}>
<Metadata className="media-right">
<h4 className="has-text-weight-bold is-ellipsis-overflow">
<ExtensionPoint
name="changeset.description"
@@ -107,18 +108,18 @@ class ChangesetRow extends React.Component<Props> {
time={dateFromNow}
/>
</p>
<p className={classNames("is-size-7", classes.authorMargin)}>
<AuthorWrapper className="is-size-7">
<ChangesetAuthor changeset={changeset} />
</p>
</div>
</AuthorWrapper>
</Metadata>
</div>
</div>
<div className={classNames("column", classes.isVcentered)}>
<VCenteredColumn className="column">
<ChangesetTags changeset={changeset} />
</div>
</VCenteredColumn>
</div>
</div>
<div className={classNames("column", classes.flexVcenter)}>
<VCenteredChildColumn className={classNames("column", "is-flex")}>
<ChangesetButtonGroup
repository={repository}
changeset={changeset}
@@ -128,11 +129,11 @@ class ChangesetRow extends React.Component<Props> {
props={{ repository, changeset }}
renderAll={true}
/>
</div>
</VCenteredChildColumn>
</div>
</div>
</Wrapper>
);
}
}
export default injectSheet(styles)(translate("repos")(ChangesetRow));
export default translate("repos")(ChangesetRow);