Skip to content

Latest commit

 

History

History
174 lines (128 loc) · 4.15 KB

README.md

File metadata and controls

174 lines (128 loc) · 4.15 KB

react-redux-polling

Just an example of polling using React-Redux using ReactiveX

example

This is an example container that queries the list of Users periodically.

The polling container executes the redux action pollUsers every 2 seconds.

To use the PollingContainer component you must install https://www.npmjs.com/package/rxjs

polling cointaner

The polling container basically creates a observable subscription when mounting and remove it when unmount.

import React, {Component} from "react";
import { timer } from 'rxjs';

/*
    props:
    - pollAction: Redux action that will execute very period
    - onDataChanged: Function to execute when detect that data changed, i.e alert that.
    - dataChanged: Boolean that determines if data has changed
    - dueTime: Delay between executions
    - periodOfScheduler Execution frecuency in ms
 */
export default class PollingContainer extends Component{

    componentDidMount() {
        this.timmer = timer(this.props.dueTime,this.props.periodOfScheduler);
        this.subscription = this.timmer.subscribe(async (val)=> {
            console.log("Polling ..." + val);
            await this.props.pollAction();
            if(this.props.dataChanged){
                await this.props.onDataChanged();
            }
        })
    }

    componentWillUnmount() {
        this.subscription.unsubscribe();
        this.timmer = null;
    }

    render=()=>{
        return <div>
            {this.props.render()}
        </div>
    }
}

example container

import React, {Component} from "react";
import {connect} from "react-redux";
import {bindActionCreators} from "redux";
import { pollUsers} from "../../actions/userActions";
import PollingContainer from "../../genericContainers/PollingContainer";
import { ToastContainer, toast } from 'react-toastify';
import 'react-toastify/dist/ReactToastify.css';
import UserList from "../../components/users/UserList";


class UsersContainer extends Component{

    render=()=>{
        return <div>
            <PollingContainer
                render={this.renderPolling}
                pollAction={this.props.pollUsers}
                dueTim={0}
                periodOfScheduler={2000}
            />
            <ToastContainer />
            <div className="container-fluid px-5">
                <UserList userList={this.props.users}/>
            </div>
        </div>

    };


    renderPolling=()=>{
        if(this.props.usersChanged){
            toast.info("Users changed")
        }
    };


}

const mapStateToProps = (state) => {
    return {
        users: state.userslReducer.users,
        usersChanged: state.usersReducer.usersChanged
    }
};

const mapDispatchToProps = (dispatch) => {
    const actions = {
        pollUsers: pollUsers
    };
    return bindActionCreators(actions, dispatch)
};

export default connect(mapStateToProps, mapDispatchToProps)(UsersContainer);

action

import axios from 'axios';
import {POLL_USERS_SUCCED} from "./types";


export const pollUsers = () =>
    (dispatch, getState) =>
        new Promise((resolve, reject) =>
            axios.get("http://localhost:8080/users").then(response => {
                 return resolve(dispatch(pollSucceed(response.data, true)))); // change data changed to a function that compares with the previous state
            })
            .catch(error => {
                console.log(JSON.stringify(error));
            })
        );
        
const pollSucceed= (users, dataChanged) => (
        {
            type: POLL_USERS_SUCCED,
            data: {users: users, usersChanged: dataChanged}
        }
)
        

reducer

import createReducer from './helpers/reducerHelper'
import {POLL_USER_SUCCEED} from "../actions/types";

const initialState = {
    users: [],
    usersDataChanged: false
};

const usersReducer = createReducer(initialState,
    {
        [POLL_USERS](state, action) {
            console.log("Dispatched poll")
            return {
                ...state,
                users:  action.data.users,
                usersChanged: action.data.usersChanged,

            };
        },

    });


export default usersReducer;