14654e4deSAdrian Lang<?php 24654e4deSAdrian Lang 34654e4deSAdrian Langclass action_plugin_recommend extends DokuWiki_Action_Plugin { 44654e4deSAdrian Lang 52937e9a5SAnna Dabrowska public function register(Doku_Event_Handler $controller) { 670f4a435SAdrian Lang foreach (array('ACTION_ACT_PREPROCESS', 'AJAX_CALL_UNKNOWN', 770f4a435SAdrian Lang 'TPL_ACT_UNKNOWN') as $event) { 82937e9a5SAnna Dabrowska $controller->register_hook($event, 'BEFORE', $this, 'handle'); 94654e4deSAdrian Lang } 1045cf09e9SAnna Dabrowska $controller->register_hook('MENU_ITEMS_ASSEMBLY', 'AFTER', $this, 'handleMenu'); 119121cb07SAnna Dabrowska $controller->register_hook('AJAX_CALL_UNKNOWN', 'BEFORE', $this, 'autocomplete'); 124654e4deSAdrian Lang } 134654e4deSAdrian Lang 149121cb07SAnna Dabrowska /** 159121cb07SAnna Dabrowska * Main processing 169121cb07SAnna Dabrowska * 179121cb07SAnna Dabrowska * @param Doku_Event $event 189121cb07SAnna Dabrowska * @return void 199121cb07SAnna Dabrowska */ 202937e9a5SAnna Dabrowska public function handle(Doku_Event $event) { 212937e9a5SAnna Dabrowska if ($event->data !=='recommend') { 224654e4deSAdrian Lang return; 234654e4deSAdrian Lang } 244654e4deSAdrian Lang 254654e4deSAdrian Lang $event->preventDefault(); 264654e4deSAdrian Lang 274654e4deSAdrian Lang if ($event->name === 'ACTION_ACT_PREPROCESS') { 284654e4deSAdrian Lang return; 294654e4deSAdrian Lang } 304654e4deSAdrian Lang 314654e4deSAdrian Lang $event->stopPropagation(); 324654e4deSAdrian Lang 33945a672eSAnna Dabrowska global $INPUT; 34945a672eSAnna Dabrowska 35945a672eSAnna Dabrowska // early output to trigger display msgs even via AJAX. 3670f4a435SAdrian Lang echo ' '; 37945a672eSAnna Dabrowska tpl_flush(); 38945a672eSAnna Dabrowska if ($INPUT->server->str('REQUEST_METHOD') === 'POST') { 39945a672eSAnna Dabrowska try { 40945a672eSAnna Dabrowska $this->handlePost(); 41945a672eSAnna Dabrowska if ($event->name === 'AJAX_CALL_UNKNOWN') { 42945a672eSAnna Dabrowska $this->ajaxSuccess(); // To signal success to AJAX. 43945a672eSAnna Dabrowska } else { 44945a672eSAnna Dabrowska msg($this->getLang('thanks'), 1); 4570f4a435SAdrian Lang } 46945a672eSAnna Dabrowska return; // we're done here 47945a672eSAnna Dabrowska } catch (\Exception $e) { 48945a672eSAnna Dabrowska msg($e->getMessage(), -1); 49945a672eSAnna Dabrowska } 50945a672eSAnna Dabrowska } 51945a672eSAnna Dabrowska 522937e9a5SAnna Dabrowska echo $this->getForm(); 534654e4deSAdrian Lang } 544654e4deSAdrian Lang 555187ba70SAnna Dabrowska /** 565187ba70SAnna Dabrowska * Page menu item 575187ba70SAnna Dabrowska * 585187ba70SAnna Dabrowska * @param Doku_Event $event 595187ba70SAnna Dabrowska * @return void 605187ba70SAnna Dabrowska */ 6145cf09e9SAnna Dabrowska public function handleMenu(Doku_Event $event) 6245cf09e9SAnna Dabrowska { 6345cf09e9SAnna Dabrowska if ($event->data['view'] !== 'page') return; 649fd061b7SAnna Dabrowska // menu item is only for logged in users 659fd061b7SAnna Dabrowska if (empty($_SERVER['REMOTE_USER'])) return; 6645cf09e9SAnna Dabrowska 6745cf09e9SAnna Dabrowska array_splice($event->data['items'], -1, 0, [new \dokuwiki\plugin\recommend\MenuItem()]); 6845cf09e9SAnna Dabrowska } 695187ba70SAnna Dabrowska 702937e9a5SAnna Dabrowska /** 719121cb07SAnna Dabrowska * Autocomplete 729121cb07SAnna Dabrowska * @param Doku_Event $event 739121cb07SAnna Dabrowska * @throws Exception 749121cb07SAnna Dabrowska * @author Andreas Gohr 759121cb07SAnna Dabrowska * 769121cb07SAnna Dabrowska */ 779121cb07SAnna Dabrowska public function autocomplete(Doku_Event $event) 789121cb07SAnna Dabrowska { 799121cb07SAnna Dabrowska 809121cb07SAnna Dabrowska if ($event->data !=='plugin_recommend_ac') { 819121cb07SAnna Dabrowska return; 829121cb07SAnna Dabrowska } 839121cb07SAnna Dabrowska 849121cb07SAnna Dabrowska $event->preventDefault(); 859121cb07SAnna Dabrowska $event->stopPropagation(); 869121cb07SAnna Dabrowska 879121cb07SAnna Dabrowska /** @var \DokuWiki_Auth_Plugin $auth */ 889121cb07SAnna Dabrowska global $auth; 899121cb07SAnna Dabrowska global $INPUT; 909121cb07SAnna Dabrowska 919121cb07SAnna Dabrowska if (!$auth->canDo('getUsers')) { 929121cb07SAnna Dabrowska throw new Exception('The user backend can not search for users'); 939121cb07SAnna Dabrowska } 949121cb07SAnna Dabrowska 959121cb07SAnna Dabrowska header('Content-Type: application/json'); 969121cb07SAnna Dabrowska 979121cb07SAnna Dabrowska // check minimum length 989121cb07SAnna Dabrowska $lookup = trim($INPUT->str('search')); 999121cb07SAnna Dabrowska if (utf8_strlen($lookup) < 3) { 1009121cb07SAnna Dabrowska echo json_encode([]); 1019121cb07SAnna Dabrowska return; 1029121cb07SAnna Dabrowska } 1039121cb07SAnna Dabrowska 1049121cb07SAnna Dabrowska // find users by login and name 1059121cb07SAnna Dabrowska $logins = $auth->retrieveUsers(0, 10, ['user' => $lookup]); 1069121cb07SAnna Dabrowska if (count($logins) < 10) { 1079121cb07SAnna Dabrowska $logins = array_merge($logins, $auth->retrieveUsers(0, 10, ['name' => $lookup])); 1089121cb07SAnna Dabrowska } 1099121cb07SAnna Dabrowska 1109121cb07SAnna Dabrowska // reformat result for jQuery UI Autocomplete 1119121cb07SAnna Dabrowska $users = []; 1129121cb07SAnna Dabrowska foreach ($logins as $login => $info) { 1139121cb07SAnna Dabrowska $users[] = [ 1149121cb07SAnna Dabrowska 'label' => $info['name'] . ' [' . $login . ']', 1159121cb07SAnna Dabrowska 'value' => $login 1169121cb07SAnna Dabrowska ]; 1179121cb07SAnna Dabrowska } 1189121cb07SAnna Dabrowska 1199121cb07SAnna Dabrowska echo json_encode($users); 1209121cb07SAnna Dabrowska } 1219121cb07SAnna Dabrowska 1229121cb07SAnna Dabrowska /** 1232937e9a5SAnna Dabrowska * Returns rendered form 1242937e9a5SAnna Dabrowska * 1252937e9a5SAnna Dabrowska * @return string 1262937e9a5SAnna Dabrowska */ 1272937e9a5SAnna Dabrowska protected function getForm() 1282937e9a5SAnna Dabrowska { 1292937e9a5SAnna Dabrowska global $INPUT; 1302937e9a5SAnna Dabrowska 131*2b1d0a41SInput-BDF $id = cleanID(getID()); // we may run in AJAX context, pathcomponent must be converted to pure namespace 1322937e9a5SAnna Dabrowska if ($id === '') throw new \RuntimeException('No ID given'); 1332937e9a5SAnna Dabrowska 1342937e9a5SAnna Dabrowska $form = new \dokuwiki\Form\Form([ 1352937e9a5SAnna Dabrowska 'action' => wl($id, ['do' => 'recommend'], false, '&'), 136945a672eSAnna Dabrowska 'id' => 'plugin__recommend', 1372937e9a5SAnna Dabrowska ]); 1382937e9a5SAnna Dabrowska $form->setHiddenField('id', $id); // we need it for the ajax call 1392937e9a5SAnna Dabrowska 140b8b86b6cSAnna Dabrowska /** @var helper_plugin_recommend_assignment $helper */ 141b8b86b6cSAnna Dabrowska $helper = plugin_load('helper', 'recommend_assignment'); 142b8b86b6cSAnna Dabrowska $template = $helper->loadMatchingTemplate(); 143b8b86b6cSAnna Dabrowska 1442937e9a5SAnna Dabrowska if ($INPUT->server->has('REMOTE_USER')) { 145e596e22bSAdrian Lang global $USERINFO; 1462937e9a5SAnna Dabrowska $form->setHiddenField('s_name', $USERINFO['name']); 1472937e9a5SAnna Dabrowska $form->setHiddenField('s_email', $USERINFO['mail']); 148e596e22bSAdrian Lang } else { 1492937e9a5SAnna Dabrowska $form->addTextInput('s_name', $this->getLang('yourname'))->addClass('edit'); 1502937e9a5SAnna Dabrowska $form->addTextInput('s_email', $this->getLang('youremailaddress'))->addClass('edit'); 151e596e22bSAdrian Lang } 152e596e22bSAdrian Lang 153d0dae647SAnna Dabrowska $form->addTextInput('r_email', $this->getLang('recipients')) 154d0dae647SAnna Dabrowska ->addClass('edit') 155d0dae647SAnna Dabrowska ->val($template['user'] ?? ''); 156d0dae647SAnna Dabrowska $form->addTextInput('subject', $this->getLang('subject')) 157d0dae647SAnna Dabrowska ->addClass('edit') 158d0dae647SAnna Dabrowska ->val($template['subject'] ?? ''); 159b8b86b6cSAnna Dabrowska $form->addTextarea('comment', $this->getLang('message')) 160b8b86b6cSAnna Dabrowska ->attr('rows', '8') 161b8b86b6cSAnna Dabrowska ->attr('cols', '40') 162b8b86b6cSAnna Dabrowska ->addClass('edit') 163d0dae647SAnna Dabrowska ->val($template['message'] ?? ''); 1642937e9a5SAnna Dabrowska 1652937e9a5SAnna Dabrowska /** @var helper_plugin_captcha $captcha */ 1662937e9a5SAnna Dabrowska $captcha = plugin_load('helper', 'captcha'); 1672937e9a5SAnna Dabrowska if ($captcha) $form->addHTML($captcha->getHTML()); 1682937e9a5SAnna Dabrowska 1692937e9a5SAnna Dabrowska $form->addTagOpen('div')->addClass('buttons'); 1702937e9a5SAnna Dabrowska $form->addButton('submit', $this->getLang('send'))->attr('type', 'submit'); 1712937e9a5SAnna Dabrowska $form->addTagClose('div'); 1722937e9a5SAnna Dabrowska 1732937e9a5SAnna Dabrowska return $form->toHTML(); 1744654e4deSAdrian Lang } 1754654e4deSAdrian Lang 1762937e9a5SAnna Dabrowska /** 177945a672eSAnna Dabrowska * Handles form submission 1782937e9a5SAnna Dabrowska * 1792937e9a5SAnna Dabrowska * @throws Exception 1802937e9a5SAnna Dabrowska */ 181945a672eSAnna Dabrowska protected function handlePost() 1822937e9a5SAnna Dabrowska { 183945a672eSAnna Dabrowska global $INPUT; 184945a672eSAnna Dabrowska 1852937e9a5SAnna Dabrowska if (!checkSecurityToken()) { 1862937e9a5SAnna Dabrowska throw new \Exception('Security token did not match'); 1872937e9a5SAnna Dabrowska } 1882937e9a5SAnna Dabrowska 1895187ba70SAnna Dabrowska /** @var helper_plugin_recommend_mail $mailHelper */ 1905187ba70SAnna Dabrowska $mailHelper = plugin_load('helper', 'recommend_mail'); 1915187ba70SAnna Dabrowska 192945a672eSAnna Dabrowska // Captcha plugin 1935187ba70SAnna Dabrowska $captcha = null; 1945187ba70SAnna Dabrowska if (@is_dir(DOKU_PLUGIN . 'captcha')) $captcha = plugin_load('helper','captcha'); 1955187ba70SAnna Dabrowska if (!is_null($captcha) && $captcha->isEnabled() && !$captcha->check()) { 196945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_captcha')); 197e596e22bSAdrian Lang } 198e596e22bSAdrian Lang 1995187ba70SAnna Dabrowska /* Validate input */ 2005187ba70SAnna Dabrowska $recipients = $INPUT->str('r_email'); 2015187ba70SAnna Dabrowska 2025187ba70SAnna Dabrowska if (empty($recipients)) { 203945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_recipient')); 2044654e4deSAdrian Lang } 205e596e22bSAdrian Lang 2065187ba70SAnna Dabrowska $recipients = $mailHelper->resolveRecipients($recipients); 2075187ba70SAnna Dabrowska $recipients = implode(',', $recipients); 2085187ba70SAnna Dabrowska 209e596e22bSAdrian Lang if (!isset($_POST['s_email']) || !mail_isvalid($_POST['s_email'])) { 210945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_sendermail')); 211e596e22bSAdrian Lang } 212c18a4184SAdrian Lang if (!isset($_POST['s_name']) || trim($_POST['s_name']) === '') { 213945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_sendername')); 214c18a4184SAdrian Lang } 215c18a4184SAdrian Lang $s_name = $_POST['s_name']; 216*2b1d0a41SInput-BDF 217*2b1d0a41SInput-BDF global $conf; 218*2b1d0a41SInput-BDF 219*2b1d0a41SInput-BDF if (!$this->getConf('usewikimail')) { 220c18a4184SAdrian Lang $sender = $s_name . ' <' . $_POST['s_email'] . '>'; 221*2b1d0a41SInput-BDF } 222*2b1d0a41SInput-BDF if ($this->getConf('usewikimail')) { 223*2b1d0a41SInput-BDF $sender = $conf['title'] . ' <' . $conf['mailfrom'] . '>'; 224*2b1d0a41SInput-BDF } 2254654e4deSAdrian Lang 2262937e9a5SAnna Dabrowska $id = $INPUT->filter('cleanID')->str('id'); 2272937e9a5SAnna Dabrowska if ($id === '' || !page_exists($id)) throw new \Exception($this->getLang('err_page')); 2284654e4deSAdrian Lang 2292937e9a5SAnna Dabrowska $comment = $INPUT->str('comment'); 2304654e4deSAdrian Lang 231f1a0181dSAnna Dabrowska $replacements = [ 232*2b1d0a41SInput-BDF 'PAGE' => noNS($id), 2334654e4deSAdrian Lang 'SITE' => $conf['title'], 2342937e9a5SAnna Dabrowska 'URL' => wl($id, '', true), 2354654e4deSAdrian Lang 'COMMENT' => $comment, 236f1a0181dSAnna Dabrowska 'AUTHOR' => $s_name 237f1a0181dSAnna Dabrowska ]; 2384654e4deSAdrian Lang 239f1a0181dSAnna Dabrowska $mailHelper->sendMail($recipients, $sender, $replacements); 240945a672eSAnna Dabrowska 2412937e9a5SAnna Dabrowska /** @var helper_plugin_recommend_log $log */ 2422937e9a5SAnna Dabrowska $log = new helper_plugin_recommend_log(date('Y-m')); 2435187ba70SAnna Dabrowska $log->writeEntry($id, $sender, $recipients, $comment); 2442937e9a5SAnna Dabrowska } 245945a672eSAnna Dabrowska 246945a672eSAnna Dabrowska /** 247945a672eSAnna Dabrowska * show success message in ajax mode 248945a672eSAnna Dabrowska */ 249945a672eSAnna Dabrowska protected function ajaxSuccess() 250945a672eSAnna Dabrowska { 251945a672eSAnna Dabrowska echo '<form id="plugin__recommend" accept-charset="utf-8" method="post" action="?do=recommend">'; 252945a672eSAnna Dabrowska echo '<div class="no">'; 253945a672eSAnna Dabrowska echo '<span class="ui-icon ui-icon-circle-check" style="float: left; margin: 0 7px 50px 0;"></span>'; 254945a672eSAnna Dabrowska echo '<p>' . $this->getLang('done') . '</p>'; 255945a672eSAnna Dabrowska echo '<button type="reset" class="button">' . $this->getLang('close') . '</button>'; 256945a672eSAnna Dabrowska echo '</div>'; 257945a672eSAnna Dabrowska echo '</form>'; 258945a672eSAnna Dabrowska } 2594654e4deSAdrian Lang} 260