Skip to content

Latest commit

 

History

History
255 lines (218 loc) · 7.1 KB

README.md

File metadata and controls

255 lines (218 loc) · 7.1 KB

slides from presentation

Prerequisites

Node

Install node.js Running the installer should install node and npm
You should be able to run 'node --version && npm --version' from your command line without errors.

create-react-app

Create-react-app is a handy utility for creating React apps with zero confiugration. We can install it and use it to create ourselves a react application below.

npm install -g create-react-app

create-react-app my-app
cd my-app/
npm start

Folder Structure

After creation, your project should look like this:

my-app/
  README.md
  node_modules/
  package.json
  public/
    index.html
    favicon.ico
  src/
    App.css
    App.js
    App.test.js
    index.css
    index.js
    logo.svg

In the project directory, you can run:

npm start

Runs the app in the development mode.
Open http://localhost:3000 to view it in the browser.

The page will reload if you make edits.
You will also see any lint errors in the console.

stage-1 - Display a list of podcasts

We're going to be editing src/app.js.

This is where our main component is that is being rendered to the screen when we run npm start.

class App extends Component {
  constructor() {
    super();
    // Here we bind the current context so we can call
    // setState on our component from componentDidMount
    this.componentDidMount = this.componentDidMount.bind(this);
    // Configure our initial state
    this.state = {
      feed: {
        item: []
      },
      loading: true
    }
  }
  // this is called when our component 'loads'
  // so this will be when we fetch items to display
  async componentDidMount(){ 
    let feed = await loadBloombergFeed();
    // Pick the first 20  
    feed.item = feed.item.slice(1,20);
    // Update our state with the items and set loading to false
    this.setState({feed: feed, loading: false});
    console.log(feed);
  }
  render() {
    return (
      return (
        <div className="App">
          <div className="App-header">
            <img src={logo} className="App-logo" alt="logo" />
            <h2>Welcome to React</h2>
          </div>
          <button type="button" onClick={this.playAll} className="btn btn-default">Play All</button>
          <div className="App-intro row">
              {this.state.feed.item
                .map((item, i) => (
                    <div>
                      <h1>{item.title[0]}</h1>
                      <div style={{'textOverflow': 'ellipsis'}}>{item['itunes:summary'][0]}</div>
                      <audio className="player" controls>
                        <source src={item.link[0]} type='audio/mp3' />
                      </audio>
                    </div>
              ))}
          </div>
        </div>
    )
  }

For each one of our items in this.state.feed.item we will render a div with the title and an audio element with the mp3 as the source. You will not see anything currently, because we never update our state with some actual items.

Fetching some podcasts

We can fetch some podcasts with the below function and we call

async function loadBloombergFeed() {
  return new Promise(function(resolve, reject) {
    // Here we can resolve with some 'fake' data if we want
    // resolve(dummyData);
    let x = new XMLHttpRequest();
    x.open("GET", "https://crossorigin.me/https://www.bloomberg.com/feeds/podcasts/etf_report.xml", true);
    x.onreadystatechange = function () {
      if (x.readyState === 4 && x.status === 200)
      {
        var doc = x.response;
        parseString(doc, function(err, result) {
          if(!err) {
            resolve(result.rss.channel[0]);
          } else {
            reject(err);
          }
        })
      }
    };
    x.send(null);
  }) 
}

Stage-2 - Filtering results by keyword & Adding Filter Animations

Input fields (Forms) are different from other components because they can be mutated by user actions. link to documentation.
We can manage the state of our input field through the state of our component.

We can add this input component to our render method.

<input
  className="form-control"
  type="text"
  value={this.state.filterValue}
  onChange={this.handleChange}
/>

// Add this to our constructor
constructor() {
...
  this.handleChange = this.handleChange.bind(this);
...
}

//Add this function after our constructor 
handleChange(event) {
  this.setState({filterValue: event.target.value});
}

Our state is now updated with our filterValue and we can now conditionally render based on this.state.filterValue in our render function.

Before we map our items on our state to React elements, we can first filter by the value.

{this.state.feed.item
  .filter((item, i) => { 
    return this.state.filterValue != "" ? item.title[0].startsWith(this.state.filterValue): true})
  .map((item, i) => {
  ...
  }

React allows us to animate filtering of our items using ReactCSSTransitionGroup.

We can add this to our index.css file.

.example-enter {
  opacity: 0.01;
}

.example-enter.example-enter-active {
  opacity: 1;
  transition: opacity 500ms ease-in;
}

.example-leave {
  opacity: 1;
}

.example-leave.example-leave-active {
  opacity: 0.01;
  transition: opacity 300ms ease-in;
}

And wrap our podcast items in this component. Notice that the transitionName property of the ReactCSSTransitionGroup component corresponds to 'example' prefix in our css.

import ReactCSSTransitionGroup from 'react/lib/ReactCSSTransitionGroup';

...

<ReactCSSTransitionGroup 
  transitionName="example" 
  transitionEnterTimeout={500} 
  transitionLeaveTimeout={300}>
    {this.state.feed.item
        .filter((item, i) => { 
          return this.state.filterValue != "" ? item.title[0].startsWith(this.state.filterValue): true})
        .map((item, i) => (
          <div className="col-md-4 panel panel-defaul" style={{'height': 400, 'overflow': 'hidden'}} key={i}>
            <div className="panel-body">
              <h1>{item.title[0]}</h1>
              <div style={{'textOverflow': 'ellipsis'}}>{item['itunes:summary'][0]}</div>
              <audio className="player" controls>
                <source src={item.link[0]} type='audio/mp3' />
              </audio>
            </div>
          </div>
      ))
    }
</ReactCSSTransitionGroup> 

Adding a loading spinner

We add the loading property to our state; initially set to true and set to false after our loading request finishes.

import Spinner from 'react-spinkit';
...
<div className="App-intro row">
  {(() => {
    if (this.state.loading) {
      return <Spinner style={{paddingTop: 20}} spinnerName="three-bounce" />
    }
  })()
  ...
</div>

Fun Challenge

Add a function to the component called 'playAll' that toggles starting and stopping all of the audio elements in the page. Add this to the 'play all' button. Hint - you can get all of the audio elements on the page using the following.

document.getElementsByClassName("player")