2018-07-25 23:18:35 -06:00
|
|
|
// @flow
|
|
|
|
|
2019-11-26 14:08:34 -05:00
|
|
|
import React from 'react';
|
2018-07-25 23:18:35 -06:00
|
|
|
import LoadingScreen from 'component/common/loading-screen';
|
|
|
|
import MarkdownPreview from 'component/common/markdown-preview';
|
2019-11-07 14:39:22 -05:00
|
|
|
import CodeViewer from 'component/viewers/codeViewer';
|
2019-11-26 14:08:34 -05:00
|
|
|
import * as https from 'https';
|
2019-03-26 23:40:02 -05:00
|
|
|
|
2018-07-25 23:18:35 -06:00
|
|
|
type Props = {
|
2018-07-27 19:42:35 -06:00
|
|
|
theme: string,
|
2018-07-25 23:18:35 -06:00
|
|
|
source: {
|
2019-11-26 14:08:34 -05:00
|
|
|
file: (?string) => any,
|
|
|
|
stream: string,
|
2018-07-25 23:18:35 -06:00
|
|
|
fileType: string,
|
2018-08-01 18:53:38 -06:00
|
|
|
contentType: string,
|
2018-07-25 23:18:35 -06:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2018-10-14 18:27:09 -06:00
|
|
|
type State = {
|
|
|
|
error: boolean,
|
|
|
|
loading: boolean,
|
|
|
|
content: ?string,
|
|
|
|
};
|
|
|
|
|
|
|
|
class DocumentViewer extends React.PureComponent<Props, State> {
|
|
|
|
constructor(props: Props) {
|
2018-07-25 23:18:35 -06:00
|
|
|
super(props);
|
|
|
|
this.state = {
|
2018-10-14 18:27:09 -06:00
|
|
|
error: false,
|
2018-07-25 23:18:35 -06:00
|
|
|
loading: true,
|
2018-10-14 18:27:09 -06:00
|
|
|
content: null,
|
2018-07-25 23:18:35 -06:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
componentDidMount() {
|
|
|
|
const { source } = this.props;
|
2019-11-26 14:08:34 -05:00
|
|
|
// @if TARGET='app'
|
|
|
|
if (source && source.file) {
|
|
|
|
const stream = source.file('utf8');
|
2018-07-25 23:18:35 -06:00
|
|
|
|
2019-03-21 09:59:31 -04:00
|
|
|
let data = '';
|
2018-07-25 23:18:35 -06:00
|
|
|
|
2019-03-21 09:59:31 -04:00
|
|
|
stream.on('data', chunk => {
|
|
|
|
data += chunk;
|
|
|
|
});
|
2018-07-25 23:18:35 -06:00
|
|
|
|
2019-03-21 09:59:31 -04:00
|
|
|
stream.on('end', () => {
|
|
|
|
this.setState({ content: data, loading: false });
|
|
|
|
});
|
|
|
|
|
|
|
|
stream.on('error', () => {
|
|
|
|
this.setState({ error: true, loading: false });
|
|
|
|
});
|
|
|
|
}
|
2019-11-26 14:08:34 -05:00
|
|
|
// @endif
|
|
|
|
// @if TARGET='web'
|
|
|
|
if (source && source.stream) {
|
|
|
|
https.get(
|
|
|
|
source.stream,
|
|
|
|
function(response) {
|
|
|
|
if (response.statusCode === 200) {
|
2019-11-27 11:53:41 -05:00
|
|
|
let data = '';
|
2019-11-26 14:08:34 -05:00
|
|
|
response.on('data', function(chunk) {
|
2019-11-27 11:53:41 -05:00
|
|
|
data += chunk;
|
2019-11-26 14:08:34 -05:00
|
|
|
});
|
|
|
|
response.on(
|
|
|
|
'end',
|
|
|
|
function() {
|
2019-11-27 11:53:41 -05:00
|
|
|
this.setState({ content: data, loading: false });
|
2019-11-26 14:08:34 -05:00
|
|
|
}.bind(this)
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
this.setState({ error: true, loading: false });
|
|
|
|
}
|
|
|
|
}.bind(this)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
// @endif
|
2018-07-25 23:18:35 -06:00
|
|
|
}
|
|
|
|
|
2018-10-14 18:27:09 -06:00
|
|
|
renderDocument() {
|
2018-07-26 18:24:00 -06:00
|
|
|
let viewer = null;
|
2018-10-14 18:27:09 -06:00
|
|
|
const { content } = this.state;
|
2018-07-27 19:42:35 -06:00
|
|
|
const { source, theme } = this.props;
|
|
|
|
const { fileType, contentType } = source;
|
2018-07-26 18:24:00 -06:00
|
|
|
const markdownType = ['md', 'markdown'];
|
2019-11-26 14:08:34 -05:00
|
|
|
if (markdownType.includes(fileType) || contentType === 'text/markdown' || contentType === 'text/md') {
|
2018-07-26 18:24:00 -06:00
|
|
|
// Render markdown
|
2019-12-03 11:41:44 -05:00
|
|
|
viewer = <MarkdownPreview content={content} />;
|
2018-08-01 18:53:38 -06:00
|
|
|
} else {
|
2018-07-26 18:24:00 -06:00
|
|
|
// Render plain text
|
2019-11-07 14:39:22 -05:00
|
|
|
viewer = <CodeViewer value={content} contentType={contentType} theme={theme} />;
|
2018-07-25 23:18:35 -06:00
|
|
|
}
|
2018-07-26 18:24:00 -06:00
|
|
|
|
|
|
|
return viewer;
|
2018-07-25 23:18:35 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
render() {
|
2018-08-01 18:53:38 -06:00
|
|
|
const { error, loading, content } = this.state;
|
|
|
|
const isReady = content && !error;
|
2018-07-25 23:18:35 -06:00
|
|
|
const loadingMessage = __('Rendering document.');
|
2018-07-27 19:54:06 -06:00
|
|
|
const errorMessage = __("Sorry, looks like we can't load the document.");
|
2018-07-25 23:18:35 -06:00
|
|
|
|
|
|
|
return (
|
2019-08-13 01:35:13 -04:00
|
|
|
<div className="file-render__viewer--document">
|
2018-07-25 23:18:35 -06:00
|
|
|
{loading && !error && <LoadingScreen status={loadingMessage} spinner />}
|
2018-08-01 18:53:38 -06:00
|
|
|
{error && <LoadingScreen status={errorMessage} spinner={!error} />}
|
2019-11-07 14:39:22 -05:00
|
|
|
{isReady && this.renderDocument()}
|
2018-07-25 23:18:35 -06:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export default DocumentViewer;
|