Developers: scaffold with WIP developers page

This commit is contained in:
Alex Gleason 2021-11-02 00:05:33 -05:00
parent 37754c610a
commit 15a91d5ae3
No known key found for this signature in database
GPG key ID: 7211D1F99744FBB7
9 changed files with 72 additions and 4 deletions

View file

@ -38,6 +38,8 @@ export const defaultSettings = ImmutableMap({
dyslexicFont: false,
demetricator: false,
isDeveloper: false,
chats: ImmutableMap({
panes: ImmutableList(),
mainWindow: 'minimized',

View file

@ -9,6 +9,7 @@ import { NavLink, withRouter } from 'react-router-dom';
import Icon from 'soapbox/components/icon';
import IconWithCounter from 'soapbox/components/icon_with_counter';
import classNames from 'classnames';
import { getSettings } from 'soapbox/actions/settings';
import { getFeatures } from 'soapbox/utils/features';
import { getSoapboxConfig } from 'soapbox/actions/soapbox';
import { isStaff, getBaseURL } from 'soapbox/utils/accounts';
@ -27,6 +28,7 @@ const mapStateToProps = state => {
chatsCount: state.get('chats').reduce((acc, curr) => acc + Math.min(curr.get('unread', 0), 1), 0),
dashboardCount: reportsCount + approvalCount,
baseURL: getBaseURL(account),
settings: getSettings(state),
features: getFeatures(instance),
instance,
};
@ -47,13 +49,14 @@ class PrimaryNavigation extends React.PureComponent {
notificationCount: PropTypes.number,
chatsCount: PropTypes.number,
baseURL: PropTypes.string,
settings: PropTypes.object.isRequired,
features: PropTypes.object.isRequired,
location: PropTypes.object,
instance: ImmutablePropTypes.map.isRequired,
};
render() {
const { account, features, notificationCount, chatsCount, dashboardCount, location, instance, baseURL } = this.props;
const { account, settings, features, notificationCount, chatsCount, dashboardCount, location, instance, baseURL } = this.props;
return (
<div className='column-header__wrapper primary-navigation__wrapper'>
@ -127,6 +130,16 @@ class PrimaryNavigation extends React.PureComponent {
</a>
)}
{(settings.get('isDeveloper')) && (
<NavLink key='developers' className='btn grouped' to='/developers'>
<Icon
src={require('@tabler/icons/icons/code.svg')}
className={classNames('primary-navigation__icon', { 'svg-icon--active': location.pathname.startsWith('/developers') })}
/>
<FormattedMessage id='navigation.developers' defaultMessage='Developers' />
</NavLink>
)}
<hr />
{features.federating ? (

View file

@ -18,6 +18,7 @@ import { logOut, switchAccount } from 'soapbox/actions/auth';
import ThemeToggle from '../features/ui/components/theme_toggle_container';
import { fetchOwnAccounts } from 'soapbox/actions/auth';
import { is as ImmutableIs } from 'immutable';
import { getSettings } from 'soapbox/actions/settings';
import { getSoapboxConfig } from 'soapbox/actions/soapbox';
import { getFeatures } from 'soapbox/utils/features';
@ -45,6 +46,7 @@ const messages = defineMessages({
donate: { id: 'donate', defaultMessage: 'Donate' },
donate_crypto: { id: 'donate_crypto', defaultMessage: 'Donate cryptocurrency' },
info: { id: 'column.info', defaultMessage: 'Server information' },
developers: { id: 'navigation.developers', defaultMessage: 'Developers' },
add_account: { id: 'profile_dropdown.add_account', defaultMessage: 'Add an existing account' },
});
@ -67,6 +69,7 @@ const makeMapStateToProps = () => {
hasCrypto: typeof soapbox.getIn(['cryptoAddresses', 0, 'ticker']) === 'string',
otherAccounts: getOtherAccounts(state),
features,
settings: getSettings(state),
siteTitle: instance.get('title'),
baseURL: getBaseURL(account),
};
@ -101,6 +104,7 @@ class SidebarMenu extends ImmutablePureComponent {
otherAccounts: ImmutablePropTypes.list,
sidebarOpen: PropTypes.bool,
onClose: PropTypes.func.isRequired,
settings: PropTypes.object.isRequired,
features: PropTypes.object.isRequired,
baseURL: PropTypes.string,
};
@ -159,7 +163,7 @@ class SidebarMenu extends ImmutablePureComponent {
}
render() {
const { sidebarOpen, intl, account, onClickLogOut, donateUrl, otherAccounts, hasCrypto, features, siteTitle, baseURL } = this.props;
const { sidebarOpen, intl, account, onClickLogOut, donateUrl, otherAccounts, hasCrypto, settings, features, siteTitle, baseURL } = this.props;
const { switcher } = this.state;
if (!account) return null;
const acct = account.get('acct');
@ -319,6 +323,13 @@ class SidebarMenu extends ImmutablePureComponent {
</Link>
</div>
{(settings.get('isDeveloper')) && (
<Link className='sidebar-menu-item' to='/developers' onClick={this.handleClose}>
<Icon src={require('@tabler/icons/icons/code.svg')} />
<span className='sidebar-menu-item__title'>{intl.formatMessage(messages.developers)}</span>
</Link>
)}
<div className='sidebar-menu__section'>
<Link className='sidebar-menu-item' to='/auth/sign_out' onClick={onClickLogOut}>
<Icon src={require('@tabler/icons/icons/logout.svg')} />

View file

@ -0,0 +1,29 @@
import React from 'react';
import PropTypes from 'prop-types';
import { injectIntl, defineMessages } from 'react-intl';
import Column from '../ui/components/column';
const messages = defineMessages({
heading: { id: 'column.developers', defaultMessage: 'Developers' },
});
export default @injectIntl
class Developers extends React.Component {
static propTypes = {
intl: PropTypes.object.isRequired,
}
render() {
const { intl } = this.props;
return (
<Column heading={intl.formatMessage(messages.heading)}>
<div style={{ padding: '20px 10px', textAlign: 'center' }}>
WIP: Developers page
</div>
</Column>
);
}
}

View file

@ -262,6 +262,10 @@ class Preferences extends ImmutablePureComponent {
hint={<FormattedMessage id='preferences.hints.demetricator' defaultMessage='Decrease social media anxiety by hiding all numbers from the site.' />}
path={['demetricator']}
/>
<SettingsCheckbox
label={<FormattedMessage id='preferences.fields.developer_label' defaultMessage='Developer tools' />}
path={['isDeveloper']}
/>
</FieldsGroup>
</SimpleForm>
</Column>

View file

@ -115,6 +115,7 @@ import {
Share,
NewStatus,
IntentionalError,
Developers,
} from './util/async-components';
// Dummy import, to make sure that <Status /> ends up in the application bundle.
@ -318,7 +319,9 @@ class SwitchingColumnsArea extends React.PureComponent {
<WrappedRoute path='/admin/log' page={AdminPage} component={ModerationLog} content={children} exact />
<WrappedRoute path='/admin/users' page={AdminPage} component={UserIndex} content={children} exact />
<WrappedRoute path='/info' page={EmptyPage} component={ServerInfo} content={children} />
<WrappedRoute path='/error' page={EmptyPage} component={IntentionalError} content={children} />
<WrappedRoute path='/developers' page={DefaultPage} component={Developers} content={children} />
<WrappedRoute path='/donate/crypto' publicRoute page={DefaultPage} component={CryptoDonate} content={children} />
<WrappedRoute path='/federation_restrictions' publicRoute page={DefaultPage} component={FederationRestrictions} content={children} />

View file

@ -425,3 +425,7 @@ export function NewStatus() {
export function IntentionalError() {
return import(/* webpackChunkName: "error" */'../../intentional_error');
}
export function Developers() {
return import(/* webpackChunkName: "features/developers" */'../../developers');
}

View file

@ -18,7 +18,8 @@
fill: currentColor;
}
svg.icon-tabler-search {
svg.icon-tabler-search,
svg.icon-tabler-code {
stroke-width: 2.3px;
}

View file

@ -18,7 +18,8 @@
}
&--active {
svg.icon-tabler-search {
svg.icon-tabler-search,
svg.icon-tabler-code {
stroke-width: 2.5px;
}
}