Support GET /patron/v1/funding
This commit is contained in:
parent
1107aa027d
commit
2a245d076a
7 changed files with 89 additions and 9 deletions
47
app/gabsocial/actions/patron.js
Normal file
47
app/gabsocial/actions/patron.js
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import api, { getLinks } from '../api';
|
||||||
|
import openDB from '../storage/db';
|
||||||
|
import { me } from 'gabsocial/initial_state';
|
||||||
|
|
||||||
|
export const PATRON_FUNDING_FETCH_REQUEST = 'PATRON_FUNDING_FETCH_REQUEST';
|
||||||
|
export const PATRON_FUNDING_FETCH_SUCCESS = 'PATRON_FUNDING_FETCH_SUCCESS';
|
||||||
|
export const PATRON_FUNDING_FETCH_FAIL = 'PATRON_FUNDING_FETCH_FAIL';
|
||||||
|
export const PATRON_FUNDING_IMPORT = 'PATRON_FUNDING_IMPORT';
|
||||||
|
|
||||||
|
export function fetchFunding() {
|
||||||
|
return (dispatch, getState) => {
|
||||||
|
api(getState).get(`/patron/v1/funding`).then(response => {
|
||||||
|
dispatch(importFetchedFunding(response.data));
|
||||||
|
}).then(() => {
|
||||||
|
dispatch(fetchFundingSuccess());
|
||||||
|
}).catch(error => {
|
||||||
|
dispatch(fetchFundingFail(error));
|
||||||
|
});
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function importFetchedFunding(funding) {
|
||||||
|
return {
|
||||||
|
type: PATRON_FUNDING_IMPORT,
|
||||||
|
funding
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function fetchFundingRequest() {
|
||||||
|
return {
|
||||||
|
type: PATRON_FUNDING_FETCH_REQUEST,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function fetchAccountSuccess() {
|
||||||
|
return {
|
||||||
|
type: PATRON_FUNDING_FETCH_SUCCESS,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export function fetchFundingFail(error) {
|
||||||
|
return {
|
||||||
|
type: PATRON_FUNDING_FETCH_FAIL,
|
||||||
|
error,
|
||||||
|
skipAlert: true,
|
||||||
|
};
|
||||||
|
};
|
|
@ -5,19 +5,30 @@ import { injectIntl, FormattedMessage } from 'react-intl';
|
||||||
import ImmutablePropTypes from 'react-immutable-proptypes';
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
||||||
import ImmutablePureComponent from 'react-immutable-pure-component';
|
import ImmutablePureComponent from 'react-immutable-pure-component';
|
||||||
import ProgressBar from '../../../components/progress_bar';
|
import ProgressBar from '../../../components/progress_bar';
|
||||||
import { funding } from '../../../initial_state';
|
import { fetchFunding } from 'gabsocial/actions/patron';
|
||||||
|
|
||||||
|
|
||||||
class FundingPanel extends ImmutablePureComponent {
|
class FundingPanel extends ImmutablePureComponent {
|
||||||
|
|
||||||
|
componentDidMount () {
|
||||||
|
this.props.dispatch(fetchFunding());
|
||||||
|
}
|
||||||
|
|
||||||
render() {
|
render() {
|
||||||
const { goal, goal_text } = this.props;
|
const { funding } = this.props;
|
||||||
const goal_reached = funding.amount >= goal;
|
|
||||||
|
if (!funding) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
const goal = funding.getIn(['goals', '0', 'amount']);
|
||||||
|
const goal_text = funding.getIn(['goals', '0', 'text']);
|
||||||
|
const goal_reached = funding.get('amount') >= goal;
|
||||||
let ratio_text;
|
let ratio_text;
|
||||||
|
|
||||||
if (goal_reached) {
|
if (goal_reached) {
|
||||||
ratio_text = <><strong>${Math.floor(goal/100)}</strong> per month<span className='funding-panel__reached'>— reached!</span></>;
|
ratio_text = <><strong>${Math.floor(goal/100)}</strong> per month<span className='funding-panel__reached'>— reached!</span></>;
|
||||||
} else {
|
} else {
|
||||||
ratio_text = <><strong>${Math.floor(funding.amount/100)} out of ${Math.floor(goal/100)}</strong> per month</>;
|
ratio_text = <><strong>${Math.floor(funding.get('amount')/100)} out of ${Math.floor(goal/100)}</strong> per month</>;
|
||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
|
@ -32,7 +43,7 @@ class FundingPanel extends ImmutablePureComponent {
|
||||||
<div className='funding-panel__ratio'>
|
<div className='funding-panel__ratio'>
|
||||||
{ratio_text}
|
{ratio_text}
|
||||||
</div>
|
</div>
|
||||||
<ProgressBar progress={funding.amount/goal} />
|
<ProgressBar progress={funding.get('amount')/goal} />
|
||||||
<div className='funding-panel__description'>
|
<div className='funding-panel__description'>
|
||||||
{goal_text}
|
{goal_text}
|
||||||
</div>
|
</div>
|
||||||
|
@ -43,9 +54,9 @@ class FundingPanel extends ImmutablePureComponent {
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
const mapStateToProps = state => {
|
const mapStateToProps = state => {
|
||||||
return {
|
return {
|
||||||
|
funding: state.getIn(['patron', 'funding'])
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { connect } from 'react-redux';
|
import { connect } from 'react-redux';
|
||||||
import { me, funding } from 'gabsocial/initial_state';
|
import { me, funding } from 'gabsocial/initial_state';
|
||||||
|
import soapbox from 'soapbox/config';
|
||||||
import PropTypes from 'prop-types';
|
import PropTypes from 'prop-types';
|
||||||
import ImmutablePureComponent from 'react-immutable-pure-component';
|
import ImmutablePureComponent from 'react-immutable-pure-component';
|
||||||
import WhoToFollowPanel from '../features/ui/components/who_to_follow_panel';
|
import WhoToFollowPanel from '../features/ui/components/who_to_follow_panel';
|
||||||
|
@ -21,7 +22,6 @@ export default @connect(mapStateToProps)
|
||||||
class HomePage extends ImmutablePureComponent {
|
class HomePage extends ImmutablePureComponent {
|
||||||
render () {
|
render () {
|
||||||
const {children, account} = this.props;
|
const {children, account} = this.props;
|
||||||
const hasFunding = funding && funding.goal && funding.goal.amount && funding.goal.text;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='page'>
|
<div className='page'>
|
||||||
|
@ -31,7 +31,7 @@ class HomePage extends ImmutablePureComponent {
|
||||||
<div className='columns-area__panels__pane columns-area__panels__pane--left'>
|
<div className='columns-area__panels__pane columns-area__panels__pane--left'>
|
||||||
<div className='columns-area__panels__pane__inner'>
|
<div className='columns-area__panels__pane__inner'>
|
||||||
<UserPanel />
|
<UserPanel />
|
||||||
{hasFunding && <FundingPanel goal={funding.goal.amount} goal_text={funding.goal.text} />}
|
{soapbox.features.patron && <FundingPanel />}
|
||||||
<PromoPanel />
|
<PromoPanel />
|
||||||
<LinkFooter />
|
<LinkFooter />
|
||||||
</div>
|
</div>
|
||||||
|
|
|
@ -37,6 +37,7 @@ import group_relationships from './group_relationships';
|
||||||
import group_lists from './group_lists';
|
import group_lists from './group_lists';
|
||||||
import group_editor from './group_editor';
|
import group_editor from './group_editor';
|
||||||
import sidebar from './sidebar';
|
import sidebar from './sidebar';
|
||||||
|
import patron from './patron';
|
||||||
|
|
||||||
const reducers = {
|
const reducers = {
|
||||||
dropdown_menu,
|
dropdown_menu,
|
||||||
|
@ -77,6 +78,7 @@ const reducers = {
|
||||||
group_lists,
|
group_lists,
|
||||||
group_editor,
|
group_editor,
|
||||||
sidebar,
|
sidebar,
|
||||||
|
patron,
|
||||||
};
|
};
|
||||||
|
|
||||||
export default combineReducers(reducers);
|
export default combineReducers(reducers);
|
||||||
|
|
13
app/gabsocial/reducers/patron.js
Normal file
13
app/gabsocial/reducers/patron.js
Normal file
|
@ -0,0 +1,13 @@
|
||||||
|
import { PATRON_FUNDING_IMPORT } from '../actions/patron';
|
||||||
|
import { Map as ImmutableMap, fromJS } from 'immutable';
|
||||||
|
|
||||||
|
const initialState = ImmutableMap();
|
||||||
|
|
||||||
|
export default function patron(state = initialState, action) {
|
||||||
|
switch(action.type) {
|
||||||
|
case PATRON_FUNDING_IMPORT:
|
||||||
|
return state.set('funding', fromJS(action.funding));
|
||||||
|
default:
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
};
|
5
app/soapbox/config.js
Normal file
5
app/soapbox/config.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
export default {
|
||||||
|
features: {
|
||||||
|
patron: true,
|
||||||
|
},
|
||||||
|
}
|
|
@ -10,6 +10,7 @@ const watchOptions = {};
|
||||||
|
|
||||||
// TODO: Make this configurable
|
// TODO: Make this configurable
|
||||||
const backendUrl = 'http://localhost:4000';
|
const backendUrl = 'http://localhost:4000';
|
||||||
|
const patronUrl = 'http://localhost:5000';
|
||||||
|
|
||||||
if (process.env.VAGRANT) {
|
if (process.env.VAGRANT) {
|
||||||
// If we are in Vagrant, we can't rely on inotify to update us with changed
|
// If we are in Vagrant, we can't rely on inotify to update us with changed
|
||||||
|
@ -68,6 +69,7 @@ module.exports = merge(sharedConfig, {
|
||||||
'/socket': backendUrl,
|
'/socket': backendUrl,
|
||||||
'/oauth/revoke': backendUrl,
|
'/oauth/revoke': backendUrl,
|
||||||
'/.well-known/webfinger': backendUrl,
|
'/.well-known/webfinger': backendUrl,
|
||||||
|
'/patron': patronUrl,
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in a new issue