diff --git a/README.md b/README.md
new file mode 100644
index 0000000000000000000000000000000000000000..74ce25c49222822b0298f4f455689a5d27eac69a
--- /dev/null
+++ b/README.md
@@ -0,0 +1,346 @@
+# liowebrtc
+A react-compatible webRTC library that makes it easy to bake peer to peer communication into react components.
+
+LioWebRTC was built on SimpleWebRTC, and modified to be compatible with React.
+
+## Usage
+
+### Import LioWebRTC
+```js
+import LioWebRTC from 'liowebrtc';
+```
+
+### Create LioWebRTC instance
+
+```js
+const webrtc = new LioWebRTC({
+    // The local video reference set within your render function, or an element ID
+    localVideoEl: 'localVid',
+    // Immediately request camera access
+    autoRequestMedia: true
+});
+```
+
+### Join a room once it's ready
+
+```js
+webrtc.on('readyToCall', function () {
+    // Create or join a room with any name
+    webrtc.joinRoom('your room name');
+});
+```
+
+
+## Example
+
+### Video Chat Component
+```jsx
+import React, { Component } from 'react';
+import LioWebRTC from 'liowebrtc';
+import 'attachMediaStream' from 'attachmediastream';
+
+class Party extends Component {
+  constructor(props) {
+    super(props);
+    this.state = {
+      nick: this.props.nick,
+      peers: [],
+      roomID: `party-${this.props.roomName}`,
+      muted: false,
+      camPaused: false
+    };
+    this.remoteVideos = {};
+  }
+
+  componentDidMount() {
+    this.webrtc = new LioWebRTC({
+      // The url for your signaling server
+      url: 'https://sandbox.simplewebrtc.com:443/',
+      // The local video reference set within your render function
+      localVideoEl: this.localVid,
+      // Immediately request camera access
+      autoRequestMedia: true,
+      // Optional: The nickname of the peer in the room
+      nick: this.state.nick,
+    });
+
+    this.webrtc.on('videoAdded', this.addVideo);
+    this.webrtc.on('videoRemoved', this.removeVideo);
+    this.webrtc.on('readyToCall', this.readyToCall);
+    this.webrtc.on('iceFailed', this.handleConnectionError);
+    this.webrtc.on('connectivityError', this.handleConnectionError);
+  }
+  
+  addVideo = (stream, peer) => {
+    this.setState({
+      peers: [...this.state.peers, peer]
+    }, () => {
+      attachMediaStream(stream, this.remoteVideos[peer.id]);
+    });
+  }
+  
+  removeVideo = (video, peer) => {
+    this.setState({
+      peers: this.state.peers.filter(p => p.id)
+    });
+  }
+  
+  handleConnectionError = (peer) => {
+    const pc = peer.pc;
+    console.log('had local relay candidate', pc.hadLocalRelayCandidate);
+    console.log('had remote relay candidate', pc.hadRemoteRelayCandidate);
+  }
+  
+  readyToCall = () => {
+    // Starts the process of joining a room.
+    this.webrtc.joinRoom(this.state.roomID, (err, desc) => {
+    });
+  }
+  
+  // Show fellow peers in the room
+  generateRemotes = () => this.state.peers.map((p) => (
+    <PeerTile key={p.id}>
+      <VideoContainer id={`container_${this.webrtc.getDomId(p)}`}>
+        <video
+          key={this.webrtc.getDomId(p)}
+          id={this.webrtc.getDomId(p)}
+          ref={(v) => this.remoteVideos[p.id] = v}
+          />
+      </VideoContainer>
+      <PeerInfo>
+        <p>{p.nick}</p>
+      </PeerInfo>
+    </PeerTile>
+    ));
+  
+  disconnect = () => {
+    this.webrtc.stopLocalVideo();
+    this.webrtc.leaveRoom();
+    this.webrtc.disconnect();
+  }
+
+  componentWillUnmount() {
+    this.disconnect();
+  }
+  
+  render() {
+    return (
+      <Wrapper>
+        <PeerTile>
+          <VideoContainer>
+            <video
+              className={styles.vid}
+              // Important: The local video element needs to have both an ID and ref
+              id="localVideo"
+              ref={(vid) => { this.localVid = vid; }}
+            />
+          </VideoContainer>
+          <PeerInfo>
+            <p>{this.state.nick}</p>
+          </PeerInfo>
+        </PeerTile>
+
+        <div id="remoteVideos">
+          {this.generateRemotes()}
+        </div>
+      </Wrapper>
+    );
+  }
+}
+
+export default Party;
+  
+```
+
+## API
+
+### Constructor
+
+`new LioWebRTC(options)`
+
+- `object options`
+  - `string url` - url for signaling server.
+  - `bool debug` - *optional* flag to set the instance to debug mode
+  - `[string|DomElement] localVideoEl` - ID or Element to contain the local video
+  element
+  - `bool autoRequestMedia` - *optional(=true)* option to automatically request
+  user media. Use `true` to request automatically, or `false` to request media
+  later with `startLocalVideo`
+  - `bool enableDataChannels` *optional(=true)* option to enable/disable data
+  channels (used for volume levels or direct messaging)
+  - `bool autoRemoveVideos` - *optional(=true)* option to automatically remove
+  video elements when streams are stopped.
+  - `bool adjustPeerVolume` - *optional(=true)* option to reduce peer volume
+  when the local participant is speaking
+  - `number peerVolumeWhenSpeaking` - *optional(=.0.25)* value used in
+  conjunction with `adjustPeerVolume`. Uses values between 0 and 1.
+  - `object media` - media options to be passed to `getUserMedia`. Defaults to
+  `{ video: true, audio: true }`. Valid configurations described
+  [on MDN](https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia)
+  with official spec
+  [at w3c](http://w3c.github.io/mediacapture-main/#dom-mediadevices-getusermedia).
+  - `object receiveMedia` - *optional* RTCPeerConnection options. Defaults to
+  `{ offerToReceiveAudio: 1, offerToReceiveVideo: 1 }`.
+  - `object localVideo` - *optional* options for attaching the local video
+  stream to the page. Defaults to
+  ```javascript
+  {
+      autoplay: true, // automatically play the video stream on the page
+      mirror: true, // flip the local video to mirror mode (for UX)
+      muted: true // mute local video stream to prevent echo
+  }
+  ```
+  - `object logger` - *optional* alternate logger for the instance; any object
+  that implements `log`, `warn`, and `error` methods.
+
+### Fields
+
+`capabilities` - the webrtcsupport module that returns an object that
+describes browser capabilities.
+
+`config` - the configuration options extended from options passed to the
+constructor
+
+`connection` - the socket (or alternate) signaling connection
+
+`webrtc` - the underlying WebRTC session manager
+
+### Events
+
+To set up event listeners, use the LioWebRTC instance created with the
+constructor. Example:
+
+```javascript
+webrtc.on('connectionReady', (sessionId) => {
+    // ...
+})
+```
+
+`'connectionReady', sessionId` - emitted when the signaling connection emits the
+`connect` event, with the unique id for the session.
+
+`'createdPeer', peer` - emitted three times:
+
+- when joining a room with existing peers, once for each peer
+- when a new peer joins a joined room
+- when sharing screen, once for each peer
+
+- `peer` - the object representing the peer and underlying peer connection
+
+`'stunservers', [...args]` - emitted when the signaling connection emits the
+same event
+
+`'turnservers', [...args]` - emitted when the signaling connection emits the
+same event
+
+`'localScreenAdded', el` - emitted after triggering the start of screen sharing
+
+- `el` the element that contains the local screen stream
+
+`'leftRoom', roomName` - emitted after successfully leaving the current room,
+ending all peers, and stopping the local screen stream
+
+`'videoAdded', videoEl, peer` - emitted when a peer stream is added
+
+- `videoEl` - the video element associated with the stream that was added
+- `peer` - the peer associated with the stream that was added
+
+`'videoRemoved', videoEl, peer` - emitted when a peer stream is removed
+
+- `videoEl` - the video element associated with the stream that was removed
+- `peer` - the peer associated with the stream that was removed
+
+### Methods
+
+`createRoom(name, callback)` - emits the `create` event on the connection with
+`name` and (if provided) invokes `callback` on response
+
+`joinRoom(name, callback)` - joins the conference in room `name`. Callback is
+invoked with `callback(err, roomDescription)` where `roomDescription` is yielded
+by the connection on the `join` event. See [signalmaster](https://github.com/andyet/signalmaster) for more details.
+
+`startLocalVideo()` - starts the local media with the `media` options provided
+in the config passed to the constructor
+
+`testReadiness()` - tests that the connection is ready and that (if media is
+enabled) streams have started
+
+`mute()` - mutes the local audio stream for all peers (pauses sending audio)
+
+`unmute()` - unmutes local audio stream for all peers (resumes sending audio)
+
+`pauseVideo()` - pauses sending video to peers
+
+`resumeVideo()` - resumes sending video to all peers
+
+`pause()` - pauses sending audio and video to all peers
+
+`resume()` - resumes sending audio and video to all peers
+
+`sendToAll(messageType, payload)` - broadcasts a message to all peers in the
+room via the signaling channel (websocket)
+
+- `string messageType` - the key for the type of message being sent
+- `object payload` - an arbitrary value or object to send to peers
+
+`sendDirectlyToAll(messageType, payload, channel)` - broadcasts a message
+to all peers in the room via a dataChannel
+
+- `string channel` - (optional) the label for the dataChannel to send on
+- `string messageType` - the key for the type of message being sent
+- `object payload` - an arbitrary value or object to send to peers
+
+`getPeers(sessionId, type)` - returns all peers by `sessionId` and/or `type`
+
+`shareScreen(callback)` - initiates screen capture request to browser, then
+adds the stream to the conference
+
+`getLocalScreen()` - returns the local screen stream
+
+`stopScreenShare()` - stops the screen share stream and removes it from the room
+
+`stopLocalVideo()` - stops all local media streams
+
+`setVolumeForAll(volume)` - used to set the volume level for all peers
+
+- `volume` - the volume level, between 0 and 1
+
+`leaveRoom()` - leaves the currently joined room and stops local screen share
+
+`disconnect()` - calls `disconnect` on the signaling connection and deletes it
+
+`handlePeerStreamAdded(peer)` - used internally to attach media stream to the
+DOM and perform other setup
+
+`handlePeerStreamRemoved(peer)` - used internally to remove the video container
+from the DOM and emit `videoRemoved`
+
+`getDomId(peer)` - used internally to get the DOM id associated with a peer
+
+`getEl(idOrEl)` - helper used internally to get an element where `idOrEl` is
+either an element, or an id of an element
+
+`getLocalVideoContainer()` - used internally to get the container that will hold
+the local video element
+
+`getRemoteVideoContainer()` - used internally to get the container that holds
+the remote video elements
+
+
+
+## Signaling
+
+### Connection
+
+For signaling, LioWebRTC uses [socket.io](http://socket.io/) to
+communicate with the signaling server. The connection object comes with these methods:
+
+- `on(ev, fn)` - A method to invoke `fn` when event `ev` is triggered
+- `emit()` - A method to send/emit arbitrary arguments on the connection
+- `getSessionId()` - A method to get a unique session Id for the connection
+- `disconnect()` - A method to disconnect the connection
+
+### Signaling Server
+
+LioWebRTC uses the signaling server provided for testing purposes by SimpleWebRTC. To use this in production,
+you will need to set up your own instance of [signalmaster](https://github.com/andyet/signalmaster).