Co-authored-by: tobi <tobi.smethurst@protonmail.com> Reviewed-on: https://codeberg.org/superseriousbusiness/masto-fe-standalone/pulls/88 Co-authored-by: Zoë Bijl <moiety@noreply.codeberg.org> Co-committed-by: Zoë Bijl <moiety@noreply.codeberg.org>
154 lines
6.1 KiB
JavaScript
154 lines
6.1 KiB
JavaScript
import PropTypes from "prop-types";
|
|
|
|
import { FormattedMessage, injectIntl, defineMessages } from "react-intl";
|
|
|
|
import { Helmet } from "react-helmet";
|
|
import { Link } from "react-router-dom";
|
|
|
|
import ImmutablePropTypes from "react-immutable-proptypes";
|
|
import ImmutablePureComponent from "react-immutable-pure-component";
|
|
import { connect } from "react-redux";
|
|
|
|
import { debounce } from "lodash";
|
|
|
|
import illustration from "mastodon/../images/elephant_ui_conversation.svg";
|
|
import { fetchAccount } from "mastodon/actions/accounts";
|
|
import { focusCompose } from "mastodon/actions/compose";
|
|
import { closeOnboarding } from "mastodon/actions/onboarding";
|
|
import Column from "mastodon/features/ui/components/column";
|
|
import { me } from "mastodon/initial_state";
|
|
import { makeGetAccount } from "mastodon/selectors";
|
|
import { assetHost } from "mastodon/utils/config";
|
|
|
|
import ArrowSmallRight from "./components/arrow_small_right";
|
|
import Step from "./components/step";
|
|
import Follows from "./follows";
|
|
import Share from "./share";
|
|
|
|
const messages = defineMessages({
|
|
template: { id: "onboarding.compose.template", defaultMessage: "Hello #Mastodon!" },
|
|
});
|
|
|
|
const mapStateToProps = () => {
|
|
const getAccount = makeGetAccount();
|
|
|
|
return state => ({
|
|
account: getAccount(state, me),
|
|
});
|
|
};
|
|
|
|
class Onboarding extends ImmutablePureComponent {
|
|
|
|
static contextTypes = {
|
|
router: PropTypes.object.isRequired,
|
|
};
|
|
|
|
static propTypes = {
|
|
dispatch: PropTypes.func.isRequired,
|
|
account: ImmutablePropTypes.map,
|
|
multiColumn: PropTypes.bool,
|
|
};
|
|
|
|
state = {
|
|
step: null,
|
|
profileClicked: false,
|
|
shareClicked: false,
|
|
};
|
|
|
|
handleClose = () => {
|
|
const { dispatch } = this.props;
|
|
const { router } = this.context;
|
|
|
|
dispatch(closeOnboarding());
|
|
router.history.push("/home");
|
|
};
|
|
|
|
handleProfileClick = () => {
|
|
this.setState({ profileClicked: true });
|
|
};
|
|
|
|
handleFollowClick = () => {
|
|
this.setState({ step: "follows" });
|
|
};
|
|
|
|
handleComposeClick = () => {
|
|
const { dispatch, intl } = this.props;
|
|
const { router } = this.context;
|
|
|
|
dispatch(focusCompose(router.history, intl.formatMessage(messages.template)));
|
|
};
|
|
|
|
handleShareClick = () => {
|
|
this.setState({ step: "share", shareClicked: true });
|
|
};
|
|
|
|
handleBackClick = () => {
|
|
this.setState({ step: null });
|
|
};
|
|
|
|
handleWindowFocus = debounce(() => {
|
|
const { dispatch, account } = this.props;
|
|
dispatch(fetchAccount(account.get("id")));
|
|
}, 1000, { trailing: true });
|
|
|
|
componentDidMount () {
|
|
window.addEventListener("focus", this.handleWindowFocus, false);
|
|
}
|
|
|
|
componentWillUnmount () {
|
|
window.removeEventListener("focus", this.handleWindowFocus);
|
|
}
|
|
|
|
render () {
|
|
const { account, multiColumn } = this.props;
|
|
const { step, shareClicked } = this.state;
|
|
|
|
switch(step) {
|
|
case "follows":
|
|
return <Follows onBack={this.handleBackClick} multiColumn={multiColumn} />;
|
|
case "share":
|
|
return <Share onBack={this.handleBackClick} multiColumn={multiColumn} />;
|
|
}
|
|
|
|
return (
|
|
<Column>
|
|
<div className='scrollable privacy-policy'>
|
|
<div className='column-title'>
|
|
<img src={illustration} alt='' className='onboarding__illustration' />
|
|
<h3><FormattedMessage id='onboarding.start.title' defaultMessage="You've made it!" /></h3>
|
|
<p><FormattedMessage id='onboarding.start.lead' defaultMessage="Your new Mastodon account is ready to go. Here's how you can make the most of it:" /></p>
|
|
</div>
|
|
|
|
<div className='onboarding__steps'>
|
|
<Step onClick={this.handleProfileClick} href='/settings/profile' completed={(!account.get("avatar").endsWith("missing.png")) || (account.get("display_name").length > 0 && account.get("note").length > 0)} icon='address-book-o' label={<FormattedMessage id='onboarding.steps.setup_profile.title' defaultMessage='Customize your profile' />} description={<FormattedMessage id='onboarding.steps.setup_profile.body' defaultMessage='Others are more likely to interact with you with a filled out profile.' />} />
|
|
<Step onClick={this.handleFollowClick} completed={(account.get("following_count") * 1) >= 7} icon='user-plus' label={<FormattedMessage id='onboarding.steps.follow_people.title' defaultMessage='Find at least {count, plural, one {one person} other {# people}} to follow' values={{ count: 7 }} />} description={<FormattedMessage id='onboarding.steps.follow_people.body' defaultMessage="You curate your own home feed. Let's fill it with interesting people." />} />
|
|
<Step onClick={this.handleComposeClick} completed={(account.get("statuses_count") * 1) >= 1} icon='pencil-square-o' label={<FormattedMessage id='onboarding.steps.publish_status.title' defaultMessage='Make your first post' />} description={<FormattedMessage id='onboarding.steps.publish_status.body' defaultMessage='Say hello to the world.' values={{ emoji: <img className='emojione' alt='🐘' src={`${assetHost}/emoji/1f418.svg`} /> }} />} />
|
|
<Step onClick={this.handleShareClick} completed={shareClicked} icon='copy' label={<FormattedMessage id='onboarding.steps.share_profile.title' defaultMessage='Share your profile' />} description={<FormattedMessage id='onboarding.steps.share_profile.body' defaultMessage='Let your friends know how to find you on Mastodon!' />} />
|
|
</div>
|
|
|
|
<p className='onboarding__lead'><FormattedMessage id='onboarding.start.skip' defaultMessage="Don't need help getting started?" /></p>
|
|
|
|
<div className='onboarding__links'>
|
|
<Link to='/explore' className='onboarding__link'>
|
|
<FormattedMessage id='onboarding.actions.go_to_explore' defaultMessage='Take me to trending' />
|
|
<ArrowSmallRight />
|
|
</Link>
|
|
|
|
<Link to='/home' className='onboarding__link'>
|
|
<FormattedMessage id='onboarding.actions.go_to_home' defaultMessage='Take me to my home feed' />
|
|
<ArrowSmallRight />
|
|
</Link>
|
|
</div>
|
|
</div>
|
|
|
|
<Helmet>
|
|
<meta name='robots' content='noindex' />
|
|
</Helmet>
|
|
</Column>
|
|
);
|
|
}
|
|
|
|
}
|
|
|
|
export default connect(mapStateToProps)(injectIntl(Onboarding));
|