ant-design/site/theme/template/Content/Article.jsx

130 lines
4.1 KiB
React
Raw Normal View History

import React, { Children, cloneElement } from 'react';
import PropTypes from 'prop-types';
import { FormattedMessage } from 'react-intl';
import DocumentTitle from 'react-document-title';
import { getChildren } from 'jsonml.js/lib/utils';
import { Timeline, Alert, Affix } from 'antd';
2017-04-07 14:29:44 +08:00
import delegate from 'delegate';
import EditButton from './EditButton';
2017-07-10 22:17:52 +08:00
import { ping } from '../utils';
export default class Article extends React.Component {
2016-09-21 11:28:38 +08:00
static contextTypes = {
intl: PropTypes.object.isRequired,
}
componentDidMount() {
2017-04-07 14:29:44 +08:00
// Add ga event click
this.delegation = delegate(this.node, '.resource-card', 'click', (e) => {
if (window.ga) {
window.ga('send', 'event', 'Download', 'resource', e.delegateTarget.href);
}
}, false);
this.componentDidUpdate();
}
componentDidUpdate() {
2016-08-23 21:00:35 +08:00
const links = [...document.querySelectorAll('.outside-link.internal')];
if (links.length === 0) {
return;
}
2017-07-10 22:17:52 +08:00
this.pingTimer = ping((status) => {
2017-02-23 00:35:16 +08:00
if (status !== 'timeout' && status !== 'error') {
links.forEach(link => (link.style.display = 'block'));
2016-08-04 17:59:07 +08:00
} else {
links.forEach(link => link.parentNode.removeChild(link));
}
});
}
2016-07-17 15:25:12 +08:00
componentWillUnmount() {
clearTimeout(this.pingTimer);
2017-04-07 14:29:44 +08:00
if (this.delegation) {
this.delegation.destroy();
}
2016-07-17 15:25:12 +08:00
}
getArticle(article) {
const { content } = this.props;
const { meta } = content;
if (!meta.timeline) {
return article;
}
const timelineItems = [];
let temp = [];
2016-08-01 11:43:42 +08:00
let i = 1;
Children.forEach(article.props.children, (child) => {
if (child.type === 'h2' && temp.length > 0) {
timelineItems.push(<Timeline.Item key={i}>{temp}</Timeline.Item>);
temp = [];
2016-08-01 11:43:42 +08:00
i += 1;
}
temp.push(child);
});
2016-08-01 11:43:42 +08:00
if (temp.length > 0) {
timelineItems.push(<Timeline.Item key={i}>{temp}</Timeline.Item>);
}
return cloneElement(article, {
children: <Timeline>{timelineItems}</Timeline>,
});
}
render() {
2017-10-09 13:23:20 +08:00
const { props } = this;
const { content } = props;
const { meta, description } = content;
2016-09-21 11:28:38 +08:00
const { title, subtitle, filename } = meta;
2017-10-09 13:23:20 +08:00
const { locale } = this.context.intl;
const isNotTranslated = locale === 'en-US' && typeof title === 'object';
return (
2016-09-21 11:28:38 +08:00
<DocumentTitle title={`${title[locale] || title} - Ant Design`}>
2017-04-07 14:29:44 +08:00
<article className="markdown" ref={(node) => { this.node = node; }}>
{isNotTranslated && (
<Alert
type="warning"
message={(
<span>
2017-11-27 00:09:59 +08:00
This article has not been translated yet. Wan&apos;t to help us out?
<a href="https://github.com/ant-design/ant-design/issues/1471">See this issue on GitHub.</a>
</span>
)}
/>
)}
<h1>
2016-09-21 11:28:38 +08:00
{title[locale] || title}
{
2016-09-21 11:28:38 +08:00
!subtitle || locale === 'en-US' ? null :
2017-02-23 11:45:48 +08:00
<span className="subtitle">{subtitle}</span>
}
<EditButton title={<FormattedMessage id="app.content.edit-page" />} filename={filename} />
</h1>
{
!description ? null :
props.utils.toReactComponent(
['section', { className: 'markdown' }].concat(getChildren(description))
)
}
{
2016-07-17 15:20:30 +08:00
(!content.toc || content.toc.length <= 1 || meta.toc === false) ? null :
<Affix className="toc-affix" offsetTop={16}>
{
props.utils.toReactComponent(
['ul', { className: 'toc' }].concat(getChildren(content.toc))
)
}
</Affix>
}
{
this.getArticle(props.utils.toReactComponent(
['section', { className: 'markdown' }].concat(getChildren(content.content))
))
}
2017-06-07 17:06:21 +08:00
{
props.utils.toReactComponent(
['section', {
className: 'markdown api-container',
}].concat(getChildren(content.api || ['placeholder']))
)
}
</article>
</DocumentTitle>
);
}
}