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; 6445cf09e9SAnna Dabrowska 6545cf09e9SAnna Dabrowska array_splice($event->data['items'], -1, 0, [new \dokuwiki\plugin\recommend\MenuItem()]); 6645cf09e9SAnna Dabrowska } 675187ba70SAnna Dabrowska 682937e9a5SAnna Dabrowska /** 699121cb07SAnna Dabrowska * Autocomplete 709121cb07SAnna Dabrowska * @param Doku_Event $event 719121cb07SAnna Dabrowska * @throws Exception 729121cb07SAnna Dabrowska * @author Andreas Gohr 739121cb07SAnna Dabrowska * 749121cb07SAnna Dabrowska */ 759121cb07SAnna Dabrowska public function autocomplete(Doku_Event $event) 769121cb07SAnna Dabrowska { 779121cb07SAnna Dabrowska 789121cb07SAnna Dabrowska if ($event->data !=='plugin_recommend_ac') { 799121cb07SAnna Dabrowska return; 809121cb07SAnna Dabrowska } 819121cb07SAnna Dabrowska 829121cb07SAnna Dabrowska $event->preventDefault(); 839121cb07SAnna Dabrowska $event->stopPropagation(); 849121cb07SAnna Dabrowska 859121cb07SAnna Dabrowska /** @var \DokuWiki_Auth_Plugin $auth */ 869121cb07SAnna Dabrowska global $auth; 879121cb07SAnna Dabrowska global $INPUT; 889121cb07SAnna Dabrowska 899121cb07SAnna Dabrowska if (!$auth->canDo('getUsers')) { 909121cb07SAnna Dabrowska throw new Exception('The user backend can not search for users'); 919121cb07SAnna Dabrowska } 929121cb07SAnna Dabrowska 939121cb07SAnna Dabrowska header('Content-Type: application/json'); 949121cb07SAnna Dabrowska 959121cb07SAnna Dabrowska // check minimum length 969121cb07SAnna Dabrowska $lookup = trim($INPUT->str('search')); 979121cb07SAnna Dabrowska if (utf8_strlen($lookup) < 3) { 989121cb07SAnna Dabrowska echo json_encode([]); 999121cb07SAnna Dabrowska return; 1009121cb07SAnna Dabrowska } 1019121cb07SAnna Dabrowska 1029121cb07SAnna Dabrowska // find users by login and name 1039121cb07SAnna Dabrowska $logins = $auth->retrieveUsers(0, 10, ['user' => $lookup]); 1049121cb07SAnna Dabrowska if (count($logins) < 10) { 1059121cb07SAnna Dabrowska $logins = array_merge($logins, $auth->retrieveUsers(0, 10, ['name' => $lookup])); 1069121cb07SAnna Dabrowska } 1079121cb07SAnna Dabrowska 1089121cb07SAnna Dabrowska // reformat result for jQuery UI Autocomplete 1099121cb07SAnna Dabrowska $users = []; 1109121cb07SAnna Dabrowska foreach ($logins as $login => $info) { 1119121cb07SAnna Dabrowska $users[] = [ 1129121cb07SAnna Dabrowska 'label' => $info['name'] . ' [' . $login . ']', 1139121cb07SAnna Dabrowska 'value' => $login 1149121cb07SAnna Dabrowska ]; 1159121cb07SAnna Dabrowska } 1169121cb07SAnna Dabrowska 1179121cb07SAnna Dabrowska echo json_encode($users); 1189121cb07SAnna Dabrowska } 1199121cb07SAnna Dabrowska 1209121cb07SAnna Dabrowska /** 1212937e9a5SAnna Dabrowska * Returns rendered form 1222937e9a5SAnna Dabrowska * 1232937e9a5SAnna Dabrowska * @return string 1242937e9a5SAnna Dabrowska */ 1252937e9a5SAnna Dabrowska protected function getForm() 1262937e9a5SAnna Dabrowska { 1272937e9a5SAnna Dabrowska global $INPUT; 1282937e9a5SAnna Dabrowska 1292937e9a5SAnna Dabrowska $id = getID(); // we may run in AJAX context 1302937e9a5SAnna Dabrowska if ($id === '') throw new \RuntimeException('No ID given'); 1312937e9a5SAnna Dabrowska 1322937e9a5SAnna Dabrowska $form = new \dokuwiki\Form\Form([ 1332937e9a5SAnna Dabrowska 'action' => wl($id, ['do' => 'recommend'], false, '&'), 134945a672eSAnna Dabrowska 'id' => 'plugin__recommend', 1352937e9a5SAnna Dabrowska ]); 1362937e9a5SAnna Dabrowska $form->setHiddenField('id', $id); // we need it for the ajax call 1372937e9a5SAnna Dabrowska 138b8b86b6cSAnna Dabrowska /** @var helper_plugin_recommend_assignment $helper */ 139b8b86b6cSAnna Dabrowska $helper = plugin_load('helper', 'recommend_assignment'); 140b8b86b6cSAnna Dabrowska $template = $helper->loadMatchingTemplate(); 141b8b86b6cSAnna Dabrowska 1422937e9a5SAnna Dabrowska if ($INPUT->server->has('REMOTE_USER')) { 143e596e22bSAdrian Lang global $USERINFO; 1442937e9a5SAnna Dabrowska $form->setHiddenField('s_name', $USERINFO['name']); 1452937e9a5SAnna Dabrowska $form->setHiddenField('s_email', $USERINFO['mail']); 146e596e22bSAdrian Lang } else { 1472937e9a5SAnna Dabrowska $form->addTextInput('s_name', $this->getLang('yourname'))->addClass('edit'); 1482937e9a5SAnna Dabrowska $form->addTextInput('s_email', $this->getLang('youremailaddress'))->addClass('edit'); 149e596e22bSAdrian Lang } 150e596e22bSAdrian Lang 151*d0dae647SAnna Dabrowska $form->addTextInput('r_email', $this->getLang('recipients')) 152*d0dae647SAnna Dabrowska ->addClass('edit') 153*d0dae647SAnna Dabrowska ->val($template['user'] ?? ''); 154*d0dae647SAnna Dabrowska $form->addTextInput('subject', $this->getLang('subject')) 155*d0dae647SAnna Dabrowska ->addClass('edit') 156*d0dae647SAnna Dabrowska ->val($template['subject'] ?? ''); 157b8b86b6cSAnna Dabrowska $form->addTextarea('comment', $this->getLang('message')) 158b8b86b6cSAnna Dabrowska ->attr('rows', '8') 159b8b86b6cSAnna Dabrowska ->attr('cols', '40') 160b8b86b6cSAnna Dabrowska ->addClass('edit') 161*d0dae647SAnna Dabrowska ->val($template['message'] ?? ''); 1622937e9a5SAnna Dabrowska 1632937e9a5SAnna Dabrowska /** @var helper_plugin_captcha $captcha */ 1642937e9a5SAnna Dabrowska $captcha = plugin_load('helper', 'captcha'); 1652937e9a5SAnna Dabrowska if ($captcha) $form->addHTML($captcha->getHTML()); 1662937e9a5SAnna Dabrowska 1672937e9a5SAnna Dabrowska $form->addTagOpen('div')->addClass('buttons'); 1682937e9a5SAnna Dabrowska $form->addButton('submit', $this->getLang('send'))->attr('type', 'submit'); 1692937e9a5SAnna Dabrowska $form->addTagClose('div'); 1702937e9a5SAnna Dabrowska 1712937e9a5SAnna Dabrowska return $form->toHTML(); 1724654e4deSAdrian Lang } 1734654e4deSAdrian Lang 1742937e9a5SAnna Dabrowska /** 175945a672eSAnna Dabrowska * Handles form submission 1762937e9a5SAnna Dabrowska * 1772937e9a5SAnna Dabrowska * @throws Exception 1782937e9a5SAnna Dabrowska */ 179945a672eSAnna Dabrowska protected function handlePost() 1802937e9a5SAnna Dabrowska { 181945a672eSAnna Dabrowska global $INPUT; 182945a672eSAnna Dabrowska 1832937e9a5SAnna Dabrowska if (!checkSecurityToken()) { 1842937e9a5SAnna Dabrowska throw new \Exception('Security token did not match'); 1852937e9a5SAnna Dabrowska } 1862937e9a5SAnna Dabrowska 1875187ba70SAnna Dabrowska /** @var helper_plugin_recommend_mail $mailHelper */ 1885187ba70SAnna Dabrowska $mailHelper = plugin_load('helper', 'recommend_mail'); 1895187ba70SAnna Dabrowska 190945a672eSAnna Dabrowska // Captcha plugin 1915187ba70SAnna Dabrowska $captcha = null; 1925187ba70SAnna Dabrowska if (@is_dir(DOKU_PLUGIN . 'captcha')) $captcha = plugin_load('helper','captcha'); 1935187ba70SAnna Dabrowska if (!is_null($captcha) && $captcha->isEnabled() && !$captcha->check()) { 194945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_captcha')); 195e596e22bSAdrian Lang } 196e596e22bSAdrian Lang 1975187ba70SAnna Dabrowska /* Validate input */ 1985187ba70SAnna Dabrowska $recipients = $INPUT->str('r_email'); 1995187ba70SAnna Dabrowska 2005187ba70SAnna Dabrowska if (empty($recipients)) { 201945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_recipient')); 2024654e4deSAdrian Lang } 203e596e22bSAdrian Lang 2045187ba70SAnna Dabrowska $recipients = $mailHelper->resolveRecipients($recipients); 2055187ba70SAnna Dabrowska $recipients = implode(',', $recipients); 2065187ba70SAnna Dabrowska 207e596e22bSAdrian Lang if (!isset($_POST['s_email']) || !mail_isvalid($_POST['s_email'])) { 208945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_sendermail')); 209e596e22bSAdrian Lang } 210c18a4184SAdrian Lang if (!isset($_POST['s_name']) || trim($_POST['s_name']) === '') { 211945a672eSAnna Dabrowska throw new \Exception($this->getLang('err_sendername')); 212c18a4184SAdrian Lang } 213c18a4184SAdrian Lang $s_name = $_POST['s_name']; 214c18a4184SAdrian Lang $sender = $s_name . ' <' . $_POST['s_email'] . '>'; 2154654e4deSAdrian Lang 2162937e9a5SAnna Dabrowska $id = $INPUT->filter('cleanID')->str('id'); 2172937e9a5SAnna Dabrowska if ($id === '' || !page_exists($id)) throw new \Exception($this->getLang('err_page')); 2184654e4deSAdrian Lang 2192937e9a5SAnna Dabrowska $comment = $INPUT->str('comment'); 2204654e4deSAdrian Lang 2215187ba70SAnna Dabrowska /* Prepare mail text */ 222369955a1SAnna Dabrowska $mailtext = file_get_contents($this->localFN('template')); 2234654e4deSAdrian Lang 2244654e4deSAdrian Lang global $conf; 2255187ba70SAnna Dabrowska foreach (array('PAGE' => $id, 2264654e4deSAdrian Lang 'SITE' => $conf['title'], 2272937e9a5SAnna Dabrowska 'URL' => wl($id, '', true), 2284654e4deSAdrian Lang 'COMMENT' => $comment, 229c18a4184SAdrian Lang 'AUTHOR' => $s_name) as $var => $val) { 2304654e4deSAdrian Lang $mailtext = str_replace('@' . $var . '@', $val, $mailtext); 2314654e4deSAdrian Lang } 2322cdbf48fSAdrian Lang /* Limit to two empty lines. */ 2334654e4deSAdrian Lang $mailtext = preg_replace('/\n{4,}/', "\n\n\n", $mailtext); 2344654e4deSAdrian Lang 2355187ba70SAnna Dabrowska $mailHelper->sendMail($recipients, $mailtext, $sender); 236945a672eSAnna Dabrowska 2372937e9a5SAnna Dabrowska /** @var helper_plugin_recommend_log $log */ 2382937e9a5SAnna Dabrowska $log = new helper_plugin_recommend_log(date('Y-m')); 2395187ba70SAnna Dabrowska $log->writeEntry($id, $sender, $recipients, $comment); 2402937e9a5SAnna Dabrowska } 241945a672eSAnna Dabrowska 242945a672eSAnna Dabrowska /** 243945a672eSAnna Dabrowska * show success message in ajax mode 244945a672eSAnna Dabrowska */ 245945a672eSAnna Dabrowska protected function ajaxSuccess() 246945a672eSAnna Dabrowska { 247945a672eSAnna Dabrowska echo '<form id="plugin__recommend" accept-charset="utf-8" method="post" action="?do=recommend">'; 248945a672eSAnna Dabrowska echo '<div class="no">'; 249945a672eSAnna Dabrowska echo '<span class="ui-icon ui-icon-circle-check" style="float: left; margin: 0 7px 50px 0;"></span>'; 250945a672eSAnna Dabrowska echo '<p>' . $this->getLang('done') . '</p>'; 251945a672eSAnna Dabrowska echo '<button type="reset" class="button">' . $this->getLang('close') . '</button>'; 252945a672eSAnna Dabrowska echo '</div>'; 253945a672eSAnna Dabrowska echo '</form>'; 254945a672eSAnna Dabrowska } 2554654e4deSAdrian Lang} 256