xref: /plugin/recommend/action.php (revision 9fd061b78af6c208ae7b9a25b5c4b3084d527765)
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;
64*9fd061b7SAnna Dabrowska        // menu item is only for logged in users
65*9fd061b7SAnna 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
1312937e9a5SAnna Dabrowska        $id = getID(); // we may run in AJAX context
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'];
216c18a4184SAdrian Lang        $sender = $s_name . ' <' . $_POST['s_email'] . '>';
2174654e4deSAdrian Lang
2182937e9a5SAnna Dabrowska        $id = $INPUT->filter('cleanID')->str('id');
2192937e9a5SAnna Dabrowska        if ($id === '' || !page_exists($id)) throw new \Exception($this->getLang('err_page'));
2204654e4deSAdrian Lang
2212937e9a5SAnna Dabrowska        $comment = $INPUT->str('comment');
2224654e4deSAdrian Lang
2235187ba70SAnna Dabrowska        /* Prepare mail text */
224369955a1SAnna Dabrowska        $mailtext = file_get_contents($this->localFN('template'));
2254654e4deSAdrian Lang
2264654e4deSAdrian Lang        global $conf;
2275187ba70SAnna Dabrowska        foreach (array('PAGE' => $id,
2284654e4deSAdrian Lang                       'SITE' => $conf['title'],
2292937e9a5SAnna Dabrowska                       'URL'  => wl($id, '', true),
2304654e4deSAdrian Lang                       'COMMENT' => $comment,
231c18a4184SAdrian Lang                       'AUTHOR' => $s_name) as $var => $val) {
2324654e4deSAdrian Lang            $mailtext = str_replace('@' . $var . '@', $val, $mailtext);
2334654e4deSAdrian Lang        }
2342cdbf48fSAdrian Lang        /* Limit to two empty lines. */
2354654e4deSAdrian Lang        $mailtext = preg_replace('/\n{4,}/', "\n\n\n", $mailtext);
2364654e4deSAdrian Lang
2375187ba70SAnna Dabrowska        $mailHelper->sendMail($recipients, $mailtext, $sender);
238945a672eSAnna Dabrowska
2392937e9a5SAnna Dabrowska        /** @var helper_plugin_recommend_log $log */
2402937e9a5SAnna Dabrowska        $log = new helper_plugin_recommend_log(date('Y-m'));
2415187ba70SAnna Dabrowska        $log->writeEntry($id, $sender, $recipients, $comment);
2422937e9a5SAnna Dabrowska    }
243945a672eSAnna Dabrowska
244945a672eSAnna Dabrowska    /**
245945a672eSAnna Dabrowska     * show success message in ajax mode
246945a672eSAnna Dabrowska     */
247945a672eSAnna Dabrowska    protected function ajaxSuccess()
248945a672eSAnna Dabrowska    {
249945a672eSAnna Dabrowska        echo '<form id="plugin__recommend" accept-charset="utf-8" method="post" action="?do=recommend">';
250945a672eSAnna Dabrowska        echo '<div class="no">';
251945a672eSAnna Dabrowska        echo '<span class="ui-icon ui-icon-circle-check" style="float: left; margin: 0 7px 50px 0;"></span>';
252945a672eSAnna Dabrowska        echo '<p>' . $this->getLang('done') . '</p>';
253945a672eSAnna Dabrowska        echo '<button type="reset" class="button">' . $this->getLang('close') . '</button>';
254945a672eSAnna Dabrowska        echo '</div>';
255945a672eSAnna Dabrowska        echo '</form>';
256945a672eSAnna Dabrowska    }
2574654e4deSAdrian Lang}
258