bigbuffet-rw/app/soapbox/features/ui/util/react_router_helpers.js

131 lines
3.8 KiB
JavaScript
Raw Normal View History

2020-03-27 13:59:38 -07:00
import React from 'react';
import { connect } from 'react-redux';
2020-03-27 13:59:38 -07:00
import PropTypes from 'prop-types';
import ImmutablePropTypes from 'react-immutable-proptypes';
2020-03-27 13:59:38 -07:00
import { Redirect, Route } from 'react-router-dom';
2022-01-10 14:01:24 -08:00
import { getSettings } from 'soapbox/actions/settings';
import { isStaff, isAdmin } from 'soapbox/utils/accounts';
2020-03-27 13:59:38 -07:00
import ColumnsAreaContainer from '../containers/columns_area_container';
import ColumnLoading from '../components/column_loading';
import ColumnForbidden from '../components/column_forbidden';
2020-03-27 13:59:38 -07:00
import BundleColumnError from '../components/bundle_column_error';
import BundleContainer from '../containers/bundle_container';
const mapStateToProps = state => {
const me = state.get('me');
return {
account: state.getIn(['accounts', me]),
settings: getSettings(state),
};
};
class WrappedRoute extends React.Component {
2020-04-14 11:44:40 -07:00
2020-03-27 13:59:38 -07:00
static propTypes = {
component: PropTypes.func.isRequired,
2021-07-09 17:56:09 -07:00
page: PropTypes.oneOfType([PropTypes.object, PropTypes.func]),
2020-03-27 13:59:38 -07:00
content: PropTypes.node,
componentParams: PropTypes.object,
layout: PropTypes.object,
account: ImmutablePropTypes.map,
settings: ImmutablePropTypes.map.isRequired,
2020-03-27 13:59:38 -07:00
publicRoute: PropTypes.bool,
staffOnly: PropTypes.bool,
adminOnly: PropTypes.bool,
developerOnly: PropTypes.bool,
2020-03-27 13:59:38 -07:00
};
static defaultProps = {
componentParams: {},
};
renderComponent = ({ match }) => {
const { component, content, componentParams, layout, page: Page } = this.props;
if (Page) {
return (
<BundleContainer fetchComponent={component} loading={this.renderLoading} error={this.renderError}>
{Component =>
(
<Page params={match.params} layout={layout} {...componentParams}>
2020-04-14 11:44:40 -07:00
<Component params={match.params} {...componentParams}>
{content}
</Component>
</Page>
2020-03-27 13:59:38 -07:00
)
}
</BundleContainer>
);
}
return (
<BundleContainer fetchComponent={component} loading={this.renderLoading} error={this.renderError}>
{Component =>
(
<ColumnsAreaContainer layout={layout}>
<Component params={match.params} {...componentParams}>
{content}
</Component>
</ColumnsAreaContainer>
)
}
</BundleContainer>
);
}
renderLoading = () => {
return (
<ColumnsAreaContainer layout={this.props.layout}>
<ColumnLoading />
</ColumnsAreaContainer>
);
2020-03-27 13:59:38 -07:00
}
renderForbidden = () => {
return (
<ColumnsAreaContainer layout={this.props.layout}>
<ColumnForbidden />
</ColumnsAreaContainer>
);
}
2020-03-27 13:59:38 -07:00
renderError = (props) => {
return (
<ColumnsAreaContainer layout={this.props.layout}>
<BundleColumnError {...props} />
</ColumnsAreaContainer>
);
2020-03-27 13:59:38 -07:00
}
loginRedirect = () => {
const actualUrl = encodeURIComponent(`${this.props.computedMatch.url}${this.props.location.search}`); // eslint-disable-line react/prop-types
return <Redirect to={`/auth/sign_in?redirect_uri=${actualUrl}`} />;
}
render() {
const { component: Component, content, account, settings, publicRoute, developerOnly, staffOnly, adminOnly, ...rest } = this.props;
const authorized = [
account || publicRoute,
developerOnly ? settings.get('isDeveloper') : true,
staffOnly ? account && isStaff(account) : true,
adminOnly ? account && isAdmin(account) : true,
].every(c => c);
if (!authorized) {
if (!account) {
return this.loginRedirect();
} else {
return this.renderForbidden();
}
2020-03-27 13:59:38 -07:00
}
return <Route {...rest} render={this.renderComponent} />;
}
2020-04-14 11:44:40 -07:00
2020-03-27 13:59:38 -07:00
}
const wrappedRoute = connect(mapStateToProps)(WrappedRoute);
2020-04-14 11:44:40 -07:00
export { wrappedRoute as WrappedRoute };