link.lib.php 68 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004
  1. <?php
  2. /* For licensing terms, see /license.txt */
  3. use Chamilo\CourseBundle\Entity\CLink;
  4. /**
  5. * Function library for the links tool.
  6. *
  7. * This is a complete remake of the original link tool.
  8. * New features:
  9. * - Organize links into categories;
  10. * - favorites/bookmarks interface;
  11. * - move links up/down within a category;
  12. * - move categories up/down;
  13. * - expand/collapse all categories;
  14. * - add link to 'root' category => category-less link is always visible.
  15. *
  16. * @author Patrick Cool, complete remake (December 2003 - January 2004)
  17. * @author René Haentjens, CSV file import (October 2004)
  18. *
  19. * @package chamilo.link
  20. */
  21. class Link extends Model
  22. {
  23. public $table;
  24. public $is_course_model = true;
  25. public $columns = [
  26. 'id',
  27. 'c_id',
  28. 'url',
  29. 'title',
  30. 'description',
  31. 'category_id',
  32. 'display_order',
  33. 'on_homepage',
  34. 'target',
  35. 'session_id',
  36. ];
  37. public $required = ['url', 'title'];
  38. private $course;
  39. /**
  40. * Link constructor.
  41. */
  42. public function __construct()
  43. {
  44. $this->table = Database::get_course_table(TABLE_LINK);
  45. }
  46. /**
  47. * @param array $course
  48. */
  49. public function setCourse($course)
  50. {
  51. $this->course = $course;
  52. }
  53. /**
  54. * @return array
  55. */
  56. public function getCourse()
  57. {
  58. return !empty($this->course) ? $this->course : api_get_course_info();
  59. }
  60. /**
  61. * Organize the saving of a link, using the parent's save method and
  62. * updating the item_property table.
  63. *
  64. * @param array $params
  65. * @param bool $show_query Whether to show the query in logs when
  66. * calling parent's save method
  67. *
  68. * @return bool True if link could be saved, false otherwise
  69. */
  70. public function save($params, $show_query = null)
  71. {
  72. $course_info = $this->getCourse();
  73. $courseId = $course_info['real_id'];
  74. $params['session_id'] = api_get_session_id();
  75. $params['category_id'] = isset($params['category_id']) ? $params['category_id'] : 0;
  76. $sql = "SELECT MAX(display_order)
  77. FROM ".$this->table."
  78. WHERE
  79. c_id = $courseId AND
  80. category_id = '".intval($params['category_id'])."'";
  81. $result = Database:: query($sql);
  82. list($orderMax) = Database:: fetch_row($result);
  83. $order = $orderMax + 1;
  84. $params['display_order'] = $order;
  85. $id = parent::save($params, $show_query);
  86. if (!empty($id)) {
  87. // iid
  88. $sql = "UPDATE ".$this->table." SET id = iid WHERE iid = $id";
  89. Database::query($sql);
  90. api_item_property_update(
  91. $course_info,
  92. TOOL_LINK,
  93. $id,
  94. 'LinkAdded',
  95. api_get_user_id()
  96. );
  97. api_set_default_visibility($id, TOOL_LINK);
  98. }
  99. return $id;
  100. }
  101. /**
  102. * Update a link in the database.
  103. *
  104. * @param int $linkId The ID of the link to update
  105. * @param string $linkUrl The new URL to be saved
  106. * @param int $courseId
  107. * @param int $sessionId
  108. *
  109. * @return bool
  110. */
  111. public function updateLink(
  112. $linkId,
  113. $linkUrl,
  114. $courseId = null,
  115. $sessionId = null
  116. ) {
  117. $tblLink = Database::get_course_table(TABLE_LINK);
  118. $linkUrl = Database::escape_string($linkUrl);
  119. $linkId = intval($linkId);
  120. if (is_null($courseId)) {
  121. $courseId = api_get_course_int_id();
  122. }
  123. $courseId = intval($courseId);
  124. if (is_null($sessionId)) {
  125. $sessionId = api_get_session_id();
  126. }
  127. $sessionId = intval($sessionId);
  128. if ($linkUrl != '') {
  129. $sql = "UPDATE $tblLink SET
  130. url = '$linkUrl'
  131. WHERE id = $linkId AND c_id = $courseId AND session_id = $sessionId";
  132. $resLink = Database::query($sql);
  133. return $resLink;
  134. }
  135. return false;
  136. }
  137. /**
  138. * Used to add a link or a category.
  139. *
  140. * @param string $type , "link" or "category"
  141. *
  142. * @todo replace strings by constants
  143. *
  144. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  145. *
  146. * @return bool True on success, false on failure
  147. */
  148. public static function addlinkcategory($type)
  149. {
  150. $ok = true;
  151. $_course = api_get_course_info();
  152. $course_id = $_course['real_id'];
  153. $session_id = api_get_session_id();
  154. if ($type == 'link') {
  155. $title = Security::remove_XSS(stripslashes($_POST['title']));
  156. $urllink = Security::remove_XSS($_POST['url']);
  157. $description = Security::remove_XSS($_POST['description']);
  158. $selectcategory = Security::remove_XSS($_POST['category_id']);
  159. $onhomepage = 0;
  160. if (isset($_POST['on_homepage'])) {
  161. $onhomepage = Security::remove_XSS($_POST['on_homepage']);
  162. }
  163. $target = '_self'; // Default target.
  164. if (!empty($_POST['target'])) {
  165. $target = Security::remove_XSS($_POST['target']);
  166. }
  167. $urllink = trim($urllink);
  168. $title = trim($title);
  169. $description = trim($description);
  170. // We ensure URL to be absolute.
  171. if (strpos($urllink, '://') === false) {
  172. $urllink = 'http://'.$urllink;
  173. }
  174. // If the title is empty, we use the URL as title.
  175. if ($title == '') {
  176. $title = $urllink;
  177. }
  178. // If the URL is invalid, an error occurs.
  179. if (!api_valid_url($urllink, true)) {
  180. // A check against an absolute URL
  181. Display::addFlash(Display::return_message(get_lang('GiveURL'), 'error'));
  182. return false;
  183. } else {
  184. // Looking for the largest order number for this category.
  185. $link = new Link();
  186. $params = [
  187. 'c_id' => $course_id,
  188. 'url' => $urllink,
  189. 'title' => $title,
  190. 'description' => $description,
  191. 'category_id' => $selectcategory,
  192. 'on_homepage' => $onhomepage,
  193. 'target' => $target,
  194. 'session_id' => $session_id,
  195. ];
  196. $link_id = $link->save($params);
  197. if ((api_get_setting('search_enabled') == 'true') &&
  198. $link_id && extension_loaded('xapian')
  199. ) {
  200. require_once api_get_path(LIBRARY_PATH).'specific_fields_manager.lib.php';
  201. $course_int_id = $_course['real_id'];
  202. $courseCode = $_course['code'];
  203. $specific_fields = get_specific_field_list();
  204. $ic_slide = new IndexableChunk();
  205. // Add all terms to db.
  206. $all_specific_terms = '';
  207. foreach ($specific_fields as $specific_field) {
  208. if (isset($_REQUEST[$specific_field['code']])) {
  209. $sterms = trim($_REQUEST[$specific_field['code']]);
  210. if (!empty($sterms)) {
  211. $all_specific_terms .= ' '.$sterms;
  212. $sterms = explode(',', $sterms);
  213. foreach ($sterms as $sterm) {
  214. $ic_slide->addTerm(
  215. trim($sterm),
  216. $specific_field['code']
  217. );
  218. add_specific_field_value(
  219. $specific_field['id'],
  220. $courseCode,
  221. TOOL_LINK,
  222. $link_id,
  223. $sterm
  224. );
  225. }
  226. }
  227. }
  228. }
  229. // Build the chunk to index.
  230. $ic_slide->addValue('title', $title);
  231. $ic_slide->addCourseId($courseCode);
  232. $ic_slide->addToolId(TOOL_LINK);
  233. $xapian_data = [
  234. SE_COURSE_ID => $courseCode,
  235. SE_TOOL_ID => TOOL_LINK,
  236. SE_DATA => [
  237. 'link_id' => (int) $link_id,
  238. ],
  239. SE_USER => (int) api_get_user_id(),
  240. ];
  241. $ic_slide->xapian_data = serialize($xapian_data);
  242. $description = $all_specific_terms.' '.$description;
  243. $ic_slide->addValue('content', $description);
  244. // Add category name if set.
  245. if (isset($selectcategory) && $selectcategory > 0) {
  246. $table_link_category = Database::get_course_table(
  247. TABLE_LINK_CATEGORY
  248. );
  249. $sql_cat = 'SELECT * FROM %s WHERE id=%d AND c_id = %d LIMIT 1';
  250. $sql_cat = sprintf(
  251. $sql_cat,
  252. $table_link_category,
  253. (int) $selectcategory,
  254. $course_int_id
  255. );
  256. $result = Database:: query($sql_cat);
  257. if (Database:: num_rows($result) == 1) {
  258. $row = Database:: fetch_array($result);
  259. $ic_slide->addValue(
  260. 'category',
  261. $row['category_title']
  262. );
  263. }
  264. }
  265. $di = new ChamiloIndexer();
  266. isset($_POST['language']) ? $lang = Database:: escape_string(
  267. $_POST['language']
  268. ) : $lang = 'english';
  269. $di->connectDb(null, null, $lang);
  270. $di->addChunk($ic_slide);
  271. // Index and return search engine document id.
  272. $did = $di->index();
  273. if ($did) {
  274. // Save it to db.
  275. $tbl_se_ref = Database::get_main_table(
  276. TABLE_MAIN_SEARCH_ENGINE_REF
  277. );
  278. $sql = 'INSERT INTO %s (id, course_code, tool_id, ref_id_high_level, search_did)
  279. VALUES (NULL , \'%s\', \'%s\', %s, %s)';
  280. $sql = sprintf(
  281. $sql,
  282. $tbl_se_ref,
  283. $course_int_id,
  284. $courseCode,
  285. TOOL_LINK,
  286. $link_id,
  287. $did
  288. );
  289. Database:: query($sql);
  290. }
  291. }
  292. Display::addFlash(Display::return_message(get_lang('LinkAdded')));
  293. return $link_id;
  294. }
  295. } elseif ($type == 'category') {
  296. $tbl_categories = Database::get_course_table(TABLE_LINK_CATEGORY);
  297. $category_title = trim($_POST['category_title']);
  298. $description = trim($_POST['description']);
  299. if (empty($category_title)) {
  300. echo Display::return_message(get_lang('GiveCategoryName'), 'error');
  301. $ok = false;
  302. } else {
  303. // Looking for the largest order number for this category.
  304. $result = Database:: query(
  305. "SELECT MAX(display_order) FROM $tbl_categories
  306. WHERE c_id = $course_id "
  307. );
  308. list($orderMax) = Database:: fetch_row($result);
  309. $order = $orderMax + 1;
  310. $order = intval($order);
  311. $session_id = api_get_session_id();
  312. $params = [
  313. 'c_id' => $course_id,
  314. 'category_title' => $category_title,
  315. 'description' => $description,
  316. 'display_order' => $order,
  317. 'session_id' => $session_id,
  318. ];
  319. $linkId = Database::insert($tbl_categories, $params);
  320. if ($linkId) {
  321. // iid
  322. $sql = "UPDATE $tbl_categories SET id = iid WHERE iid = $linkId";
  323. Database:: query($sql);
  324. // add link_category visibility
  325. // course ID is taken from context in api_set_default_visibility
  326. //api_set_default_visibility($linkId, TOOL_LINK_CATEGORY);
  327. api_item_property_update(
  328. $_course,
  329. TOOL_LINK_CATEGORY,
  330. $linkId,
  331. 'LinkCategoryAdded',
  332. api_get_user_id()
  333. );
  334. api_set_default_visibility($linkId, TOOL_LINK_CATEGORY);
  335. }
  336. Display::addFlash(Display::return_message(get_lang('CategoryAdded')));
  337. return $linkId;
  338. }
  339. }
  340. return $ok;
  341. }
  342. /**
  343. * Used to delete a link or a category.
  344. *
  345. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  346. *
  347. * @param int $id
  348. * @param string $type The type of item to delete
  349. *
  350. * @return bool
  351. */
  352. public static function deletelinkcategory($id, $type)
  353. {
  354. $courseInfo = api_get_course_info();
  355. $tbl_link = Database::get_course_table(TABLE_LINK);
  356. $tbl_categories = Database::get_course_table(TABLE_LINK_CATEGORY);
  357. $course_id = $courseInfo['real_id'];
  358. $id = intval($id);
  359. if (empty($id)) {
  360. return false;
  361. }
  362. $result = false;
  363. switch ($type) {
  364. case 'link':
  365. // -> Items are no longer physically deleted,
  366. // but the visibility is set to 2 (in item_property).
  367. // This will make a restore function possible for the platform administrator.
  368. $sql = "UPDATE $tbl_link SET on_homepage='0'
  369. WHERE c_id = $course_id AND id='".$id."'";
  370. Database:: query($sql);
  371. api_item_property_update(
  372. $courseInfo,
  373. TOOL_LINK,
  374. $id,
  375. 'delete',
  376. api_get_user_id()
  377. );
  378. self::delete_link_from_search_engine(api_get_course_id(), $id);
  379. Skill::deleteSkillsFromItem($id, ITEM_TYPE_LINK);
  380. Display::addFlash(Display::return_message(get_lang('LinkDeleted')));
  381. $result = true;
  382. break;
  383. case 'category':
  384. // First we delete the category itself and afterwards all the links of this category.
  385. $sql = "DELETE FROM ".$tbl_categories."
  386. WHERE c_id = $course_id AND id='".$id."'";
  387. Database:: query($sql);
  388. $sql = "DELETE FROM ".$tbl_link."
  389. WHERE c_id = $course_id AND category_id='".$id."'";
  390. Database:: query($sql);
  391. api_item_property_update(
  392. $courseInfo,
  393. TOOL_LINK_CATEGORY,
  394. $id,
  395. 'delete',
  396. api_get_user_id()
  397. );
  398. Display::addFlash(Display::return_message(get_lang('CategoryDeleted')));
  399. $result = true;
  400. break;
  401. }
  402. return $result;
  403. }
  404. /**
  405. * Removes a link from search engine database.
  406. *
  407. * @param string $course_id Course code
  408. * @param int $link_id Document id to delete
  409. */
  410. public static function delete_link_from_search_engine($course_id, $link_id)
  411. {
  412. // Remove from search engine if enabled.
  413. if (api_get_setting('search_enabled') === 'true') {
  414. $tbl_se_ref = Database::get_main_table(
  415. TABLE_MAIN_SEARCH_ENGINE_REF
  416. );
  417. $sql = 'SELECT * FROM %s WHERE course_code=\'%s\' AND tool_id=\'%s\' AND ref_id_high_level=%s LIMIT 1';
  418. $sql = sprintf($sql, $tbl_se_ref, $course_id, TOOL_LINK, $link_id);
  419. $res = Database:: query($sql);
  420. if (Database:: num_rows($res) > 0) {
  421. $row = Database::fetch_array($res);
  422. $di = new ChamiloIndexer();
  423. $di->remove_document($row['search_did']);
  424. }
  425. $sql = 'DELETE FROM %s WHERE course_code=\'%s\' AND tool_id=\'%s\' AND ref_id_high_level=%s LIMIT 1';
  426. $sql = sprintf($sql, $tbl_se_ref, $course_id, TOOL_LINK, $link_id);
  427. Database:: query($sql);
  428. // Remove terms from db.
  429. require_once api_get_path(LIBRARY_PATH).'specific_fields_manager.lib.php';
  430. delete_all_values_for_item($course_id, TOOL_DOCUMENT, $link_id);
  431. }
  432. }
  433. /**
  434. * Get link info.
  435. *
  436. * @param int $id
  437. *
  438. * @return array link info
  439. */
  440. public static function getLinkInfo($id)
  441. {
  442. $tbl_link = Database::get_course_table(TABLE_LINK);
  443. $course_id = api_get_course_int_id();
  444. if (empty($id) || empty($course_id)) {
  445. return [];
  446. }
  447. $sql = "SELECT * FROM $tbl_link
  448. WHERE c_id = $course_id AND id='".intval($id)."' ";
  449. $result = Database::query($sql);
  450. $data = [];
  451. if (Database::num_rows($result)) {
  452. $data = Database::fetch_array($result);
  453. }
  454. return $data;
  455. }
  456. /**
  457. * @param int $id
  458. * @param array $values
  459. */
  460. public static function editLink($id, $values = [])
  461. {
  462. $tbl_link = Database::get_course_table(TABLE_LINK);
  463. $_course = api_get_course_info();
  464. $course_id = $_course['real_id'];
  465. $id = intval($id);
  466. $values['url'] = trim($values['url']);
  467. $values['title'] = trim($values['title']);
  468. $values['description'] = trim($values['description']);
  469. $values['target'] = empty($values['target']) ? '_self' : $values['target'];
  470. $values['on_homepage'] = isset($values['on_homepage']) ? $values['on_homepage'] : '';
  471. $categoryId = intval($values['category_id']);
  472. // We ensure URL to be absolute.
  473. if (strpos($values['url'], '://') === false) {
  474. $values['url'] = 'http://'.$_POST['url'];
  475. }
  476. // If the title is empty, we use the URL as title.
  477. if ($values['title'] == '') {
  478. $values['title'] = $values['url'];
  479. }
  480. // If the URL is invalid, an error occurs.
  481. if (!api_valid_url($values['url'], true)) {
  482. Display::addFlash(
  483. Display::return_message(get_lang('GiveURL'), 'error')
  484. );
  485. return false;
  486. }
  487. if (empty($id) || empty($course_id)) {
  488. return false;
  489. }
  490. // Finding the old category_id.
  491. $sql = "SELECT * FROM $tbl_link
  492. WHERE c_id = $course_id AND id='".$id."'";
  493. $result = Database:: query($sql);
  494. $row = Database:: fetch_array($result);
  495. $category_id = $row['category_id'];
  496. if ($category_id != $values['category_id']) {
  497. $sql = "SELECT MAX(display_order)
  498. FROM $tbl_link
  499. WHERE
  500. c_id = $course_id AND
  501. category_id='".intval($values['category_id'])."'";
  502. $result = Database:: query($sql);
  503. list($max_display_order) = Database:: fetch_row($result);
  504. $max_display_order++;
  505. } else {
  506. $max_display_order = $row['display_order'];
  507. }
  508. $params = [
  509. 'url' => $values['url'],
  510. 'title' => $values['title'],
  511. 'description' => $values['description'],
  512. 'category_id' => $values['category_id'],
  513. 'display_order' => $max_display_order,
  514. 'on_homepage' => $values['on_homepage'],
  515. 'target' => $values['target'],
  516. 'category_id' => $values['category_id'],
  517. ];
  518. Database::update(
  519. $tbl_link,
  520. $params,
  521. ['c_id = ? AND id = ?' => [$course_id, $id]]
  522. );
  523. // Update search enchine and its values table if enabled.
  524. if (api_get_setting('search_enabled') == 'true') {
  525. $course_int_id = api_get_course_int_id();
  526. $course_id = api_get_course_id();
  527. $link_title = Database:: escape_string($values['title']);
  528. $link_description = Database:: escape_string($values['description']);
  529. // Actually, it consists on delete terms from db,
  530. // insert new ones, create a new search engine document, and remove the old one.
  531. // Get search_did.
  532. $tbl_se_ref = Database::get_main_table(
  533. TABLE_MAIN_SEARCH_ENGINE_REF
  534. );
  535. $sql = 'SELECT * FROM %s WHERE course_code=\'%s\' AND tool_id=\'%s\' AND ref_id_high_level=%s LIMIT 1';
  536. $sql = sprintf(
  537. $sql,
  538. $tbl_se_ref,
  539. $course_id,
  540. TOOL_LINK,
  541. $id
  542. );
  543. $res = Database:: query($sql);
  544. if (Database:: num_rows($res) > 0) {
  545. require_once api_get_path(LIBRARY_PATH).'specific_fields_manager.lib.php';
  546. $se_ref = Database:: fetch_array($res);
  547. $specific_fields = get_specific_field_list();
  548. $ic_slide = new IndexableChunk();
  549. $all_specific_terms = '';
  550. foreach ($specific_fields as $specific_field) {
  551. delete_all_specific_field_value(
  552. $course_id,
  553. $specific_field['id'],
  554. TOOL_LINK,
  555. $id
  556. );
  557. if (isset($_REQUEST[$specific_field['code']])) {
  558. $sterms = trim(
  559. $_REQUEST[$specific_field['code']]
  560. );
  561. if (!empty($sterms)) {
  562. $all_specific_terms .= ' '.$sterms;
  563. $sterms = explode(',', $sterms);
  564. foreach ($sterms as $sterm) {
  565. $ic_slide->addTerm(
  566. trim($sterm),
  567. $specific_field['code']
  568. );
  569. add_specific_field_value(
  570. $specific_field['id'],
  571. $course_id,
  572. TOOL_LINK,
  573. $id,
  574. $sterm
  575. );
  576. }
  577. }
  578. }
  579. }
  580. // Build the chunk to index.
  581. $ic_slide->addValue("title", $link_title);
  582. $ic_slide->addCourseId($course_id);
  583. $ic_slide->addToolId(TOOL_LINK);
  584. $xapian_data = [
  585. SE_COURSE_ID => $course_id,
  586. SE_TOOL_ID => TOOL_LINK,
  587. SE_DATA => [
  588. 'link_id' => (int) $id,
  589. ],
  590. SE_USER => (int) api_get_user_id(),
  591. ];
  592. $ic_slide->xapian_data = serialize($xapian_data);
  593. $link_description = $all_specific_terms.' '.$link_description;
  594. $ic_slide->addValue('content', $link_description);
  595. // Add category name if set.
  596. if (isset($categoryId) && $categoryId > 0) {
  597. $table_link_category = Database::get_course_table(
  598. TABLE_LINK_CATEGORY
  599. );
  600. $sql_cat = 'SELECT * FROM %s WHERE id=%d and c_id = %d LIMIT 1';
  601. $sql_cat = sprintf(
  602. $sql_cat,
  603. $table_link_category,
  604. $categoryId,
  605. $course_int_id
  606. );
  607. $result = Database:: query($sql_cat);
  608. if (Database:: num_rows($result) == 1) {
  609. $row = Database:: fetch_array($result);
  610. $ic_slide->addValue(
  611. 'category',
  612. $row['category_title']
  613. );
  614. }
  615. }
  616. $di = new ChamiloIndexer();
  617. isset($_POST['language']) ? $lang = Database:: escape_string($_POST['language']) : $lang = 'english';
  618. $di->connectDb(null, null, $lang);
  619. $di->remove_document($se_ref['search_did']);
  620. $di->addChunk($ic_slide);
  621. // Index and return search engine document id.
  622. $did = $di->index();
  623. if ($did) {
  624. // Save it to db.
  625. $sql = 'DELETE FROM %s
  626. WHERE course_code=\'%s\'
  627. AND tool_id=\'%s\'
  628. AND ref_id_high_level=\'%s\'';
  629. $sql = sprintf(
  630. $sql,
  631. $tbl_se_ref,
  632. $course_id,
  633. TOOL_LINK,
  634. $id
  635. );
  636. Database:: query($sql);
  637. $sql = 'INSERT INTO %s (c_id, id, course_code, tool_id, ref_id_high_level, search_did)
  638. VALUES (NULL , \'%s\', \'%s\', %s, %s)';
  639. $sql = sprintf(
  640. $sql,
  641. $tbl_se_ref,
  642. $course_int_id,
  643. $course_id,
  644. TOOL_LINK,
  645. $id,
  646. $did
  647. );
  648. Database:: query($sql);
  649. }
  650. }
  651. }
  652. // "WHAT'S NEW" notification: update table last_toolEdit.
  653. api_item_property_update(
  654. $_course,
  655. TOOL_LINK,
  656. $id,
  657. 'LinkUpdated',
  658. api_get_user_id()
  659. );
  660. Display::addFlash(Display::return_message(get_lang('LinkModded')));
  661. }
  662. /**
  663. * @param int $id
  664. * @param array $values
  665. *
  666. * @return bool
  667. */
  668. public static function editCategory($id, $values)
  669. {
  670. $table = Database::get_course_table(TABLE_LINK_CATEGORY);
  671. $course_id = api_get_course_int_id();
  672. $id = intval($id);
  673. // This is used to put the modified info of the category-form into the database.
  674. $params = [
  675. 'category_title' => $values['category_title'],
  676. 'description' => $values['description'],
  677. ];
  678. Database::update(
  679. $table,
  680. $params,
  681. ['c_id = ? AND id = ?' => [$course_id, $id]]
  682. );
  683. Display::addFlash(Display::return_message(get_lang('CategoryModded')));
  684. return true;
  685. }
  686. /**
  687. * Changes the visibility of a link.
  688. *
  689. * @todo add the changing of the visibility of a course
  690. *
  691. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  692. */
  693. public static function change_visibility_link($id, $scope)
  694. {
  695. $_course = api_get_course_info();
  696. $_user = api_get_user_info();
  697. if ($scope == TOOL_LINK) {
  698. api_item_property_update(
  699. $_course,
  700. TOOL_LINK,
  701. $id,
  702. $_GET['action'],
  703. $_user['user_id']
  704. );
  705. Display::addFlash(Display::return_message(get_lang('VisibilityChanged')));
  706. } elseif ($scope == TOOL_LINK_CATEGORY) {
  707. api_item_property_update(
  708. $_course,
  709. TOOL_LINK_CATEGORY,
  710. $id,
  711. $_GET['action'],
  712. $_user['user_id']
  713. );
  714. Display::addFlash(Display::return_message(get_lang('VisibilityChanged')));
  715. }
  716. }
  717. /**
  718. * Generate SQL to select all the links categories in the current course and
  719. * session.
  720. *
  721. * @param int $courseId
  722. * @param int $sessionId
  723. * @param bool $withBaseContent
  724. *
  725. * @return array
  726. */
  727. public static function getLinkCategories($courseId, $sessionId, $withBaseContent = true)
  728. {
  729. $tblLinkCategory = Database::get_course_table(TABLE_LINK_CATEGORY);
  730. $tblItemProperty = Database::get_course_table(TABLE_ITEM_PROPERTY);
  731. $courseId = intval($courseId);
  732. $courseInfo = api_get_course_info_by_id($courseId);
  733. // Condition for the session.
  734. $sessionCondition = api_get_session_condition(
  735. $sessionId,
  736. true,
  737. $withBaseContent,
  738. 'linkcat.session_id'
  739. );
  740. // Getting links
  741. $sql = "SELECT *, linkcat.id
  742. FROM $tblLinkCategory linkcat
  743. WHERE
  744. linkcat.c_id = $courseId
  745. $sessionCondition
  746. ORDER BY linkcat.display_order DESC";
  747. $result = Database::query($sql);
  748. $categories = Database::store_result($result);
  749. $sql = "SELECT *, linkcat.id
  750. FROM $tblLinkCategory linkcat
  751. INNER JOIN $tblItemProperty ip
  752. ON (linkcat.id = ip.ref AND linkcat.c_id = ip.c_id)
  753. WHERE
  754. ip.tool = '".TOOL_LINK_CATEGORY."' AND
  755. (ip.visibility = '0' OR ip.visibility = '1')
  756. $sessionCondition AND
  757. linkcat.c_id = ".$courseId."
  758. ORDER BY linkcat.display_order DESC";
  759. $result = Database::query($sql);
  760. $categoryInItemProperty = [];
  761. if (Database::num_rows($result)) {
  762. while ($row = Database::fetch_array($result, 'ASSOC')) {
  763. $categoryInItemProperty[$row['id']] = $row;
  764. }
  765. }
  766. foreach ($categories as &$category) {
  767. if (!isset($categoryInItemProperty[$category['id']])) {
  768. api_item_property_update(
  769. $courseInfo,
  770. TOOL_LINK_CATEGORY,
  771. $category['id'],
  772. 'LinkCategoryAdded',
  773. api_get_user_id()
  774. );
  775. }
  776. }
  777. $sql = "SELECT DISTINCT linkcat.*, visibility
  778. FROM $tblLinkCategory linkcat
  779. INNER JOIN $tblItemProperty ip
  780. ON (linkcat.id = ip.ref AND linkcat.c_id = ip.c_id)
  781. WHERE
  782. ip.tool = '".TOOL_LINK_CATEGORY."' AND
  783. (ip.visibility = '0' OR ip.visibility = '1')
  784. $sessionCondition AND
  785. linkcat.c_id = ".$courseId."
  786. ORDER BY linkcat.display_order DESC
  787. ";
  788. $result = Database::query($sql);
  789. return Database::store_result($result, 'ASSOC');
  790. }
  791. /**
  792. * @param int $categoryId
  793. * @param $courseId
  794. * @param $sessionId
  795. * @param bool $withBaseContent
  796. *
  797. * @return array
  798. */
  799. public static function getLinksPerCategory(
  800. $categoryId,
  801. $courseId,
  802. $sessionId,
  803. $withBaseContent = true
  804. ) {
  805. $tbl_link = Database::get_course_table(TABLE_LINK);
  806. $TABLE_ITEM_PROPERTY = Database::get_course_table(TABLE_ITEM_PROPERTY);
  807. $courseId = (int) $courseId;
  808. $sessionId = (int) $sessionId;
  809. $categoryId = (int) $categoryId;
  810. // Condition for the session.
  811. $condition_session = api_get_session_condition(
  812. $sessionId,
  813. true,
  814. false,
  815. 'ip.session_id'
  816. );
  817. if (!empty($sessionId)) {
  818. $conditionBaseSession = api_get_session_condition(
  819. 0,
  820. true,
  821. $withBaseContent,
  822. 'ip.session_id'
  823. );
  824. $condition = " AND
  825. (
  826. (ip.visibility = '1' $conditionBaseSession) OR
  827. (
  828. (ip.visibility = '0' OR ip.visibility = '1')
  829. $condition_session
  830. )
  831. )
  832. ";
  833. } else {
  834. $condition = api_get_session_condition(
  835. 0,
  836. true,
  837. false,
  838. 'ip.session_id'
  839. );
  840. $condition .= " AND (ip.visibility = '0' OR ip.visibility = '1') $condition ";
  841. }
  842. $sql = "SELECT
  843. link.id,
  844. ip.session_id,
  845. link.session_id link_session_id,
  846. url,
  847. category_id,
  848. visibility,
  849. description,
  850. title,
  851. target,
  852. on_homepage
  853. FROM $tbl_link link
  854. INNER JOIN $TABLE_ITEM_PROPERTY ip
  855. ON (link.id = ip.ref AND link.c_id = ip.c_id)
  856. WHERE
  857. ip.tool = '".TOOL_LINK."' AND
  858. link.category_id = '".$categoryId."' AND
  859. link.c_id = $courseId AND
  860. ip.c_id = $courseId
  861. $condition
  862. ORDER BY link.display_order ASC, ip.session_id DESC";
  863. $result = Database:: query($sql);
  864. return Database::store_result($result);
  865. }
  866. /**
  867. * Displays all the links of a given category.
  868. *
  869. * @param int $catid
  870. * @param int $courseId
  871. * @param int $session_id
  872. * @param bool $showActionLinks
  873. *
  874. * @return string
  875. *
  876. * @author Julio Montoya
  877. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  878. */
  879. public static function showLinksPerCategory($catid, $courseId, $session_id, $showActionLinks = true)
  880. {
  881. global $token;
  882. $_user = api_get_user_info();
  883. $catid = (int) $catid;
  884. $links = self::getLinksPerCategory($catid, $courseId, $session_id);
  885. $content = '';
  886. $numberOfLinks = count($links);
  887. if (!empty($links)) {
  888. $content .= '<div class="link list-group">';
  889. $i = 1;
  890. $linksAdded = [];
  891. foreach ($links as $myrow) {
  892. $linkId = $myrow['id'];
  893. if (in_array($linkId, $linksAdded)) {
  894. continue;
  895. }
  896. $linksAdded[] = $linkId;
  897. $categoryId = $myrow['category_id'];
  898. // Validation when belongs to a session.
  899. $session_img = api_get_session_image(
  900. $myrow['link_session_id'],
  901. $_user['status']
  902. );
  903. $toolbar = '';
  904. $link_validator = '';
  905. if (api_is_allowed_to_edit(null, true)) {
  906. $toolbar .= Display::toolbarButton(
  907. '',
  908. 'javascript:void(0);',
  909. 'check-circle-o',
  910. 'default btn-sm',
  911. [
  912. 'onclick' => "check_url('".$linkId."', '".addslashes($myrow['url'])."');",
  913. 'title' => get_lang('CheckURL'),
  914. ]
  915. );
  916. $link_validator .= Display::span(
  917. '',
  918. [
  919. 'id' => 'url_id_'.$linkId,
  920. 'class' => 'check-link',
  921. ]
  922. );
  923. if ($session_id == $myrow['link_session_id']) {
  924. $url = api_get_self().'?'.api_get_cidreq().'&action=editlink&id='.$linkId;
  925. $title = get_lang('Edit');
  926. $toolbar .= Display::toolbarButton(
  927. '',
  928. $url,
  929. 'pencil',
  930. 'default btn-sm',
  931. [
  932. 'title' => $title,
  933. ]
  934. );
  935. }
  936. $urlVisibility = api_get_self().'?'.api_get_cidreq().
  937. '&sec_token='.$token.
  938. '&id='.$linkId.
  939. '&scope=link&category_id='.$categoryId;
  940. switch ($myrow['visibility']) {
  941. case '1':
  942. $urlVisibility .= '&action=invisible';
  943. $title = get_lang('MakeInvisible');
  944. $toolbar .= Display::toolbarButton(
  945. '',
  946. $urlVisibility,
  947. 'eye',
  948. 'default btn-sm',
  949. [
  950. 'title' => $title,
  951. ]
  952. );
  953. break;
  954. case '0':
  955. $urlVisibility .= '&action=visible';
  956. $title = get_lang('MakeVisible');
  957. $toolbar .= Display::toolbarButton(
  958. '',
  959. $urlVisibility,
  960. 'eye-slash',
  961. 'primary btn-sm',
  962. [
  963. 'title' => $title,
  964. ]
  965. );
  966. break;
  967. }
  968. if ($session_id == $myrow['link_session_id']) {
  969. $moveLinkParams = [
  970. 'id' => $linkId,
  971. 'scope' => 'category',
  972. 'category_id' => $categoryId,
  973. 'action' => 'move_link_up',
  974. ];
  975. $toolbar .= Display::toolbarButton(
  976. get_lang('MoveUp'),
  977. api_get_self().'?'.api_get_cidreq().'&'.http_build_query($moveLinkParams),
  978. 'level-up',
  979. 'default',
  980. ['class' => 'btn-sm '.($i === 1 ? 'disabled' : '')],
  981. false
  982. );
  983. $moveLinkParams['action'] = 'move_link_down';
  984. $toolbar .= Display::toolbarButton(
  985. get_lang('MoveDown'),
  986. api_get_self().'?'.api_get_cidreq().'&'.http_build_query($moveLinkParams),
  987. 'level-down',
  988. 'default',
  989. ['class' => 'btn-sm '.($i === $numberOfLinks ? 'disabled' : '')],
  990. false
  991. );
  992. $url = api_get_self().'?'.api_get_cidreq().'&sec_token='.$token.'&action=deletelink&id='.$linkId.'&category_id='.$categoryId;
  993. $event = "javascript: if(!confirm('".get_lang('LinkDelconfirm')."'))return false;";
  994. $title = get_lang('Delete');
  995. $toolbar .= Display::toolbarButton(
  996. '',
  997. $url,
  998. 'trash',
  999. 'default btn-sm',
  1000. [
  1001. 'onclick' => $event,
  1002. 'title' => $title,
  1003. ]
  1004. );
  1005. }
  1006. }
  1007. $showLink = false;
  1008. $titleClass = '';
  1009. if ($myrow['visibility'] == '1') {
  1010. $showLink = true;
  1011. } else {
  1012. if (api_is_allowed_to_edit(null, true)) {
  1013. $showLink = true;
  1014. $titleClass = 'text-muted';
  1015. }
  1016. }
  1017. if ($showLink) {
  1018. $iconLink = Display::return_icon(
  1019. 'url.png',
  1020. get_lang('Link'),
  1021. null,
  1022. ICON_SIZE_SMALL
  1023. );
  1024. $url = api_get_path(WEB_CODE_PATH).'link/link_goto.php?'.api_get_cidreq().'&link_id='.$linkId.'&link_url='.urlencode($myrow['url']);
  1025. $content .= '<div class="list-group-item">';
  1026. if ($showActionLinks) {
  1027. $content .= '<div class="pull-right"><div class="btn-group">'.$toolbar.'</div></div>';
  1028. }
  1029. $content .= '<h4 class="list-group-item-heading">';
  1030. $content .= $iconLink;
  1031. $content .= Display::tag(
  1032. 'a',
  1033. Security::remove_XSS($myrow['title']),
  1034. [
  1035. 'href' => $url,
  1036. 'target' => $myrow['target'],
  1037. 'class' => $titleClass,
  1038. ]
  1039. );
  1040. $content .= $link_validator;
  1041. $content .= $session_img;
  1042. $content .= '</h4>';
  1043. $content .= '<p class="list-group-item-text">'.$myrow['description'].'</p>';
  1044. $content .= '</div>';
  1045. }
  1046. $i++;
  1047. }
  1048. $content .= '</div>';
  1049. }
  1050. return $content;
  1051. }
  1052. /**
  1053. * Displays the edit, delete and move icons.
  1054. *
  1055. * @param int Category ID
  1056. * @param int $currentCategory
  1057. * @param int $countCategories
  1058. *
  1059. * @return string
  1060. *
  1061. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  1062. */
  1063. public static function showCategoryAdminTools($category, $currentCategory, $countCategories)
  1064. {
  1065. $categoryId = $category['id'];
  1066. $token = null;
  1067. $tools = '<a href="'.api_get_self().'?'.api_get_cidreq().'&sec_token='.$token.'&action=editcategory&id='.$categoryId.'&category_id='.$categoryId.'" title='.get_lang('Modify').'">'.
  1068. Display:: return_icon(
  1069. 'edit.png',
  1070. get_lang('Modify'),
  1071. [],
  1072. ICON_SIZE_SMALL
  1073. ).'</a>';
  1074. // DISPLAY MOVE UP COMMAND only if it is not the top link.
  1075. if ($currentCategory != 0) {
  1076. $tools .= '<a href="'.api_get_self().'?'.api_get_cidreq().'&sec_token='.$token.'&action=up&up='.$categoryId.'&category_id='.$categoryId.'" title="'.get_lang('Up').'">'.
  1077. Display:: return_icon(
  1078. 'up.png',
  1079. get_lang('Up'),
  1080. [],
  1081. ICON_SIZE_SMALL
  1082. ).'</a>';
  1083. } else {
  1084. $tools .= Display:: return_icon(
  1085. 'up_na.png',
  1086. get_lang('Up'),
  1087. [],
  1088. ICON_SIZE_SMALL
  1089. ).'</a>';
  1090. }
  1091. // DISPLAY MOVE DOWN COMMAND only if it is not the bottom link.
  1092. if ($currentCategory < $countCategories - 1) {
  1093. $tools .= '<a href="'.api_get_self().'?'.api_get_cidreq().'&sec_token='.$token.'&action=down&down='.$categoryId.'&category_id='.$categoryId.'">'.
  1094. Display:: return_icon(
  1095. 'down.png',
  1096. get_lang('Down'),
  1097. [],
  1098. ICON_SIZE_SMALL
  1099. ).'</a>';
  1100. } else {
  1101. $tools .= Display:: return_icon(
  1102. 'down_na.png',
  1103. get_lang('Down'),
  1104. [],
  1105. ICON_SIZE_SMALL
  1106. ).'</a>';
  1107. }
  1108. $tools .= '<a href="'.api_get_self().'?'.api_get_cidreq().'&sec_token='.$token.'&action=deletecategory&id='.$categoryId."&category_id=$categoryId\"
  1109. onclick=\"javascript: if(!confirm('".get_lang('CategoryDelconfirm')."')) return false;\">".
  1110. Display:: return_icon(
  1111. 'delete.png',
  1112. get_lang('Delete'),
  1113. [],
  1114. ICON_SIZE_SMALL
  1115. ).'</a>';
  1116. return $tools;
  1117. }
  1118. /**
  1119. * move a link or a linkcategory up or down.
  1120. *
  1121. * @param int Category ID
  1122. * @param int Course ID
  1123. * @param int Session ID
  1124. *
  1125. * @author Patrick Cool <patrick.cool@UGent.be>, Ghent University
  1126. *
  1127. * @todo support sessions
  1128. */
  1129. public static function movecatlink($action, $catlinkid, $courseId = null, $sessionId = null)
  1130. {
  1131. $tbl_categories = Database::get_course_table(TABLE_LINK_CATEGORY);
  1132. if (is_null($courseId)) {
  1133. $courseId = api_get_course_int_id();
  1134. }
  1135. $courseId = intval($courseId);
  1136. if (is_null($sessionId)) {
  1137. $sessionId = api_get_session_id();
  1138. }
  1139. $sessionId = intval($sessionId);
  1140. $thiscatlinkId = intval($catlinkid);
  1141. if ($action == 'down') {
  1142. $sortDirection = 'DESC';
  1143. }
  1144. if ($action == 'up') {
  1145. $sortDirection = 'ASC';
  1146. }
  1147. $movetable = $tbl_categories;
  1148. if (!empty($sortDirection)) {
  1149. if (!in_array(trim(strtoupper($sortDirection)), ['ASC', 'DESC'])) {
  1150. $sortDirection = 'ASC';
  1151. }
  1152. $sql = "SELECT id, display_order FROM $movetable
  1153. WHERE c_id = $courseId
  1154. ORDER BY display_order $sortDirection";
  1155. $linkresult = Database:: query($sql);
  1156. $thislinkOrder = 1;
  1157. while ($sortrow = Database:: fetch_array($linkresult)) {
  1158. // STEP 2 : FOUND THE NEXT LINK ID AND ORDER, COMMIT SWAP
  1159. // This part seems unlogic, but it isn't . We first look for the current link with the querystring ID
  1160. // and we know the next iteration of the while loop is the next one. These should be swapped.
  1161. if (isset($thislinkFound) && $thislinkFound) {
  1162. $nextlinkId = $sortrow['id'];
  1163. $nextlinkOrder = $sortrow['display_order'];
  1164. Database:: query(
  1165. "UPDATE ".$movetable."
  1166. SET display_order = '$nextlinkOrder'
  1167. WHERE c_id = $courseId AND id = '$thiscatlinkId'"
  1168. );
  1169. Database:: query(
  1170. "UPDATE ".$movetable."
  1171. SET display_order = '$thislinkOrder'
  1172. WHERE c_id = $courseId AND id = '$nextlinkId'"
  1173. );
  1174. break;
  1175. }
  1176. if ($sortrow['id'] == $thiscatlinkId) {
  1177. $thislinkOrder = $sortrow['display_order'];
  1178. $thislinkFound = true;
  1179. }
  1180. }
  1181. }
  1182. Display::addFlash(Display::return_message(get_lang('LinkMoved')));
  1183. }
  1184. /**
  1185. * CSV file import functions.
  1186. *
  1187. * @author René Haentjens , Ghent University
  1188. *
  1189. * @param string $catname
  1190. *
  1191. * @return int
  1192. */
  1193. public static function get_cat($catname)
  1194. {
  1195. // Get category id (existing or make new).
  1196. $tbl_categories = Database::get_course_table(TABLE_LINK_CATEGORY);
  1197. $course_id = api_get_course_int_id();
  1198. $result = Database:: query(
  1199. "SELECT id FROM ".$tbl_categories."
  1200. WHERE c_id = $course_id AND category_title='".Database::escape_string($catname)."'"
  1201. );
  1202. if (Database:: num_rows($result) >= 1 && ($row = Database:: fetch_array($result))) {
  1203. return $row['id']; // Several categories with same name: take the first.
  1204. }
  1205. $result = Database:: query(
  1206. "SELECT MAX(display_order) FROM ".$tbl_categories." WHERE c_id = $course_id "
  1207. );
  1208. list($max_order) = Database:: fetch_row($result);
  1209. $params = [
  1210. 'c_id' => $course_id,
  1211. 'category_title' => $catname,
  1212. 'description' => '',
  1213. 'display_order' => $max_order + 1,
  1214. ];
  1215. $id = Database::insert($tbl_categories, $params);
  1216. return $id;
  1217. }
  1218. /**
  1219. * CSV file import functions.
  1220. *
  1221. * @author René Haentjens , Ghent University
  1222. *
  1223. * @param string $url
  1224. * @param string $title
  1225. * @param string $description
  1226. * @param string $on_homepage
  1227. * @param string $hidden
  1228. */
  1229. public static function put_link($url, $cat, $title, $description, $on_homepage, $hidden)
  1230. {
  1231. $_course = api_get_course_info();
  1232. $_user = api_get_user_info();
  1233. $tbl_link = Database::get_course_table(TABLE_LINK);
  1234. $course_id = api_get_course_int_id();
  1235. $urleq = "url='".Database:: escape_string($url)."'";
  1236. $cateq = "category_id=".intval($cat);
  1237. $result = Database:: query(
  1238. "
  1239. SELECT id FROM $tbl_link
  1240. WHERE c_id = $course_id AND ".$urleq.' AND '.$cateq
  1241. );
  1242. if (Database:: num_rows($result) >= 1 && ($row = Database:: fetch_array($result))) {
  1243. $sql = "UPDATE $tbl_link SET
  1244. title = '".Database:: escape_string($title)."',
  1245. description = '".Database:: escape_string($description)."'
  1246. WHERE c_id = $course_id AND id='".Database:: escape_string($row['id'])."'";
  1247. Database:: query($sql);
  1248. $ipu = 'LinkUpdated';
  1249. $rv = 1; // 1 = upd
  1250. } else {
  1251. // Add new link
  1252. $result = Database:: query(
  1253. "SELECT MAX(display_order) FROM $tbl_link
  1254. WHERE c_id = $course_id AND category_id='".intval($cat)."'"
  1255. );
  1256. list($max_order) = Database:: fetch_row($result);
  1257. Database:: query(
  1258. "INSERT INTO $tbl_link (c_id, url, title, description, category_id, display_order, on_homepage)
  1259. VALUES (".api_get_course_int_id().",
  1260. '".Database:: escape_string($url)."',
  1261. '".Database:: escape_string($title)."',
  1262. '".Database:: escape_string($description)."',
  1263. '".intval($cat)."','".(intval($max_order) + 1)."',
  1264. '".intval($on_homepage).
  1265. "')"
  1266. );
  1267. $id = Database:: insert_id();
  1268. $ipu = 'LinkAdded';
  1269. $rv = 2; // 2 = new
  1270. }
  1271. api_item_property_update(
  1272. $_course,
  1273. TOOL_LINK,
  1274. $id,
  1275. $ipu,
  1276. $_user['user_id']
  1277. );
  1278. if ($hidden && $ipu == 'LinkAdded') {
  1279. api_item_property_update(
  1280. $_course,
  1281. TOOL_LINK,
  1282. $id,
  1283. 'invisible',
  1284. $_user['user_id']
  1285. );
  1286. }
  1287. return $rv;
  1288. }
  1289. /**
  1290. * This function checks if the url is a vimeo link.
  1291. *
  1292. * @author Julio Montoya
  1293. *
  1294. * @version 1.0
  1295. */
  1296. public static function isVimeoLink($url)
  1297. {
  1298. $isLink = strrpos($url, 'vimeo.com');
  1299. return $isLink;
  1300. }
  1301. /**
  1302. * Get vimeo id from URL.
  1303. *
  1304. * @param string $url
  1305. *
  1306. * @return bool|mixed
  1307. */
  1308. public static function getVimeoLinkId($url)
  1309. {
  1310. $possibleUrls = [
  1311. 'http://www.vimeo.com/',
  1312. 'http://vimeo.com/',
  1313. 'https://www.vimeo.com/',
  1314. 'https://vimeo.com/',
  1315. ];
  1316. $url = str_replace($possibleUrls, '', $url);
  1317. if (is_numeric($url)) {
  1318. return $url;
  1319. }
  1320. return false;
  1321. }
  1322. /**
  1323. * This function checks if the url is a youtube link.
  1324. *
  1325. * @author Jorge Frisancho
  1326. * @author Julio Montoya - Fixing code
  1327. *
  1328. * @version 1.0
  1329. */
  1330. public static function is_youtube_link($url)
  1331. {
  1332. $is_youtube_link = strrpos($url, 'youtube') || strrpos(
  1333. $url,
  1334. 'youtu.be'
  1335. );
  1336. return $is_youtube_link;
  1337. }
  1338. /**
  1339. * This function checks if the url is a PDF File link.
  1340. *
  1341. * @author Jorge Frisancho
  1342. * @author Alex Aragón - Fixing code
  1343. *
  1344. * @version 1.0
  1345. */
  1346. public static function isPdfLink($url)
  1347. {
  1348. $isPdfLink = strrpos(strtolower($url), '.pdf');
  1349. return $isPdfLink;
  1350. }
  1351. /**
  1352. * Get youtube id from an URL.
  1353. *
  1354. * @param string $url
  1355. *
  1356. * @return string
  1357. */
  1358. public static function get_youtube_video_id($url)
  1359. {
  1360. // This is the length of YouTube's video IDs
  1361. $len = 11;
  1362. // The ID string starts after "v=", which is usually right after
  1363. // "youtube.com/watch?" in the URL
  1364. $pos = strpos($url, "v=");
  1365. $id = '';
  1366. //If false try other options
  1367. if ($pos === false) {
  1368. $url_parsed = parse_url($url);
  1369. //Youtube shortener
  1370. //http://youtu.be/ID
  1371. $pos = strpos($url, "youtu.be");
  1372. if ($pos == false) {
  1373. $id = '';
  1374. } else {
  1375. return substr($url_parsed['path'], 1);
  1376. }
  1377. //if empty try the youtube.com/embed/ID
  1378. if (empty($id)) {
  1379. $pos = strpos($url, "embed");
  1380. if ($pos === false) {
  1381. return '';
  1382. } else {
  1383. return substr($url_parsed['path'], 7);
  1384. }
  1385. }
  1386. } else {
  1387. // Offset the start location to match the beginning of the ID string
  1388. $pos += 2;
  1389. // Get the ID string and return it
  1390. $id = substr($url, $pos, $len);
  1391. return $id;
  1392. }
  1393. }
  1394. /**
  1395. * @param int $course_id
  1396. * @param int $session_id
  1397. * @param int $categoryId
  1398. * @param string $show
  1399. * @param null $token
  1400. * @param bool $showActionLinks
  1401. * @param bool $forceOpenCategories
  1402. *
  1403. * @return string
  1404. */
  1405. public static function listLinksAndCategories(
  1406. $course_id,
  1407. $session_id,
  1408. $categoryId,
  1409. $show = 'none',
  1410. $token = null,
  1411. $showActionLinks = true,
  1412. $forceOpenCategories = false
  1413. ) {
  1414. $categoryId = (int) $categoryId;
  1415. $content = '';
  1416. $categories = self::getLinkCategories($course_id, $session_id);
  1417. $countCategories = count($categories);
  1418. $linksPerCategory = self::showLinksPerCategory(0, $course_id, $session_id, $showActionLinks);
  1419. if ($showActionLinks) {
  1420. /* Action Links */
  1421. $content = '<div class="actions">';
  1422. if (api_is_allowed_to_edit(null, true)) {
  1423. $content .= '<a href="'.api_get_self().'?'.api_get_cidreq(
  1424. ).'&action=addlink&category_id='.$categoryId.'">'.
  1425. Display::return_icon('new_link.png', get_lang('LinkAdd'), '', ICON_SIZE_MEDIUM).'</a>';
  1426. $content .= '<a href="'.api_get_self().'?'.api_get_cidreq(
  1427. ).'&action=addcategory&category_id='.$categoryId.'">'.
  1428. Display::return_icon('new_folder.png', get_lang('CategoryAdd'), '', ICON_SIZE_MEDIUM).'</a>';
  1429. }
  1430. if (!empty($countCategories)) {
  1431. $content .= '<a href="'.api_get_self().'?'.api_get_cidreq().'&action=list&show=none">';
  1432. $content .= Display::return_icon(
  1433. 'forum_listview.png',
  1434. get_lang('FlatView'),
  1435. '',
  1436. ICON_SIZE_MEDIUM
  1437. ).' </a>';
  1438. $content .= '<a href="'.api_get_self().'?'.api_get_cidreq().'&action=list&show=all">';
  1439. $content .= Display::return_icon(
  1440. 'forum_nestedview.png',
  1441. get_lang('NestedView'),
  1442. '',
  1443. ICON_SIZE_MEDIUM
  1444. ).'</a>';
  1445. }
  1446. $content .= Display::url(
  1447. Display::return_icon('pdf.png', get_lang('ExportToPdf'), '', ICON_SIZE_MEDIUM),
  1448. api_get_self().'?'.api_get_cidreq().'&action=export'
  1449. );
  1450. $content .= '</div>';
  1451. }
  1452. if (empty($countCategories)) {
  1453. $content .= $linksPerCategory;
  1454. } else {
  1455. if (!empty($linksPerCategory)) {
  1456. $content .= Display::panel($linksPerCategory, get_lang('NoCategory'));
  1457. }
  1458. }
  1459. $counter = 0;
  1460. foreach ($categories as $myrow) {
  1461. // Student don't see invisible categories.
  1462. if (!api_is_allowed_to_edit(null, true)) {
  1463. if ($myrow['visibility'] == 0) {
  1464. continue;
  1465. }
  1466. }
  1467. // Validation when belongs to a session
  1468. $showChildren = $categoryId == $myrow['id'] || $show === 'all';
  1469. if ($forceOpenCategories) {
  1470. $showChildren = true;
  1471. }
  1472. $strVisibility = '';
  1473. $visibilityClass = null;
  1474. if ($myrow['visibility'] == '1') {
  1475. $strVisibility = '<a href="link.php?'.api_get_cidreq().'&sec_token='.$token.'&action=invisible&id='.$myrow['id'].'&scope='.TOOL_LINK_CATEGORY.'" title="'.get_lang('Hide').'">'.
  1476. Display::return_icon('visible.png', get_lang('Hide'), [], ICON_SIZE_SMALL).'</a>';
  1477. } elseif ($myrow['visibility'] == '0') {
  1478. $visibilityClass = 'text-muted';
  1479. $strVisibility = ' <a href="link.php?'.api_get_cidreq().'&sec_token='.$token.'&action=visible&id='.$myrow['id'].'&scope='.TOOL_LINK_CATEGORY.'" title="'.get_lang('Show').'">'.
  1480. Display::return_icon('invisible.png', get_lang('Show'), [], ICON_SIZE_SMALL).'</a>';
  1481. }
  1482. $header = '';
  1483. if ($showChildren) {
  1484. $header .= '<a class="'.$visibilityClass.'" href="'.api_get_self().'?'.api_get_cidreq().'&category_id=">';
  1485. $header .= Display::return_icon('forum_nestedview.png');
  1486. } else {
  1487. $header .= '<a class="'.$visibilityClass.'" href="'.api_get_self().'?'.api_get_cidreq().'&category_id='.$myrow['id'].'">';
  1488. $header .= Display::return_icon('forum_listview.png');
  1489. }
  1490. $header .= Security::remove_XSS($myrow['category_title']).'</a>';
  1491. if ($showActionLinks) {
  1492. if (api_is_allowed_to_edit(null, true)) {
  1493. if ($session_id == $myrow['session_id']) {
  1494. $header .= $strVisibility;
  1495. $header .= self::showCategoryAdminTools($myrow, $counter, count($categories));
  1496. } else {
  1497. $header .= get_lang('EditionNotAvailableFromSession');
  1498. }
  1499. }
  1500. }
  1501. $childrenContent = '';
  1502. if ($showChildren) {
  1503. $childrenContent = self::showLinksPerCategory(
  1504. $myrow['id'],
  1505. api_get_course_int_id(),
  1506. api_get_session_id()
  1507. );
  1508. }
  1509. $content .= Display::panel($myrow['description'].$childrenContent, $header);
  1510. $counter++;
  1511. }
  1512. return $content;
  1513. }
  1514. /**
  1515. * @param int $linkId
  1516. * @param string $action
  1517. * @param null $token
  1518. *
  1519. * @return FormValidator
  1520. */
  1521. public static function getLinkForm($linkId, $action, $token = null)
  1522. {
  1523. $course_id = api_get_course_int_id();
  1524. $session_id = api_get_session_id();
  1525. $linkInfo = self::getLinkInfo($linkId);
  1526. $categoryId = isset($linkInfo['category_id']) ? $linkInfo['category_id'] : '';
  1527. $lpId = isset($_GET['lp_id']) ? Security::remove_XSS($_GET['lp_id']) : null;
  1528. $form = new FormValidator(
  1529. 'link',
  1530. 'post',
  1531. api_get_self().'?action='.$action.
  1532. '&category_id='.$categoryId.
  1533. '&'.api_get_cidreq().
  1534. '&id='.$linkId.
  1535. '&sec_token='.$token
  1536. );
  1537. if ($action == 'addlink') {
  1538. $form->addHeader(get_lang('LinkAdd'));
  1539. } else {
  1540. $form->addHeader(get_lang('LinkMod'));
  1541. }
  1542. $target_link = '_blank';
  1543. $title = '';
  1544. $category = '';
  1545. $onhomepage = '';
  1546. $description = '';
  1547. if (!empty($linkInfo)) {
  1548. $urllink = $linkInfo['url'];
  1549. $title = $linkInfo['title'];
  1550. $description = $linkInfo['description'];
  1551. $category = $linkInfo['category_id'];
  1552. if ($linkInfo['on_homepage'] != 0) {
  1553. $onhomepage = 1;
  1554. }
  1555. $target_link = $linkInfo['target'];
  1556. }
  1557. $form->addHidden('id', $linkId);
  1558. $form->addText('url', 'URL');
  1559. $form->addRule('url', get_lang('GiveURL'), 'url');
  1560. $form->addText('title', get_lang('LinkName'));
  1561. $form->addTextarea('description', get_lang('Description'));
  1562. $resultcategories = self::getLinkCategories($course_id, $session_id);
  1563. $options = ['0' => '--'];
  1564. if (!empty($resultcategories)) {
  1565. foreach ($resultcategories as $myrow) {
  1566. $options[$myrow['id']] = $myrow['category_title'];
  1567. }
  1568. }
  1569. $form->addSelect('category_id', get_lang('Category'), $options);
  1570. $form->addCheckBox('on_homepage', null, get_lang('OnHomepage'));
  1571. $targets = [
  1572. '_self' => get_lang('LinkOpenSelf'),
  1573. '_blank' => get_lang('LinkOpenBlank'),
  1574. '_parent' => get_lang('LinkOpenParent'),
  1575. '_top' => get_lang('LinkOpenTop'),
  1576. ];
  1577. $form->addSelect(
  1578. 'target',
  1579. [
  1580. get_lang('LinkTarget'),
  1581. get_lang('AddTargetOfLinkOnHomepage'),
  1582. ],
  1583. $targets
  1584. );
  1585. $defaults = [
  1586. 'url' => empty($urllink) ? 'http://' : Security::remove_XSS($urllink),
  1587. 'title' => Security::remove_XSS($title),
  1588. 'category_id' => $category,
  1589. 'on_homepage' => $onhomepage,
  1590. 'description' => $description,
  1591. 'target' => $target_link,
  1592. ];
  1593. if (api_get_setting('search_enabled') == 'true') {
  1594. require_once api_get_path(LIBRARY_PATH).'specific_fields_manager.lib.php';
  1595. $specific_fields = get_specific_field_list();
  1596. $form->addCheckBox('index_document', get_lang('SearchFeatureDoIndexLink'), get_lang('Yes'));
  1597. foreach ($specific_fields as $specific_field) {
  1598. $default_values = '';
  1599. if ($action == 'editlink') {
  1600. $filter = [
  1601. 'field_id' => $specific_field['id'],
  1602. 'ref_id' => intval($_GET['id']),
  1603. 'tool_id' => '\''.TOOL_LINK.'\'',
  1604. ];
  1605. $values = get_specific_field_values_list($filter, ['value']);
  1606. if (!empty($values)) {
  1607. $arr_str_values = [];
  1608. foreach ($values as $value) {
  1609. $arr_str_values[] = $value['value'];
  1610. }
  1611. $default_values = implode(', ', $arr_str_values);
  1612. }
  1613. }
  1614. $form->addText($specific_field['name'], $specific_field['code']);
  1615. $defaults[$specific_field['name']] = $default_values;
  1616. }
  1617. }
  1618. $skillList = Skill::addSkillsToForm($form, ITEM_TYPE_LINK, $linkId);
  1619. $form->addHidden('lp_id', $lpId);
  1620. $form->addButtonSave(get_lang('SaveLink'), 'submitLink');
  1621. $defaults['skills'] = array_keys($skillList);
  1622. $form->setDefaults($defaults);
  1623. return $form;
  1624. }
  1625. /**
  1626. * @param int $id
  1627. * @param string $action
  1628. *
  1629. * @return FormValidator
  1630. */
  1631. public static function getCategoryForm($id, $action)
  1632. {
  1633. $id = (int) $id;
  1634. $action = Security::remove_XSS($action);
  1635. $form = new FormValidator(
  1636. 'category',
  1637. 'post',
  1638. api_get_self().'?action='.$action.'&'.api_get_cidreq()
  1639. );
  1640. $defaults = [];
  1641. if ($action == 'addcategory') {
  1642. $form->addHeader(get_lang('CategoryAdd'));
  1643. $my_cat_title = get_lang('CategoryAdd');
  1644. } else {
  1645. $form->addHeader(get_lang('CategoryMod'));
  1646. $my_cat_title = get_lang('CategoryMod');
  1647. $defaults = self::getCategory($id);
  1648. }
  1649. $form->addHidden('id', $id);
  1650. $form->addText('category_title', get_lang('CategoryName'));
  1651. $form->addTextarea('description', get_lang('Description'));
  1652. $form->addButtonSave($my_cat_title, 'submitCategory');
  1653. $form->setDefaults($defaults);
  1654. return $form;
  1655. }
  1656. /**
  1657. * @param int $id
  1658. *
  1659. * @return array
  1660. */
  1661. public static function getCategory($id)
  1662. {
  1663. $table = Database::get_course_table(TABLE_LINK_CATEGORY);
  1664. $id = (int) $id;
  1665. $courseId = api_get_course_int_id();
  1666. if (empty($id) || empty($courseId)) {
  1667. return [];
  1668. }
  1669. $sql = "SELECT * FROM $table
  1670. WHERE id = $id AND c_id = $courseId";
  1671. $result = Database::query($sql);
  1672. $category = Database::fetch_array($result, 'ASSOC');
  1673. return $category;
  1674. }
  1675. /**
  1676. * Move a link up in its category.
  1677. *
  1678. * @param int $id
  1679. *
  1680. * @return bool
  1681. */
  1682. public static function moveLinkUp($id)
  1683. {
  1684. return self::moveLinkDisplayOrder($id, 'ASC');
  1685. }
  1686. /**
  1687. * Move a link down in its category.
  1688. *
  1689. * @param int $id
  1690. *
  1691. * @return bool
  1692. */
  1693. public static function moveLinkDown($id)
  1694. {
  1695. return self::moveLinkDisplayOrder($id, 'DESC');
  1696. }
  1697. /**
  1698. * @param string $url
  1699. *
  1700. * @return bool
  1701. */
  1702. public static function checkUrl($url)
  1703. {
  1704. // Check if curl is available.
  1705. if (!in_array('curl', get_loaded_extensions())) {
  1706. return false;
  1707. }
  1708. // set URL and other appropriate options
  1709. $defaults = [
  1710. CURLOPT_URL => $url,
  1711. CURLOPT_FOLLOWLOCATION => true, // follow redirects accept youtube.com
  1712. CURLOPT_HEADER => 0,
  1713. CURLOPT_RETURNTRANSFER => true,
  1714. CURLOPT_TIMEOUT => 4,
  1715. ];
  1716. $proxySettings = api_get_configuration_value('proxy_settings');
  1717. if (!empty($proxySettings) &&
  1718. isset($proxySettings['curl_setopt_array'])
  1719. ) {
  1720. $defaults[CURLOPT_PROXY] = $proxySettings['curl_setopt_array']['CURLOPT_PROXY'];
  1721. $defaults[CURLOPT_PROXYPORT] = $proxySettings['curl_setopt_array']['CURLOPT_PROXYPORT'];
  1722. }
  1723. // Create a new cURL resource
  1724. $ch = curl_init();
  1725. curl_setopt_array($ch, $defaults);
  1726. // grab URL and pass it to the browser
  1727. ob_start();
  1728. $result = curl_exec($ch);
  1729. ob_get_clean();
  1730. // close cURL resource, and free up system resources
  1731. curl_close($ch);
  1732. return $result;
  1733. }
  1734. /**
  1735. * Move a link inside its category (display_order field).
  1736. *
  1737. * @param int $id The link ID
  1738. * @param string $direction The direction to sort the links
  1739. *
  1740. * @return bool
  1741. */
  1742. private static function moveLinkDisplayOrder($id, $direction)
  1743. {
  1744. $em = Database::getManager();
  1745. /** @var CLink $link */
  1746. $link = $em->find('ChamiloCourseBundle:CLink', $id);
  1747. if (!$link) {
  1748. return false;
  1749. }
  1750. $compareLinks = $em
  1751. ->getRepository('ChamiloCourseBundle:CLink')
  1752. ->findBy(
  1753. [
  1754. 'cId' => $link->getCId(),
  1755. 'categoryId' => $link->getCategoryId(),
  1756. ],
  1757. ['displayOrder' => $direction]
  1758. );
  1759. /** @var CLink $prevLink */
  1760. $prevLink = null;
  1761. /** @var CLink $compareLink */
  1762. foreach ($compareLinks as $compareLink) {
  1763. if ($compareLink->getId() !== $link->getId()) {
  1764. $prevLink = $compareLink;
  1765. continue;
  1766. }
  1767. if (!$prevLink) {
  1768. return false;
  1769. }
  1770. $newPrevLinkDisplayOrder = $link->getDisplayOrder();
  1771. $newLinkDisplayOrder = $prevLink->getDisplayOrder();
  1772. $link->setDisplayOrder($newLinkDisplayOrder);
  1773. $prevLink->setDisplayOrder($newPrevLinkDisplayOrder);
  1774. $em->merge($prevLink);
  1775. $em->merge($link);
  1776. break;
  1777. }
  1778. $em->flush();
  1779. return true;
  1780. }
  1781. }