diff --git a/src/components/transaction/container/transactionContainer.js b/src/components/transaction/container/transactionContainer.js index a6d1899b6..6cd96f355 100644 --- a/src/components/transaction/container/transactionContainer.js +++ b/src/components/transaction/container/transactionContainer.js @@ -1,4 +1,5 @@ import React, { PureComponent } from 'react'; +import { connect } from 'react-redux'; // Component import TransactionView from '../view/transactionView'; @@ -21,10 +22,14 @@ class TransactionContainer extends PureComponent { // Component Functions render() { - const { walletData } = this.props; + const { walletData, steemPerMVests } = this.props; - return ; + return ; } } -export default TransactionContainer; +const mapStateToProps = state => ({ + steemPerMVests: state.account.globalProps.steemPerMVests, +}); + +export default connect(mapStateToProps)(TransactionContainer); diff --git a/src/components/transaction/view/transactionView.js b/src/components/transaction/view/transactionView.js index 14f11e138..1b72e9e46 100644 --- a/src/components/transaction/view/transactionView.js +++ b/src/components/transaction/view/transactionView.js @@ -36,7 +36,7 @@ class TransactionView extends PureComponent { walletData: { transactions }, intl, intl: { formatNumber }, - walletData, + steemPerMVests, } = this.props; return ( @@ -53,16 +53,17 @@ class TransactionView extends PureComponent { {transactions && transactions.map((item, index) => { - const transactionData = groomingTransactionData(item, walletData, formatNumber); + const transactionData = groomingTransactionData(item, steemPerMVests, formatNumber); const value = transactionData.value.split(' '); + return ( {(!!transactionData.details || !!transactionData.memo) && ( ({ upvotePercent: state.application.upvotePercent, pinCode: state.account.pin, currentAccount: state.account.currentAccount, + globalProps: state.account.globalProps, }); export default connect(mapStateToProps)(UpvoteContainer); diff --git a/src/components/upvote/view/upvoteView.js b/src/components/upvote/view/upvoteView.js index 26159790c..d56a95331 100644 --- a/src/components/upvote/view/upvoteView.js +++ b/src/components/upvote/view/upvoteView.js @@ -6,12 +6,17 @@ import { injectIntl } from 'react-intl'; import { Popover, PopoverController } from 'react-native-modal-popover'; import Slider from 'react-native-slider'; +// Utils +import parseToken from '../../../utils/parseToken'; +import { vestsToRshares } from '../../../utils/conversions'; + // Components import { Icon } from '../../icon'; import { PulseAnimation } from '../../animations'; import { TextButton } from '../../buttons'; + // STEEM -import { upvoteAmount, vote } from '../../../providers/steem/dsteem'; +import { vote } from '../../../providers/steem/dsteem'; // Styles import styles from './upvoteStyles'; @@ -53,21 +58,22 @@ class UpvoteView extends Component { // Component Functions _calculateEstimatedAmount = async () => { - const { currentAccount } = this.props; - // Calculate total vesting shares + const { currentAccount, globalProps } = this.props; + if (currentAccount) { const { sliderValue } = this.state; - const totalVests = parseFloat(currentAccount.vesting_shares) - + parseFloat(currentAccount.received_vesting_shares) - - parseFloat(currentAccount.delegated_vesting_shares); + const { + fundRecentClaims, fundRewardBalance, base, quote, + } = globalProps; - const finalVest = totalVests * 1e6; + const votingPower = currentAccount.voting_power; + const totalVests = parseToken(currentAccount.vesting_shares) + + parseToken(currentAccount.received_vesting_shares) + - parseToken(currentAccount.delegated_vesting_shares); + const votePct = sliderValue * 10000; - const power = (currentAccount.voting_power * (sliderValue * 10000)) / 10000 / 50; - - const rshares = (power * finalVest) / 10000; - - const estimated = await upvoteAmount(rshares); + const rShares = vestsToRshares(totalVests, votingPower, votePct); + const estimated = (rShares / fundRecentClaims) * fundRewardBalance * (base / quote); this.setState({ amount: estimated.toFixed(5), @@ -252,18 +258,18 @@ class UpvoteView extends Component { { - closePopover(); - this._upvoteContent(); - }} + closePopover(); + this._upvoteContent(); + }} style={styles.upvoteButton} > + size={20} + style={[styles.upvoteIcon, { color: '#007ee5' }]} + active={!isLoggedIn} + iconType={iconType} + name={iconName} + /> {_amount} { - this.setState({ sliderValue: value }, () => { - this._calculateEstimatedAmount(); - }); - }} + this.setState({ sliderValue: value }, () => { + this._calculateEstimatedAmount(); + }); + }} /> {_percent} diff --git a/src/components/wallet/container/walletContainer.js b/src/components/wallet/container/walletContainer.js index 4543e5f59..7c47c2b0e 100644 --- a/src/components/wallet/container/walletContainer.js +++ b/src/components/wallet/container/walletContainer.js @@ -47,8 +47,8 @@ class WalletContainer extends Component { // Components functions _getWalletData = async (selectedUser) => { - const { setEstimatedWalletValue } = this.props; - const walletData = await groomingWalletData(selectedUser); + const { setEstimatedWalletValue, globalProps } = this.props; + const walletData = await groomingWalletData(selectedUser, globalProps); this.setState({ walletData }); setEstimatedWalletValue(walletData.estimatedValue); @@ -133,6 +133,7 @@ const mapStateToProps = state => ({ currentAccount: state.account.currentAccount, pinCode: state.account.pin, isDarkTheme: state.application.isDarkTheme, + globalProps: state.account.globalProps, }); export default injectIntl(connect(mapStateToProps)(WalletContainer)); diff --git a/src/providers/steem/dsteem.js b/src/providers/steem/dsteem.js index f82ed366f..a74c20d6d 100644 --- a/src/providers/steem/dsteem.js +++ b/src/providers/steem/dsteem.js @@ -10,13 +10,12 @@ import { decryptKey } from '../../utils/crypto'; import { parsePosts, parsePost, parseComments } from '../../utils/postParser'; import { getName, getAvatar } from '../../utils/user'; import { getReputation } from '../../utils/reputation'; +import parseToken from '../../utils/parseToken'; // Constant import AUTH_TYPE from '../../constants/authType'; const DEFAULT_SERVER = 'https://api.steemit.com'; -let rewardFund = null; -let medianPrice = null; let client = new Client(DEFAULT_SERVER); const _getClient = async () => { @@ -35,6 +34,50 @@ _getClient(); export const getDigitPinCode = pin => decryptKey(pin, Config.PIN_KEY); +export const getDynamicGlobalProperties = () => client.database.getDynamicGlobalProperties(); + +export const getRewardFund = () => client.database.call('get_reward_fund', ['post']); + +export const getFeedHistory = async () => { + try { + const feedHistory = await client.database.call('get_feed_history'); + return feedHistory; + } catch (error) { + return error; + } +}; + +export const fetchGlobalProps = async () => { + let globalDynamic; + let feedHistory; + let rewardFund; + + try { + globalDynamic = await getDynamicGlobalProperties(); + feedHistory = await getFeedHistory(); + rewardFund = await getRewardFund(); + } catch (e) { + return; + } + + const steemPerMVests = (parseToken(globalDynamic.total_vesting_fund_steem) + / parseToken(globalDynamic.total_vesting_shares)) + * 1e6; + const base = parseToken(feedHistory.current_median_history.base); + const quote = parseToken(feedHistory.current_median_history.quote); + const fundRecentClaims = rewardFund.recent_claims; + const fundRewardBalance = parseToken(rewardFund.reward_balance); + const globalProps = { + steemPerMVests, + base, + quote, + fundRecentClaims, + fundRewardBalance, + }; + + return globalProps; +}; + /** * @method getAccount get account data * @param user username @@ -127,22 +170,6 @@ export const getFollows = user => new Promise((resolve, reject) => { }); }); -/** - * @method getFollowers - * @param user username - * TODO: Pagination - */ -// export const getFollowers = (user, limit = 100) => new Promise((resolve, reject) => { -// client -// .call('follow_api', 'get_followers', [user, '', 'blog', limit]) -// .then((result) => { -// resolve(result); -// }) -// .catch((err) => { -// reject(err); -// }); -// }); - /** * @method getFollowing * @param user username @@ -426,18 +453,12 @@ export const vote = async (currentAccount, pin, author, permlink, weight) => { * @method upvoteAmount estimate upvote amount */ export const upvoteAmount = async (input) => { - if (!rewardFund || !medianPrice) { - rewardFund = await client.database.call('get_reward_fund', ['post']); + let medianPrice; + const rewardFund = await getRewardFund(); - await client.database - .getCurrentMedianHistoryPrice() - .then((res) => { - medianPrice = res; - }) - .catch((err) => { - // reject(err); - }); - } + await client.database.getCurrentMedianHistoryPrice().then((res) => { + medianPrice = res; + }); const estimated = (input / parseFloat(rewardFund.recent_claims)) * parseFloat(rewardFund.reward_balance) @@ -563,24 +584,6 @@ export const delegate = (data, activeKey) => { }); }; -export const globalProps = async () => { - try { - const globalProperties = await client.database.getDynamicGlobalProperties(); - return globalProperties; - } catch (error) { - return error; - } -}; - -export const getFeedHistory = async () => { - try { - const feedHistory = await client.database.call('get_feed_history'); - return feedHistory; - } catch (error) { - return error; - } -}; - export const transferToVesting = (data, activeKey) => { const privateKey = PrivateKey.fromString(activeKey); @@ -832,15 +835,4 @@ const getAnyPrivateKey = (local, pin) => { } return false; - - // ['postingKey', 'activeKey'].forEach((key) => { - // console.log('key :', key); - // console.log('pin :', pin); - // console.log('local[key] :', local[key]); - // if (key) { - // const privateKey = decryptKey(local[key], pin); - // console.log('privateKey :', privateKey); - // return true; - // } - // }); }; diff --git a/src/redux/actions/accountAction.js b/src/redux/actions/accountAction.js index 90b3dbaa7..ba9421ca6 100644 --- a/src/redux/actions/accountAction.js +++ b/src/redux/actions/accountAction.js @@ -1,24 +1,18 @@ -import { getUser } from '../../providers/steem/dsteem'; +import { fetchGlobalProps } from '../../providers/steem/dsteem'; import { - FETCH_ACCOUNT_FAIL, - FETCHING_ACCOUNT, ADD_OTHER_ACCOUNT, + FETCH_ACCOUNT_FAIL, + REMOVE_OTHER_ACCOUNT, + SET_GLOBAL_PROPS, + SET_PIN_CODE, UPDATE_CURRENT_ACCOUNT, UPDATE_UNREAD_ACTIVITY_COUNT, - REMOVE_OTHER_ACCOUNT, - SET_PIN_CODE, } from '../constants/constants'; -export const fetchAccountFromSteem = (username, password) => (dispatch) => { - dispatch({ type: FETCHING_ACCOUNT }); - - return getUser(username) - .then(res => dispatch({ - type: UPDATE_CURRENT_ACCOUNT, - payload: { ...res, password }, - })) - .catch(err => dispatch({ type: FETCH_ACCOUNT_FAIL, payload: err })); -}; +export const fetchGlobalProperties = () => dispatch => fetchGlobalProps().then(res => dispatch({ + type: SET_GLOBAL_PROPS, + payload: { ...res }, +})); export const updateCurrentAccount = data => ({ type: UPDATE_CURRENT_ACCOUNT, @@ -49,3 +43,8 @@ export const setPinCode = data => ({ type: SET_PIN_CODE, payload: data, }); + +export const setGlobalProps = data => ({ + type: SET_GLOBAL_PROPS, + payload: data, +}); diff --git a/src/redux/constants/constants.js b/src/redux/constants/constants.js index 51cf422b5..a69f73e32 100644 --- a/src/redux/constants/constants.js +++ b/src/redux/constants/constants.js @@ -24,11 +24,12 @@ export const SET_UPVOTE_PERCENT = 'SET_UPVOTE_PERCENT'; // Accounts export const ADD_OTHER_ACCOUNT = 'ADD_OTHER_ACCOUNT'; export const FETCH_ACCOUNT_FAIL = 'FETCH_ACCOUNT_FAIL'; -export const FETCHING_ACCOUNT = 'FETCHING_ACCOUNT'; export const REMOVE_OTHER_ACCOUNT = 'REMOVE_OTHER_ACCOUNT'; export const UPDATE_CURRENT_ACCOUNT = 'UPDATE_CURRENT_ACCOUNT'; export const UPDATE_UNREAD_ACTIVITY_COUNT = 'UPDATE_UNREAD_ACTIVITY_COUNT'; export const SET_PIN_CODE = 'SET_PIN_CODE'; +export const SET_GLOBAL_PROPS = 'SET_GLOBAL_PROPS'; +export const FETCH_GLOBAL_PROPS = 'FETCH_GLOBAL_PROPS'; // UI export const IS_COLLAPSE_POST_BUTTON = 'IS_COLLAPSE_POST_BUTTON'; diff --git a/src/redux/reducers/accountReducer.js b/src/redux/reducers/accountReducer.js index 4de63d4f3..0d8685f1b 100644 --- a/src/redux/reducers/accountReducer.js +++ b/src/redux/reducers/accountReducer.js @@ -7,6 +7,7 @@ import { REMOVE_OTHER_ACCOUNT, LOGOUT_FAIL, SET_PIN_CODE, + SET_GLOBAL_PROPS, } from '../constants/constants'; const initialState = { @@ -29,6 +30,12 @@ export default function (state = initialState, action) { errorMessage: null, }; + case SET_GLOBAL_PROPS: + return { + ...state, + globalProps: action.payload, + }; + case FETCH_ACCOUNT_FAIL: return { ...state, diff --git a/src/screens/application/container/applicationContainer.js b/src/screens/application/container/applicationContainer.js index e465ffd1d..16310eaca 100644 --- a/src/screens/application/container/applicationContainer.js +++ b/src/screens/application/container/applicationContainer.js @@ -5,6 +5,7 @@ import { addLocaleData } from 'react-intl'; import Config from 'react-native-config'; import AppCenter from 'appcenter'; import { NavigationActions } from 'react-navigation'; +import { bindActionCreators } from 'redux'; // Constants import en from 'react-intl/locale-data/en'; @@ -37,6 +38,7 @@ import { updateCurrentAccount, updateUnreadActivityCount, removeOtherAccount, + fetchGlobalProperties, } from '../../../redux/actions/accountAction'; import { activeApplication, @@ -69,8 +71,11 @@ class ApplicationContainer extends Component { componentDidMount = async () => { BackHandler.addEventListener('hardwareBackPress', this._onBackPress); + this._refreshGlobalProps(); await this._getUserData(); await this._getSettings(); + + this.globalInterval = setInterval(this._refreshGlobalProps, 60000); }; componentWillReceiveProps(nextProps) { @@ -87,6 +92,7 @@ class ApplicationContainer extends Component { componentWillUnmount() { BackHandler.removeEventListener('hardwareBackPress', this.onBackPress); + clearInterval(this.globalInterval); } _onBackPress = () => { @@ -100,6 +106,12 @@ class ApplicationContainer extends Component { return true; }; + _refreshGlobalProps = () => { + const { actions } = this.props; + + actions.fetchGlobalProperties(); + }; + _getUserData = async () => { const { dispatch, pinCode } = this.props; let realmData = []; @@ -280,20 +292,25 @@ class ApplicationContainer extends Component { } } -const mapStateToProps = state => ({ - // Application - isDarkTheme: state.application.isDarkTheme, - selectedLanguage: state.application.language, - notificationSettings: state.application.isNotificationOpen, - isLogingOut: state.application.isLogingOut, - isLoggedIn: state.application.isLoggedIn, - nav: state.nav.routes, +export default connect( + state => ({ + // Application + isDarkTheme: state.application.isDarkTheme, + selectedLanguage: state.application.language, + notificationSettings: state.application.isNotificationOpen, + isLogingOut: state.application.isLogingOut, + isLoggedIn: state.application.isLoggedIn, - // Account - unreadActivityCount: state.account.currentAccount.unread_activity_count, - currentAccount: state.account.currentAccount, - otherAccounts: state.account.otherAccounts, - pinCode: state.account.pin, -}); - -export default connect(mapStateToProps)(ApplicationContainer); + // Account + unreadActivityCount: state.account.currentAccount.unread_activity_count, + currentAccount: state.account.currentAccount, + otherAccounts: state.account.otherAccounts, + pinCode: state.account.pin, + }), + (dispatch, props) => ({ + dispatch, + actions: { + ...bindActionCreators({ fetchGlobalProperties }, dispatch), + }, + }), +)(ApplicationContainer); \ No newline at end of file diff --git a/src/utils/conversions.js b/src/utils/conversions.js index 495b7a0c2..d62bfcfb2 100644 --- a/src/utils/conversions.js +++ b/src/utils/conversions.js @@ -1 +1,7 @@ export const vestsToSp = (vests, steemPerMVests) => (vests / 1e6) * steemPerMVests; + +export const vestsToRshares = (vests, votingPower, votePerc) => { + const vestingShares = parseInt(vests * 1e6, 10); + const power = (votingPower * votePerc) / 1e4 / 50 + 1; + return (power * vestingShares) / 1e4; +}; diff --git a/src/utils/wallet.js b/src/utils/wallet.js index 0cd9243ea..f9ee862a4 100644 --- a/src/utils/wallet.js +++ b/src/utils/wallet.js @@ -1,10 +1,11 @@ import parseDate from './parseDate'; import parseToken from './parseToken'; import { vestsToSp } from './conversions'; -import { globalProps, getState, getFeedHistory } from '../providers/steem/dsteem'; +import { getState, getFeedHistory } from '../providers/steem/dsteem'; -export const groomingTransactionData = (transaction, walletData, formatNumber) => { - if (!transaction || !walletData) { + +export const groomingTransactionData = (transaction, steemPerMVests, formatNumber) => { + if (!transaction || !steemPerMVests) { return []; } @@ -23,7 +24,7 @@ export const groomingTransactionData = (transaction, walletData, formatNumber) = const { reward } = opData; const { comment_author: commentAuthor, comment_permlink: commentPermlink } = opData; - result.value = `${formatNumber(vestsToSp(parseToken(reward), walletData.steemPerMVests), { + result.value = `${formatNumber(vestsToSp(parseToken(reward), steemPerMVests), { minimumFractionDigits: 3, })} SP`; result.details = `@${commentAuthor}/${commentPermlink}`; @@ -41,7 +42,7 @@ export const groomingTransactionData = (transaction, walletData, formatNumber) = sbdPayout = formatNumber(parseToken(sbdPayout), { minimumFractionDigits: 3 }); steemPayout = formatNumber(parseToken(steemPayout), { minimumFractionDigits: 3 }); vestingPayout = formatNumber( - vestsToSp(parseToken(vestingPayout), walletData.steemPerMVests), + vestsToSp(parseToken(vestingPayout), steemPerMVests), { minimumFractionDigits: 3 }, ); @@ -59,7 +60,7 @@ export const groomingTransactionData = (transaction, walletData, formatNumber) = rewardSdb = formatNumber(parseToken(rewardSdb), { minimumFractionDigits: 3 }); rewardSteem = formatNumber(parseToken(rewardSteem), { minimumFractionDigits: 3 }); - rewardVests = formatNumber(vestsToSp(parseToken(rewardVests), walletData.steemPerMVests), { + rewardVests = formatNumber(vestsToSp(parseToken(rewardVests), steemPerMVests), { minimumFractionDigits: 3, }); @@ -83,7 +84,7 @@ export const groomingTransactionData = (transaction, walletData, formatNumber) = let { vesting_shares: opVestingShares } = opData; opVestingShares = parseToken(opVestingShares); - result.value = `${formatNumber(vestsToSp(opVestingShares, walletData.steemPerMVests), { + result.value = `${formatNumber(vestsToSp(opVestingShares, steemPerMVests), { minimumFractionDigits: 3, })} SP`; result.icon = 'attach-money'; @@ -101,14 +102,13 @@ export const groomingTransactionData = (transaction, walletData, formatNumber) = return result; }; -export const groomingWalletData = async (user) => { +export const groomingWalletData = async (user, globalProps) => { const walletData = {}; if (!user) { return walletData; } - const global = await globalProps(); const state = await getState(`/@${user.name}/transfers`); const { accounts } = state; @@ -133,9 +133,7 @@ export const groomingWalletData = async (user) => { const base = parseToken(feedHistory.current_median_history.base); const quote = parseToken(feedHistory.current_median_history.quote); - walletData.steemPerMVests = ( - parseToken(global.total_vesting_fund_steem) / parseToken(global.total_vesting_shares) - ) * 1e6; + walletData.steemPerMVests = globalProps.steemPerMVests; walletData.estimatedValue = ( vestsToSp(walletData.vestingShares, walletData.steemPerMVests) * (base / quote)