Merge branch 'MDL-81457-main' of https://github.com/andrewnicols/moodle
[moodle.git] / repository / youtube / lib.php
blobe0e830b9e43c72da255cb88cd27a385c82079d97
1 <?php
2 // This file is part of Moodle - http://moodle.org/
3 //
4 // Moodle is free software: you can redistribute it and/or modify
5 // it under the terms of the GNU General Public License as published by
6 // the Free Software Foundation, either version 3 of the License, or
7 // (at your option) any later version.
8 //
9 // Moodle is distributed in the hope that it will be useful,
10 // but WITHOUT ANY WARRANTY; without even the implied warranty of
11 // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
12 // GNU General Public License for more details.
14 // You should have received a copy of the GNU General Public License
15 // along with Moodle. If not, see <http://www.gnu.org/licenses/>.
17 /**
18 * This plugin is used to access youtube videos
20 * @since Moodle 2.0
21 * @package repository_youtube
22 * @copyright 2010 Dongsheng Cai {@link http://dongsheng.org}
23 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
25 require_once($CFG->dirroot . '/repository/lib.php');
27 /**
28 * repository_youtube class
30 * @since Moodle 2.0
31 * @package repository_youtube
32 * @copyright 2009 Dongsheng Cai {@link http://dongsheng.org}
33 * @license http://www.gnu.org/copyleft/gpl.html GNU GPL v3 or later
36 class repository_youtube extends repository {
37 /** @var int maximum number of thumbs per page */
38 const YOUTUBE_THUMBS_PER_PAGE = 27;
40 /**
41 * API key for using the YouTube Data API.
42 * @var mixed
44 private $apikey;
46 /**
47 * Google Client.
48 * @var Google_Client
50 private $client = null;
52 /**
53 * YouTube Service.
54 * @var Google_Service_YouTube
56 private $service = null;
58 /**
59 * Search keyword text.
60 * @var string
62 protected $keyword;
64 /**
65 * Youtube plugin constructor
66 * @param int $repositoryid
67 * @param object $context
68 * @param array $options
70 public function __construct($repositoryid, $context = SYSCONTEXTID, $options = array()) {
71 parent::__construct($repositoryid, $context, $options);
73 $this->apikey = $this->get_option('apikey');
75 // Without an API key, don't show this repo to users as its useless without it.
76 if (empty($this->apikey)) {
77 $this->disabled = true;
81 /**
82 * Init all the youtube client service stuff.
84 * Instead of instantiating the service in the constructor, we delay
85 * it until really neeed because it's really memory hungry (2MB). That
86 * way the editor or any other artifact requiring repository instantiation
87 * can do it in a cheap way. Sort of lazy loading the plugin.
89 private function init_youtube_service() {
90 global $CFG;
92 if (!isset($this->service)) {
93 require_once($CFG->libdir . '/google/lib.php');
94 $this->client = get_google_client();
95 $this->client->setDeveloperKey($this->apikey);
96 $this->client->setScopes(array(Google_Service_YouTube::YOUTUBE_READONLY));
97 $this->service = new Google_Service_YouTube($this->client);
102 * Save apikey in config table.
103 * @param array $options
104 * @return boolean
106 public function set_option($options = array()) {
107 if (!empty($options['apikey'])) {
108 set_config('apikey', trim($options['apikey']), 'youtube');
110 unset($options['apikey']);
111 return parent::set_option($options);
115 * Get apikey from config table.
117 * @param string $config
118 * @return mixed
120 public function get_option($config = '') {
121 if ($config === 'apikey') {
122 return trim(get_config('youtube', 'apikey'));
123 } else {
124 $options['apikey'] = trim(get_config('youtube', 'apikey'));
126 return parent::get_option($config);
129 public function check_login() {
130 return !empty($this->keyword);
134 * Return search results
135 * @param string $search_text
136 * @return array
138 public function search($search_text, $page = 0) {
139 global $SESSION;
140 $sort = optional_param('youtube_sort', '', PARAM_TEXT);
141 $sess_keyword = 'youtube_'.$this->id.'_keyword';
142 $sess_sort = 'youtube_'.$this->id.'_sort';
144 // This is the request of another page for the last search, retrieve the cached keyword and sort
145 if ($page && !$search_text && isset($SESSION->{$sess_keyword})) {
146 $search_text = $SESSION->{$sess_keyword};
148 if ($page && !$sort && isset($SESSION->{$sess_sort})) {
149 $sort = $SESSION->{$sess_sort};
151 if (!$sort) {
152 $sort = 'relevance'; // default
155 // Save this search in session
156 $SESSION->{$sess_keyword} = $search_text;
157 $SESSION->{$sess_sort} = $sort;
159 $this->keyword = $search_text;
160 $ret = array();
161 $ret['nologin'] = true;
162 $ret['page'] = (int)$page;
163 if ($ret['page'] < 1) {
164 $ret['page'] = 1;
166 $start = ($ret['page'] - 1) * self::YOUTUBE_THUMBS_PER_PAGE + 1;
167 $max = self::YOUTUBE_THUMBS_PER_PAGE;
168 $ret['list'] = $this->_get_collection($search_text, $start, $max, $sort);
169 $ret['norefresh'] = true;
170 $ret['nosearch'] = true;
171 // If the number of results is smaller than $max, it means we reached the last page.
172 $ret['pages'] = (count($ret['list']) < $max) ? $ret['page'] : -1;
173 return $ret;
177 * Private method to get youtube search results
178 * @param string $keyword
179 * @param int $start
180 * @param int $max max results
181 * @param string $sort
182 * @throws moodle_exception If the google API returns an error.
183 * @return array
185 private function _get_collection($keyword, $start, $max, $sort) {
186 global $SESSION;
188 // The new API doesn't use "page" numbers for browsing through results.
189 // It uses a prev and next token in each set that you need to use to
190 // request the next page of results.
191 $sesspagetoken = 'youtube_'.$this->id.'_nextpagetoken';
192 $pagetoken = '';
193 if ($start > 1 && isset($SESSION->{$sesspagetoken})) {
194 $pagetoken = $SESSION->{$sesspagetoken};
197 $list = array();
198 $error = null;
199 try {
200 $this->init_youtube_service(); // About to use the service, ensure it's loaded.
201 $response = $this->service->search->listSearch('id,snippet', array(
202 'q' => $keyword,
203 'maxResults' => $max,
204 'order' => $sort,
205 'pageToken' => $pagetoken,
206 'type' => 'video',
207 'videoEmbeddable' => 'true',
210 // Track the next page token for the next request (when a user
211 // scrolls down in the file picker for more videos).
212 $SESSION->{$sesspagetoken} = $response['nextPageToken'];
214 foreach ($response['items'] as $result) {
215 $title = $result->snippet->title;
216 $source = 'http://www.youtube.com/v/' . $result->id->videoId . '#' . $title;
217 $thumb = $result->snippet->getThumbnails()->getDefault();
219 $list[] = array(
220 'shorttitle' => $title,
221 'thumbnail_title' => $result->snippet->description,
222 'title' => $title.'.avi', // This is a hack so we accept this file by extension.
223 'thumbnail' => $thumb->url,
224 'thumbnail_width' => (int)$thumb->width,
225 'thumbnail_height' => (int)$thumb->height,
226 'size' => '',
227 'date' => '',
228 'source' => $source,
231 } catch (Google_Service_Exception $e) {
232 // If we throw the google exception as-is, we may expose the apikey
233 // to end users. The full message in the google exception includes
234 // the apikey param, so we take just the part pertaining to the
235 // actual error.
236 $error = $e->getErrors()[0]['message'];
237 throw new moodle_exception('apierror', 'repository_youtube', '', $error);
240 return $list;
244 * Youtube plugin doesn't support global search
246 public function global_search() {
247 return false;
250 public function get_listing($path='', $page = '') {
251 return array();
255 * Generate search form
257 public function print_login($ajax = true) {
258 $ret = array();
259 $search = new stdClass();
260 $search->type = 'text';
261 $search->id = 'youtube_search';
262 $search->name = 's';
263 $search->label = get_string('search', 'repository_youtube').': ';
264 $sort = new stdClass();
265 $sort->type = 'select';
266 $sort->options = array(
267 (object)array(
268 'value' => 'relevance',
269 'label' => get_string('sortrelevance', 'repository_youtube')
271 (object)array(
272 'value' => 'date',
273 'label' => get_string('sortpublished', 'repository_youtube')
275 (object)array(
276 'value' => 'rating',
277 'label' => get_string('sortrating', 'repository_youtube')
279 (object)array(
280 'value' => 'viewCount',
281 'label' => get_string('sortviewcount', 'repository_youtube')
284 $sort->id = 'youtube_sort';
285 $sort->name = 'youtube_sort';
286 $sort->label = get_string('sortby', 'repository_youtube').': ';
287 $ret['login'] = array($search, $sort);
288 $ret['login_btn_label'] = get_string('search');
289 $ret['login_btn_action'] = 'search';
290 $ret['allowcaching'] = true; // indicates that login form can be cached in filepicker.js
291 return $ret;
295 * file types supported by youtube plugin
296 * @return array
298 public function supported_filetypes() {
299 return array('video');
303 * Youtube plugin only return external links
304 * @return int
306 public function supported_returntypes() {
307 return FILE_EXTERNAL;
311 * Is this repository accessing private data?
313 * @return bool
315 public function contains_private_data() {
316 return false;
320 * Add plugin settings input to Moodle form.
321 * @param object $mform
322 * @param string $classname
324 public static function type_config_form($mform, $classname = 'repository') {
325 parent::type_config_form($mform, $classname);
326 $apikey = get_config('youtube', 'apikey');
327 if (empty($apikey)) {
328 $apikey = '';
331 $mform->addElement('text', 'apikey', get_string('apikey', 'repository_youtube'), array('value' => $apikey, 'size' => '40'));
332 $mform->setType('apikey', PARAM_RAW_TRIMMED);
333 $mform->addRule('apikey', get_string('required'), 'required', null, 'client');
335 $mform->addElement('static', null, '', get_string('information', 'repository_youtube'));
339 * Names of the plugin settings
340 * @return array
342 public static function get_type_option_names() {
343 return array('apikey', 'pluginname');