123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448 |
- <?php
- namespace dokuwiki;
- use dokuwiki\Ui;
- use dokuwiki\Utf8\Sort;
- /**
- * Manage all builtin AJAX calls
- *
- * @todo The calls should be refactored out to their own proper classes
- * @package dokuwiki
- */
- class Ajax {
- /**
- * Execute the given call
- *
- * @param string $call name of the ajax call
- */
- public function __construct($call) {
- $callfn = 'call' . ucfirst($call);
- if(method_exists($this, $callfn)) {
- $this->$callfn();
- } else {
- $evt = new Extension\Event('AJAX_CALL_UNKNOWN', $call);
- if($evt->advise_before()) {
- print "AJAX call '" . hsc($call) . "' unknown!\n";
- } else {
- $evt->advise_after();
- unset($evt);
- }
- }
- }
- /**
- * Searches for matching pagenames
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- protected function callQsearch() {
- global $lang;
- global $INPUT;
- $maxnumbersuggestions = 50;
- $query = $INPUT->post->str('q');
- if(empty($query)) $query = $INPUT->get->str('q');
- if(empty($query)) return;
- $query = urldecode($query);
- $data = ft_pageLookup($query, true, useHeading('navigation'));
- if(!count($data)) return;
- print '<strong>' . $lang['quickhits'] . '</strong>';
- print '<ul>';
- $counter = 0;
- foreach($data as $id => $title) {
- if(useHeading('navigation')) {
- $name = $title;
- } else {
- $ns = getNS($id);
- if($ns) {
- $name = noNS($id) . ' (' . $ns . ')';
- } else {
- $name = $id;
- }
- }
- echo '<li>' . html_wikilink(':' . $id, $name) . '</li>';
- $counter++;
- if($counter > $maxnumbersuggestions) {
- echo '<li>...</li>';
- break;
- }
- }
- print '</ul>';
- }
- /**
- * Support OpenSearch suggestions
- *
- * @link http://www.opensearch.org/Specifications/OpenSearch/Extensions/Suggestions/1.0
- * @author Mike Frysinger <vapier@gentoo.org>
- */
- protected function callSuggestions() {
- global $INPUT;
- $query = cleanID($INPUT->post->str('q'));
- if(empty($query)) $query = cleanID($INPUT->get->str('q'));
- if(empty($query)) return;
- $data = ft_pageLookup($query);
- if(!count($data)) return;
- $data = array_keys($data);
- // limit results to 15 hits
- $data = array_slice($data, 0, 15);
- $data = array_map('trim', $data);
- $data = array_map('noNS', $data);
- $data = array_unique($data);
- Sort::sort($data);
- /* now construct a json */
- $suggestions = array(
- $query, // the original query
- $data, // some suggestions
- array(), // no description
- array() // no urls
- );
- header('Content-Type: application/x-suggestions+json');
- print json_encode($suggestions);
- }
- /**
- * Refresh a page lock and save draft
- *
- * Andreas Gohr <andi@splitbrain.org>
- */
- protected function callLock() {
- global $ID;
- global $INFO;
- global $INPUT;
- $ID = cleanID($INPUT->post->str('id'));
- if(empty($ID)) return;
- $INFO = pageinfo();
- $response = [
- 'errors' => [],
- 'lock' => '0',
- 'draft' => '',
- ];
- if(!$INFO['writable']) {
- $response['errors'][] = 'Permission to write this page has been denied.';
- echo json_encode($response);
- return;
- }
- if(!checklock($ID)) {
- lock($ID);
- $response['lock'] = '1';
- }
- $draft = new Draft($ID, $INFO['client']);
- if ($draft->saveDraft()) {
- $response['draft'] = $draft->getDraftMessage();
- } else {
- $response['errors'] = array_merge($response['errors'], $draft->getErrors());
- }
- echo json_encode($response);
- }
- /**
- * Delete a draft
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- protected function callDraftdel() {
- global $INPUT;
- $id = cleanID($INPUT->str('id'));
- if(empty($id)) return;
- $client = $INPUT->server->str('REMOTE_USER');
- if(!$client) $client = clientIP(true);
- $draft = new Draft($id, $client);
- if ($draft->isDraftAvailable() && checkSecurityToken()) {
- $draft->deleteDraft();
- }
- }
- /**
- * Return subnamespaces for the Mediamanager
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- protected function callMedians() {
- global $conf;
- global $INPUT;
- // wanted namespace
- $ns = cleanID($INPUT->post->str('ns'));
- $dir = utf8_encodeFN(str_replace(':', '/', $ns));
- $lvl = count(explode(':', $ns));
- $data = array();
- search($data, $conf['mediadir'], 'search_index', array('nofiles' => true), $dir);
- foreach(array_keys($data) as $item) {
- $data[$item]['level'] = $lvl + 1;
- }
- echo html_buildlist($data, 'idx', 'media_nstree_item', 'media_nstree_li');
- }
- /**
- * Return list of files for the Mediamanager
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- protected function callMedialist() {
- global $NS;
- global $INPUT;
- $NS = cleanID($INPUT->post->str('ns'));
- $sort = $INPUT->post->bool('recent') ? 'date' : 'natural';
- if($INPUT->post->str('do') == 'media') {
- tpl_mediaFileList();
- } else {
- tpl_mediaContent(true, $sort);
- }
- }
- /**
- * Return the content of the right column
- * (image details) for the Mediamanager
- *
- * @author Kate Arzamastseva <pshns@ukr.net>
- */
- protected function callMediadetails() {
- global $IMG, $JUMPTO, $REV, $fullscreen, $INPUT;
- $fullscreen = true;
- require_once(DOKU_INC . 'lib/exe/mediamanager.php');
- $image = '';
- if($INPUT->has('image')) $image = cleanID($INPUT->str('image'));
- if(isset($IMG)) $image = $IMG;
- if(isset($JUMPTO)) $image = $JUMPTO;
- $rev = false;
- if(isset($REV) && !$JUMPTO) $rev = $REV;
- html_msgarea();
- tpl_mediaFileDetails($image, $rev);
- }
- /**
- * Returns image diff representation for mediamanager
- *
- * @author Kate Arzamastseva <pshns@ukr.net>
- */
- protected function callMediadiff() {
- global $INPUT;
- $image = '';
- if($INPUT->has('image')) $image = cleanID($INPUT->str('image'));
- (new Ui\MediaDiff($image))->preference('fromAjax', true)->show();
- }
- /**
- * Manages file uploads
- *
- * @author Kate Arzamastseva <pshns@ukr.net>
- */
- protected function callMediaupload() {
- global $NS, $MSG, $INPUT;
- $id = '';
- if(isset($_FILES['qqfile']['tmp_name'])) {
- $id = $INPUT->post->str('mediaid', $_FILES['qqfile']['name']);
- } elseif($INPUT->get->has('qqfile')) {
- $id = $INPUT->get->str('qqfile');
- }
- $id = cleanID($id);
- $NS = $INPUT->str('ns');
- $ns = $NS . ':' . getNS($id);
- $AUTH = auth_quickaclcheck("$ns:*");
- if($AUTH >= AUTH_UPLOAD) {
- io_createNamespace("$ns:xxx", 'media');
- }
- if(isset($_FILES['qqfile']['error']) && $_FILES['qqfile']['error']) unset($_FILES['qqfile']);
- $res = false;
- if(isset($_FILES['qqfile']['tmp_name'])) $res = media_upload($NS, $AUTH, $_FILES['qqfile']);
- if($INPUT->get->has('qqfile')) $res = media_upload_xhr($NS, $AUTH);
- if($res) {
- $result = array(
- 'success' => true,
- 'link' => media_managerURL(array('ns' => $ns, 'image' => $NS . ':' . $id), '&'),
- 'id' => $NS . ':' . $id,
- 'ns' => $NS
- );
- } else {
- $error = '';
- if(isset($MSG)) {
- foreach($MSG as $msg) {
- $error .= $msg['msg'];
- }
- }
- $result = array(
- 'error' => $error,
- 'ns' => $NS
- );
- }
- header('Content-Type: application/json');
- echo json_encode($result);
- }
- /**
- * Return sub index for index view
- *
- * @author Andreas Gohr <andi@splitbrain.org>
- */
- protected function callIndex() {
- global $conf;
- global $INPUT;
- // wanted namespace
- $ns = cleanID($INPUT->post->str('idx'));
- $dir = utf8_encodeFN(str_replace(':', '/', $ns));
- $lvl = count(explode(':', $ns));
- $data = array();
- search($data, $conf['datadir'], 'search_index', array('ns' => $ns), $dir);
- foreach (array_keys($data) as $item) {
- $data[$item]['level'] = $lvl + 1;
- }
- $idx = new Ui\Index;
- echo html_buildlist($data, 'idx', [$idx,'formatListItem'], [$idx,'tagListItem']);
- }
- /**
- * List matching namespaces and pages for the link wizard
- *
- * @author Andreas Gohr <gohr@cosmocode.de>
- */
- protected function callLinkwiz() {
- global $conf;
- global $lang;
- global $INPUT;
- $q = ltrim(trim($INPUT->post->str('q')), ':');
- $id = noNS($q);
- $ns = getNS($q);
- $ns = cleanID($ns);
- $id = cleanID($id);
- $nsd = utf8_encodeFN(str_replace(':', '/', $ns));
- $data = array();
- if($q !== '' && $ns === '') {
- // use index to lookup matching pages
- $pages = ft_pageLookup($id, true);
- // If 'useheading' option is 'always' or 'content',
- // search page titles with original query as well.
- if ($conf['useheading'] == '1' || $conf['useheading'] == 'content') {
- $pages = array_merge($pages, ft_pageLookup($q, true, true));
- asort($pages, SORT_STRING);
- }
-
- // result contains matches in pages and namespaces
- // we now extract the matching namespaces to show
- // them seperately
- $dirs = array();
- foreach($pages as $pid => $title) {
- if(strpos(getNS($pid), $id) !== false) {
- // match was in the namespace
- $dirs[getNS($pid)] = 1; // assoc array avoids dupes
- } else {
- // it is a matching page, add it to the result
- $data[] = array(
- 'id' => $pid,
- 'title' => $title,
- 'type' => 'f',
- );
- }
- unset($pages[$pid]);
- }
- foreach($dirs as $dir => $junk) {
- $data[] = array(
- 'id' => $dir,
- 'type' => 'd',
- );
- }
- } else {
- $opts = array(
- 'depth' => 1,
- 'listfiles' => true,
- 'listdirs' => true,
- 'pagesonly' => true,
- 'firsthead' => true,
- 'sneakyacl' => $conf['sneaky_index'],
- );
- if($id) $opts['filematch'] = '^.*\/' . $id;
- if($id) $opts['dirmatch'] = '^.*\/' . $id;
- search($data, $conf['datadir'], 'search_universal', $opts, $nsd);
- // add back to upper
- if($ns) {
- array_unshift(
- $data, array(
- 'id' => getNS($ns),
- 'type' => 'u',
- )
- );
- }
- }
- // fixme sort results in a useful way ?
- if(!count($data)) {
- echo $lang['nothingfound'];
- exit;
- }
- // output the found data
- $even = 1;
- foreach($data as $item) {
- $even *= -1; //zebra
- if(($item['type'] == 'd' || $item['type'] == 'u') && $item['id'] !== '') $item['id'] .= ':';
- $link = wl($item['id']);
- echo '<div class="' . (($even > 0) ? 'even' : 'odd') . ' type_' . $item['type'] . '">';
- if($item['type'] == 'u') {
- $name = $lang['upperns'];
- } else {
- $name = hsc($item['id']);
- }
- echo '<a href="' . $link . '" title="' . hsc($item['id']) . '" class="wikilink1">' . $name . '</a>';
- if(!blank($item['title'])) {
- echo '<span>' . hsc($item['title']) . '</span>';
- }
- echo '</div>';
- }
- }
- }
|