├── .gitignore ├── .travis.yml ├── README.md ├── demo └── src │ ├── index.js │ └── styles.scss ├── nwb.config.js ├── package.json ├── src ├── OptionsTemplate.js └── index.js └── tests ├── .eslintrc └── index-test.js /.gitignore: -------------------------------------------------------------------------------- 1 | /coverage 2 | /demo/dist 3 | /es6 4 | /lib 5 | /node_modules 6 | /umd 7 | npm-debug.log 8 | -------------------------------------------------------------------------------- /.travis.yml: -------------------------------------------------------------------------------- 1 | sudo: false 2 | 3 | language: node_js 4 | node_js: 5 | - 4.2 6 | 7 | cache: 8 | directories: 9 | - node_modules 10 | 11 | before_install: 12 | - npm install codecov.io coveralls 13 | 14 | after_success: 15 | - cat ./coverage/lcov.info | ./node_modules/codecov.io/bin/codecov.io.js 16 | - cat ./coverage/lcov.info | ./node_modules/coveralls/bin/coveralls.js 17 | 18 | branches: 19 | only: 20 | - master 21 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # react-youtube-autocomplete 2 | A responsive React-based auto-suggest search box for Youtube apps. 3 | 4 | I like to build apps on top of Youtube. Sometimes you need to let users search for videos on Youtube within your app. 5 | Just drop this component into your Youtube-friendly React.js app and you'll get a fully functional auto-suggest-enabled search box. 6 | 7 | ## Demo 8 | 9 | See this [compenent in action](http://hackingbeauty.github.io/react-youtube-autocomplete/) 10 | 11 | ## Installation 12 | 13 | `npm install --save react-youtube-autocomplete` 14 | 15 | ## Features 16 | 17 | - Autocomplete text entry 18 | - Search Youtube based on text input 19 | - Retrieve list of results from Youtube 20 | - Display drop-down list of search results 21 | 22 | ## Usage 23 | 24 | ```js 25 | 50. Number of video search results you want 28 | placeHolder={string} // defaults -> "Search Youtube" 29 | callback={function} // callback to execute when search results are retrieved 30 | className={string} // defaults -> random string 31 | /> 32 | ``` 33 | 34 | ## Example 35 | 36 | ```js 37 | import YoutubeAutocomplete from 'react-youtube-autocomplete'; 38 | 39 | class Example extends React.Component { 40 | render() { 41 | return ( 42 | 47 | ); 48 | } 49 | 50 | _onSearchResultsFound(results) { 51 | // Results is an array of retreived search results 52 | // I use flux, so I dispatch results to an action 53 | flux.actions.showSearchResults(results); 54 | } 55 | } 56 | ``` 57 | 58 | ## License 59 | 60 | MIT 61 | 62 | ## Course 63 | 64 | Are you looking to build a professional app for the Web using React & Redux? 65 | 66 | Check out my course ["How to Write a Single Page Application".](http://www.singlepageapplication.com) 67 | 68 | www.singlepageapplication.com -------------------------------------------------------------------------------- /demo/src/index.js: -------------------------------------------------------------------------------- 1 | import React, { PropTypes } from 'react'; 2 | import {render} from 'react-dom'; 3 | import { Dialog } from 'material-ui'; 4 | import injectTapEventPlugin from 'react-tap-event-plugin'; 5 | import Code from 'react-embed-code'; 6 | 7 | import Component from '../../src'; 8 | 9 | injectTapEventPlugin(); 10 | 11 | require('./styles.scss'); 12 | 13 | const cssDownload = ` 14 | .react-typeahead-options { 15 | margin: 0; 16 | padding: 0; 17 | list-style-type: none; 18 | border: 1px solid #ccc; 19 | cursor: default; 20 | box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2); 21 | z-index: 1; 22 | width: 100%; 23 | } 24 | 25 | .react-typeahead-options { 26 | margin-left: -8px; 27 | } 28 | 29 | .react-typeahead-options li[role="option"] { 30 | padding: 5px; 31 | text-align: left; 32 | cursor: pointer; 33 | } 34 | 35 | .react-typeahead-options li[role="option"][aria-selected="true"] { 36 | background: #00bcd4; 37 | color: white; 38 | font-weight: bold; 39 | } 40 | 41 | .react-typeahead-container { 42 | border: 1px solid #024e6a; 43 | padding: 5px 8px; 44 | border-radius: 0px; 45 | background-color: white; 46 | margin: 0 auto 47 | } 48 | 49 | .react-typeahead-input { 50 | position: relative; 51 | background: white; 52 | outline: none; 53 | width: 100%; 54 | font-size: 24px; 55 | line-height: 30px; 56 | border: none; 57 | } 58 | `; 59 | 60 | const componentEmbed = ` 61 | import YoutubeAutocomplete from 'react-youtube-autocomplete'; 62 | 63 | 68 | `; 69 | 70 | const Demo = React.createClass({ 71 | getInitialState() { 72 | return { 73 | open: false, 74 | searchResults : [] 75 | } 76 | }, 77 | 78 | getFormattedResults(searchResults) { 79 | return searchResults.map(function(result) { 80 | return
  • 81 | 82 | {result.snippet.title} 83 | {result.snippet.title} 84 | 85 |
  • 86 | }); 87 | }, 88 | 89 | showResults(searchResults) { 90 | this.setState({ 91 | open : true, 92 | searchResults : searchResults 93 | }) 94 | }, 95 | 96 | handleClose() { 97 | this.setState({open: false}); 98 | }, 99 | 100 | onRequestClose() { 101 | this.setState({open: false}); 102 | }, 103 | 104 | render() { 105 | var searchResults = this.state.searchResults; 106 | var formattedResults; 107 | 108 | if(searchResults) { 109 | formattedResults = this.getFormattedResults(searchResults); 110 | } 111 | 112 | return
    113 |
    114 |
    115 |
    116 |
    117 |
    118 |
    119 |

    react-youtube-autocomplete

    120 |

    A responsive & React-based auto-suggest search box for Youtube apps

    121 |
    122 | 125 |
    126 |
    127 |
    128 |
    129 | Demo 130 |
    131 |
    132 |
    133 | 139 |
    140 |
    141 |
    142 |
    143 |
    144 | Step 1 - NPM install 145 |
    146 |
    147 |
    npm install --save react-youtube-autocomplete
    148 |
    149 |
    150 | Step 2 - Embed component 151 |
    152 |
    153 |
    154 | 155 |
    156 |
    157 |
    158 | Step 3 - Download base styles 159 |
    160 |
    161 |
    162 | 163 |
    164 |
    165 |
    166 |
    167 | 170 | 175 |
      176 | {formattedResults} 177 |
    178 |
    179 | 180 |
    181 | } 182 | }) 183 | 184 | render(, document.querySelector('#demo')) 185 | -------------------------------------------------------------------------------- /demo/src/styles.scss: -------------------------------------------------------------------------------- 1 | /* component styles */ 2 | /* ---------------- */ 3 | 4 | .react-typeahead-options { 5 | margin: 0; 6 | padding: 0; 7 | list-style-type: none; 8 | border: 1px solid #ccc; 9 | cursor: default; 10 | box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2); 11 | z-index: 1; 12 | width: 100%; 13 | } 14 | 15 | .react-typeahead-options { 16 | margin-left: -8px; 17 | } 18 | 19 | .react-typeahead-options li[role="option"] { 20 | padding: 5px; 21 | text-align: left; 22 | cursor: pointer; 23 | } 24 | 25 | .react-typeahead-options li[role="option"][aria-selected="true"] { 26 | background: #00bcd4; 27 | color: white; 28 | font-weight: bold; 29 | } 30 | 31 | .react-typeahead-container { 32 | border: 1px solid #024e6a; 33 | padding: 5px 8px; 34 | border-radius: 0px; 35 | background-color: white; 36 | margin: 0 auto 37 | } 38 | 39 | .react-typeahead-input { 40 | position: relative; 41 | background: white; 42 | outline: none; 43 | width: 100%; 44 | font-size: 24px; 45 | line-height: 30px; 46 | border: none; 47 | } 48 | 49 | /* base styles */ 50 | /* ----------- */ 51 | 52 | 53 | html, 54 | body { 55 | background-color: white; 56 | font: normal normal normal 18px/1.2 "Helvetica Neue", Roboto, "Segoe UI", Calibri, sans-serif; 57 | color: #292f33; 58 | padding-top: 20px; 59 | margin: 0; 60 | } 61 | 62 | .container { 63 | max-width: 900px; 64 | margin: 0 auto; 65 | text-align: center; 66 | } 67 | 68 | #main { 69 | max-width: 700px; 70 | margin:0 auto; 71 | padding:0 20px 20px 20px; 72 | } 73 | 74 | #demo-box { 75 | max-width: 490px; 76 | margin:0 auto; 77 | } 78 | 79 | h1 { 80 | color: #00bcd4; 81 | margin: 0; 82 | } 83 | 84 | a { 85 | color: #00bcd4; 86 | } 87 | 88 | ul, 89 | li { 90 | margin:0px; 91 | } 92 | 93 | #github-links { 94 | position: relative; 95 | top: 9px; 96 | display:inline-block; 97 | } 98 | 99 | iframe { 100 | margin-top: 10px; 101 | margin-left: 20px; 102 | } 103 | 104 | footer { 105 | padding:30px; 106 | border-top: 1px solid #dddddd; 107 | background: #efefef; 108 | } 109 | 110 | .text-left{ 111 | text-align: left; 112 | } 113 | 114 | .smaller-text { 115 | font-size: 16px; 116 | } 117 | 118 | .headline { 119 | margin-top: 30px; 120 | } 121 | 122 | .youtube-icon, 123 | .react-icon { 124 | display: inline-block; 125 | margin:0 20px; 126 | } 127 | 128 | .youtube-icon { 129 | height: 68px; 130 | width: 105px; 131 | background-image: url(); 132 | background-repeat: no-repeat; 133 | background-size:contain; 134 | } 135 | 136 | .react-icon { 137 | height: 68px; 138 | width: 74px; 139 | background-image: url(); 140 | background-repeat: no-repeat; 141 | background-size: contain; 142 | } 143 | 144 | .code-snippet { 145 | border: 1px solid #dddddd; 146 | background: #efefef; 147 | padding: 10px; 148 | } 149 | 150 | #search-result-list { 151 | display: inline-block; 152 | padding-left:0px; 153 | } 154 | 155 | .search-result { 156 | cursor: pointer; 157 | float: left; 158 | width: 30%; 159 | height: 120px; 160 | padding:0 10px; 161 | list-style-position: inside; 162 | list-style-type:none; 163 | } 164 | 165 | .search-result img { 166 | max-width: 100px; 167 | float: left; 168 | margin-right: 10px; 169 | } 170 | 171 | .search-result span { 172 | font-size: 15px; 173 | width: 87px; 174 | height: 74px; 175 | display: inline-block; 176 | color: black; 177 | overflow: hidden; 178 | } 179 | 180 | 181 | @media only screen and (max-width: 1012px) { 182 | .search-result { 183 | width: 29%; 184 | height: 168px 185 | } 186 | } 187 | 188 | @media only screen and (max-width: 815px) { 189 | .search-result { 190 | width: 100%; 191 | height: 100px 192 | } 193 | } 194 | 195 | @media only screen and (max-width: 500px) { 196 | #github-links { 197 | display: block; 198 | } 199 | 200 | } -------------------------------------------------------------------------------- /nwb.config.js: -------------------------------------------------------------------------------- 1 | module.exports = { 2 | // Let nwb know this is a React component module when generic build commands 3 | // are used. 4 | type: 'react-component', 5 | 6 | // Should nwb create a UMD build for this module? 7 | umd: true, 8 | // The name of the global variable the UMD build of this module will export 9 | global: 'YoutubeAutocomplete', 10 | // Mapping from the npm package names of this module's peerDependencies to the 11 | // global variables they're expected to be available as for use by the UMD 12 | // build. 13 | externals: { 14 | 'react': 'React' 15 | }, 16 | 17 | // Should nwb create a build with untranspiled ES6 modules for tree-shaking 18 | // module bundlers? If you change your mind later, add or remove this line in 19 | // package.json: "jsnext:main": "es6/index.js" 20 | jsNext: true 21 | } 22 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "react-youtube-autocomplete", 3 | "version": "1.0.19", 4 | "description": "A responsive & React-based auto-suggest search box for Youtube apps", 5 | "keywords": [ 6 | "react-component" 7 | ], 8 | "main": "lib/index.js", 9 | "jsnext:main": "es6/index.js", 10 | "files": [ 11 | "css", 12 | "es6", 13 | "lib", 14 | "umd" 15 | ], 16 | "scripts": { 17 | "build": "nwb build", 18 | "clean": "nwb clean", 19 | "start": "nwb serve", 20 | "test": "nwb test", 21 | "deploy": "gh-pages -d demo/dist" 22 | }, 23 | "dependencies": { 24 | "jsonp": "^0.2.0", 25 | "react-typeahead-component2": "^0.10.2", 26 | "youtube-finder": "^1.0.0" 27 | }, 28 | "peerDependencies": { 29 | "react": "^15.0.0", 30 | "react-dom": "^15.0.0" 31 | }, 32 | "devDependencies": { 33 | "gh-pages": "^0.12.0", 34 | "material-ui": "0.14.4", 35 | "nwb": "0.7.x", 36 | "nwb-sass": "^0.5.0", 37 | "react": "0.14.x", 38 | "react-dom": "0.14.x", 39 | "react-embed-code": "^0.1.0", 40 | "react-tap-event-plugin": "^0.2.2" 41 | }, 42 | "author": "", 43 | "homepage": "", 44 | "license": "MIT", 45 | "repository": "https://github.com/hackingbeauty/react-youtube-autocomplete" 46 | } 47 | -------------------------------------------------------------------------------- /src/OptionsTemplate.js: -------------------------------------------------------------------------------- 1 | import React from 'react'; 2 | 3 | export default React.createClass({ 4 | render: function() { 5 | var searchResult = this.props.data[0]; 6 | return ( 7 |
    {searchResult}
    8 | ); 9 | } 10 | }); -------------------------------------------------------------------------------- /src/index.js: -------------------------------------------------------------------------------- 1 | import React, {Component} from 'react'; 2 | import Typeahead from 'react-typeahead-component2'; 3 | import JSONP from 'jsonp'; 4 | import OptionsTemplate from './OptionsTemplate'; 5 | import YoutubeFinder from 'youtube-finder'; 6 | 7 | const googleAutoSuggestURL = '//suggestqueries.google.com/complete/search?client=youtube&ds=yt&q='; 8 | 9 | class YoutubeAutocomplete extends Component { 10 | 11 | constructor(props) { 12 | super(props); 13 | this.state = { 14 | inputValue: '' 15 | } 16 | } 17 | 18 | handleChange(event) { 19 | const 20 | self = this, 21 | query = event.target.value, 22 | url = googleAutoSuggestURL + query; 23 | 24 | this.setState({ 25 | inputValue: query 26 | }); 27 | 28 | JSONP(url, function(error, data){ 29 | if (error) { 30 | console.log(error); 31 | } else { 32 | const searchResults = data[1]; 33 | self.setState({ 34 | options: searchResults 35 | }); 36 | } 37 | }); 38 | } 39 | 40 | onClick(event, optionData) { 41 | const searchTerm = optionData[0]; 42 | this.setState({ 43 | inputValue: searchTerm 44 | }); 45 | } 46 | 47 | onOptionChange(event, optionData, index) { 48 | const 49 | self = this, 50 | searchTerm = optionData[0], 51 | apiKey = this.props.apiKey, 52 | maxResults = this.props.maxResults ? this.props.maxResults : '50'; 53 | 54 | this.setState({ 55 | inputValue: searchTerm 56 | }); 57 | } 58 | 59 | onDropDownClose(event) { 60 | const 61 | self = this, 62 | searchTerm = this.state.inputValue, 63 | maxResults = this.props.maxResults <= 50 ? this.props.maxResults : '50', 64 | YoutubeClient = YoutubeFinder.createClient({ key: this.props.apiKey }), 65 | params = { 66 | part : 'id,snippet', 67 | type : 'video', 68 | q : searchTerm, 69 | maxResults : maxResults 70 | }; 71 | 72 | YoutubeClient.search(params, function(error,results){ 73 | if(error) return console.log(error); 74 | self.props.callback(results.items); 75 | }); 76 | 77 | } 78 | 79 | render() { 80 | // React components using ES6 classes no longer autobind this to non React methods. In your constructor, add: 81 | // this.onChange = this.onChange.bind(this) 82 | // this is why you have to do onChange={this.handleChange.bind(this)} 83 | return
    84 | 95 |
    96 | } 97 | } 98 | 99 | export default YoutubeAutocomplete; 100 | -------------------------------------------------------------------------------- /tests/.eslintrc: -------------------------------------------------------------------------------- 1 | { 2 | "env": { 3 | "mocha": true 4 | } 5 | } 6 | -------------------------------------------------------------------------------- /tests/index-test.js: -------------------------------------------------------------------------------- 1 | import expect from 'expect' 2 | import React from 'react' 3 | import {render, unmountComponentAtNode} from 'react-dom' 4 | 5 | import Component from 'src/' 6 | 7 | describe('Component', () => { 8 | let node 9 | 10 | beforeEach(() => { 11 | node = document.createElement('div') 12 | }) 13 | 14 | afterEach(() => { 15 | unmountComponentAtNode(node) 16 | }) 17 | 18 | it('displays a welcome message', () => { 19 | render(, node, () => { 20 | expect(node.innerHTML).toContain('Welcome to React components') 21 | }) 22 | }) 23 | }) 24 | --------------------------------------------------------------------------------