Files
SCM-Manager/scm-ui/src/repos/sources/containers/Content.js

231 lines
5.8 KiB
JavaScript
Raw Normal View History

2018-10-15 16:45:54 +02:00
// @flow
import React from "react";
2018-10-29 11:50:55 +01:00
import { translate } from "react-i18next";
import { getSources } from "../modules/sources";
2018-11-21 16:55:13 +01:00
import type { File, Repository } from "@scm-manager/ui-types";
import {
2018-11-21 16:55:13 +01:00
DateFromNow,
ErrorNotification,
2018-11-21 16:55:13 +01:00
Loading
2018-10-25 14:37:26 +02:00
} from "@scm-manager/ui-components";
2018-10-25 13:45:52 +02:00
import { connect } from "react-redux";
import ImageViewer from "../components/content/ImageViewer";
import SourcecodeViewer from "../components/content/SourcecodeViewer";
import DownloadViewer from "../components/content/DownloadViewer";
2018-10-29 09:32:30 +01:00
import FileSize from "../components/FileSize";
2018-10-29 09:59:04 +01:00
import injectSheet from "react-jss";
2018-10-29 09:32:30 +01:00
import classNames from "classnames";
2018-11-01 09:54:11 +01:00
import { ExtensionPoint } from "@scm-manager/ui-extensions";
2018-11-01 10:23:08 +01:00
import { getContentType } from "./contentType";
2018-10-15 16:45:54 +02:00
type Props = {
2018-10-25 13:45:52 +02:00
loading: boolean,
error: Error,
file: File,
repository: Repository,
revision: string,
path: string,
classes: any,
2018-11-01 10:23:08 +01:00
t: string => string
2018-10-15 16:45:54 +02:00
};
type State = {
2018-10-25 14:24:53 +02:00
contentType: string,
language: string,
2018-10-29 16:45:32 +01:00
loaded: boolean,
2018-11-01 10:33:35 +01:00
collapsed: boolean,
2018-11-01 10:58:47 +01:00
error?: Error
2018-10-15 16:45:54 +02:00
};
2018-10-29 09:59:04 +01:00
const styles = {
toCenterContent: {
display: "block"
2018-10-29 16:45:32 +01:00
},
pointer: {
cursor: "pointer"
2018-10-29 09:59:04 +01:00
}
};
2018-10-15 16:45:54 +02:00
class Content extends React.Component<Props, State> {
constructor(props: Props) {
super(props);
this.state = {
2018-10-25 14:24:53 +02:00
contentType: "",
language: "",
2018-10-29 16:45:32 +01:00
loaded: false,
2018-11-01 09:19:29 +01:00
collapsed: true
2018-10-15 16:45:54 +02:00
};
}
2018-10-25 13:45:52 +02:00
componentDidMount() {
const { file } = this.props;
2018-10-25 14:24:53 +02:00
getContentType(file._links.self.href)
.then(result => {
if (result.error) {
this.setState({
...this.state,
error: result.error,
loaded: true
2018-10-25 14:24:53 +02:00
});
} else {
this.setState({
...this.state,
contentType: result.type,
language: result.language,
loaded: true
2018-10-25 14:24:53 +02:00
});
}
})
.catch(err => {});
2018-10-25 13:45:52 +02:00
}
2018-10-15 16:45:54 +02:00
2018-10-29 16:45:32 +01:00
toggleCollapse = () => {
this.setState(prevState => ({
collapsed: !prevState.collapsed
}));
};
2018-10-29 09:32:30 +01:00
showHeader() {
2018-11-21 16:55:13 +01:00
const { file, classes, t } = this.props;
2018-10-29 16:45:32 +01:00
const collapsed = this.state.collapsed;
const icon = collapsed ? "fa-angle-right" : "fa-angle-down";
2018-10-29 09:32:30 +01:00
return (
2018-10-29 16:45:32 +01:00
<span className={classes.pointer} onClick={this.toggleCollapse}>
2018-10-29 09:32:30 +01:00
<article className="media">
2018-10-29 16:45:32 +01:00
<div className="media-left">
2018-11-22 10:26:13 +01:00
<i className={classNames("fa", icon)} />
2018-10-29 16:45:32 +01:00
</div>
<div className="media-content">
<div className="content">{file.name}</div>
</div>
2018-11-21 16:55:13 +01:00
<p className="media-right">
2018-11-22 10:26:13 +01:00
<a href="#">
2018-11-22 09:57:56 +01:00
<span className="icon is-medium">
2018-11-22 10:26:13 +01:00
<i className="fas fa-history" />
</span>
<span className="is-hidden-mobile">
{t("sources.content.historyLink")}
</span>
2018-11-21 16:55:13 +01:00
</a>
</p>
2018-10-29 09:32:30 +01:00
</article>
2018-10-29 16:45:32 +01:00
</span>
2018-10-29 09:32:30 +01:00
);
}
2018-10-29 16:45:32 +01:00
showMoreInformation() {
const collapsed = this.state.collapsed;
2018-11-21 16:55:13 +01:00
const { classes, file, revision, t } = this.props;
2018-11-01 09:19:29 +01:00
const date = <DateFromNow date={file.lastModified} />;
const description = file.description ? (
<p>
{file.description.split("\n").map((item, key) => {
return (
<span key={key}>
{item}
<br />
</span>
);
})}
</p>
) : null;
2018-11-21 16:55:13 +01:00
const fileSize = file.directory ? "" : <FileSize bytes={file.length} />;
2018-10-29 16:45:32 +01:00
if (!collapsed) {
return (
<div className={classNames("panel-block", classes.toCenterContent)}>
2018-11-01 09:19:29 +01:00
<table className="table">
<tbody>
<tr>
2018-11-21 16:55:13 +01:00
<td>{t("sources.content.path")}</td>
2018-11-01 10:58:47 +01:00
<td>{file.path}</td>
2018-11-01 09:19:29 +01:00
</tr>
<tr>
2018-11-21 16:55:13 +01:00
<td>{t("sources.content.branch")}</td>
2018-11-01 10:58:47 +01:00
<td>{revision}</td>
2018-11-01 09:19:29 +01:00
</tr>
<tr>
2018-11-21 16:55:13 +01:00
<td>{t("sources.content.size")}</td>
<td>{fileSize}</td>
</tr>
<tr>
<td>{t("sources.content.lastModified")}</td>
2018-11-01 10:58:47 +01:00
<td>{date}</td>
2018-11-01 09:19:29 +01:00
</tr>
<tr>
2018-11-21 16:55:13 +01:00
<td>{t("sources.content.description")}</td>
2018-11-01 10:58:47 +01:00
<td>{description}</td>
2018-11-01 09:19:29 +01:00
</tr>
</tbody>
</table>
2018-10-29 16:45:32 +01:00
</div>
);
}
return null;
}
2018-10-29 09:32:30 +01:00
showContent() {
2018-11-01 09:50:58 +01:00
const { file, revision } = this.props;
2018-11-01 10:58:47 +01:00
const { contentType, language } = this.state;
if (contentType.startsWith("image/")) {
2018-10-29 10:46:16 +01:00
return <ImageViewer file={file} />;
} else if (language) {
2018-10-29 16:45:32 +01:00
return <SourcecodeViewer file={file} language={language} />;
2018-11-01 10:58:47 +01:00
} else if (contentType.startsWith("text/")) {
return <SourcecodeViewer file={file} language="none" />;
2018-10-29 09:32:30 +01:00
} else {
2018-11-01 09:50:58 +01:00
return (
<ExtensionPoint
name="repos.sources.view"
props={{ file, contentType, revision }}
>
<DownloadViewer file={file} />
</ExtensionPoint>
);
2018-10-29 09:32:30 +01:00
}
}
render() {
2018-10-29 09:59:04 +01:00
const { file, classes } = this.props;
2018-11-01 10:58:47 +01:00
const { loaded, error } = this.state;
2018-10-25 14:24:53 +02:00
if (!file || !loaded) {
2018-10-25 13:45:52 +02:00
return <Loading />;
}
2018-11-01 10:33:35 +01:00
if (error) {
2018-10-25 14:24:53 +02:00
return <ErrorNotification error={error} />;
}
2018-10-29 09:32:30 +01:00
const header = this.showHeader();
const content = this.showContent();
2018-10-29 16:45:32 +01:00
const moreInformation = this.showMoreInformation();
2018-10-29 09:32:30 +01:00
return (
<div>
<nav className="panel">
2018-10-29 16:45:32 +01:00
<article className="panel-heading">{header}</article>
{moreInformation}
2018-10-29 09:59:04 +01:00
<div className={classNames("panel-block", classes.toCenterContent)}>
{content}
</div>
2018-10-29 09:32:30 +01:00
</nav>
</div>
);
2018-10-15 16:45:54 +02:00
}
}
2018-10-25 13:45:52 +02:00
const mapStateToProps = (state: any, ownProps: Props) => {
const { repository, revision, path } = ownProps;
const file = getSources(state, repository, revision, path);
return {
file
};
};
2018-10-29 09:59:04 +01:00
export default injectSheet(styles)(
connect(mapStateToProps)(translate("repos")(Content))
);