Skip to content
Snippets Groups Projects
ajax.tickets.php 48 KiB
Newer Older
Peter Rotich's avatar
Peter Rotich committed
                case 'deleted':
                    if (!$thisstaff->hasPerm(TicketModel::PERM_DELETE, false))
                        $errors['err'] = sprintf(__('You do not have permission %s'),
                                __('to archive/delete tickets'));
Peter Rotich's avatar
Peter Rotich committed
                    break;
                default:
                    $errors['err'] = sprintf('%s %s',
                            __('Unknown or invalid'), __('status'));
        $count = count($_REQUEST['tids']);
Peter Rotich's avatar
Peter Rotich committed
        if (!$errors) {
            $i = 0;
            $comments = $_REQUEST['comments'];
            foreach ($_REQUEST['tids'] as $tid) {
Peter Rotich's avatar
Peter Rotich committed
                if (($ticket=Ticket::lookup($tid))
                        && $ticket->getStatusId() != $status->getId()
                        && $ticket->checkStaffPerm($thisstaff)
                        && $ticket->setStatus($status, $comments, $errors))
            if (!$i) {
                $errors['err'] = $errors['err']
                    ?: sprintf(__('Unable to change status for %s'),
                        _N('the selected ticket', 'any of the selected tickets', $count));
Peter Rotich's avatar
Peter Rotich committed
            else {
                // Assume success
                if ($i==$count) {
Peter Rotich's avatar
Peter Rotich committed

                    if (!strcasecmp($status->getState(), 'deleted')) {
                        $msg = sprintf(__( 'Successfully deleted %s'),
Peter Rotich's avatar
Peter Rotich committed
                                _N('selected ticket', 'selected tickets',
                                    $count));
                    } else {
                       $msg = sprintf(
                            __(
                                /* 1$ will be 'selected ticket(s)', 2$ is the new status */
Peter Rotich's avatar
Peter Rotich committed
                                'Successfully changed status of %1$s to %2$s'),
Peter Rotich's avatar
Peter Rotich committed
                            _N('selected ticket', 'selected tickets',
                                $count),
                            $status->getName());
Peter Rotich's avatar
Peter Rotich committed
                    }

                    $_SESSION['::sysmsgs']['msg'] = $msg;
Peter Rotich's avatar
Peter Rotich committed
                } else {
Peter Rotich's avatar
Peter Rotich committed

                    if (!strcasecmp($status->getState(), 'deleted')) {
                        $warn = sprintf(__('Successfully deleted %s'),
Peter Rotich's avatar
Peter Rotich committed
                                sprintf(__('%1$d of %2$d selected tickets'),
                                    $i, $count)
                                );
                    } else {

                        $warn = sprintf(
                                __('%1$d of %2$d %3$s status changed to %4$s'),$i, $count,
                                _N('selected ticket', 'selected tickets',
                                    $count),
                                $status->getName());
                    }

                    $_SESSION['::sysmsgs']['warn'] = $warn;
Peter Rotich's avatar
Peter Rotich committed
                }

                Http::response(201, 'Successfully processed');
            }
        }

Peter Rotich's avatar
Peter Rotich committed
        return self::_changeSelectedTicketsStatus($state, $info, $errors);
    }

    function triggerThreadAction($ticket_id, $thread_id, $action) {
        $thread = ThreadEntry::lookup($thread_id);
        if (!$thread)
            Http::response(404, 'No such ticket thread entry');
        if ($thread->getThread()->getObjectId() != $ticket_id)
            Http::response(404, 'No such ticket thread entry');

        $valid = false;
        foreach ($thread->getActions() as $group=>$list) {
            foreach ($list as $name=>$A) {
                if ($A->getId() == $action) {
                    $valid = true; break;
                }
            }
        }
        if (!$valid)
            Http::response(400, 'Not a valid action for this thread');

        $thread->triggerAction($action);
    }

Peter Rotich's avatar
Peter Rotich committed
    private function _changeSelectedTicketsStatus($state, $info=array(), $errors=array()) {

        $count = $_REQUEST['count'] ?:
            ($_REQUEST['tids'] ?  count($_REQUEST['tids']) : 0);

Peter Rotich's avatar
Peter Rotich committed
        $info['title'] = sprintf(__('Change Status — %1$d %2$s selected'),
                 $count,
                 _N('ticket', 'tickets', $count)
                 );
Peter Rotich's avatar
Peter Rotich committed
        if (!strcasecmp($state, 'deleted')) {
Peter Rotich's avatar
Peter Rotich committed
            $info['warn'] = sprintf(__(
                        'Are you sure you want to DELETE %s?'),
                    _N('selected ticket', 'selected tickets', $count)
                    );
Peter Rotich's avatar
Peter Rotich committed
            $info['extra'] = sprintf('<strong>%s</strong>', __(
                        'Deleted tickets CANNOT be recovered, including any associated attachments.')
                    );

            $info['placeholder'] = sprintf(__(
                        'Optional reason for deleting %s'),
                    _N('selected ticket', 'selected tickets', $count));
        }

        $info['status_id'] = $info['status_id'] ?: $_REQUEST['status_id'];
        $info['comments'] = Format::htmlchars($_REQUEST['comments']);
Peter Rotich's avatar
Peter Rotich committed

        return self::_changeStatus($state, $info, $errors);
Peter Rotich's avatar
Peter Rotich committed
    private function _changeTicketStatus($ticket, $state, $info=array(), $errors=array()) {

        $verb = TicketStateField::getVerb($state);

        $info['action'] = sprintf('#tickets/%d/status', $ticket->getId());
        $info['title'] = sprintf(__(
                    /* 1$ will be a verb, like 'open', 2$ will be the ticket number */
                    '%1$s Ticket #%2$s'),
                $verb ?: $state,
                $ticket->getNumber()
                );

        // Deleting?
        if (!strcasecmp($state, 'deleted')) {

            $info['placeholder'] = sprintf(__(
                        'Optional reason for deleting %s'),
                    __('this ticket'));
            $info[ 'warn'] = sprintf(__(
                        'Are you sure you want to DELETE %s?'),
                        __('this ticket'));
            //TODO: remove message below once we ship data retention plug
            $info[ 'extra'] = sprintf('<strong>%s</strong>',
                        __('Deleted tickets CANNOT be recovered, including any associated attachments.')
                        );
        }

        $info['status_id'] = $info['status_id'] ?: $ticket->getStatusId();
        $info['comments'] = Format::htmlchars($_REQUEST['comments']);

        return self::_changeStatus($state, $info, $errors);
    }

    private function _changeStatus($state, $info=array(), $errors=array()) {
Peter Rotich's avatar
Peter Rotich committed

        if ($info && isset($info['errors']))
Peter Rotich's avatar
Peter Rotich committed
            $errors = array_merge($errors, $info['errors']);
Peter Rotich's avatar
Peter Rotich committed

        if (!$info['error'] && isset($errors['err']))
            $info['error'] = $errors['err'];

        include(STAFFINC_DIR . 'templates/ticket-status.tmpl.php');
    }

    function tasks($tid) {
        global $thisstaff;

        if (!($ticket=Ticket::lookup($tid))
                || !$ticket->checkStaffPerm($thisstaff))
            Http::response(404, 'Unknown ticket');

         include STAFFINC_DIR . 'ticket-tasks.inc.php';
    }

    function addTask($tid) {
        global $thisstaff;

        if (!($ticket=Ticket::lookup($tid)))
            Http::response(404, 'Unknown ticket');

        if (!$ticket->checkStaffPerm($thisstaff, Task::PERM_CREATE))
            Http::response(403, 'Permission denied');

        $info=$errors=array();

        if ($_POST) {
            Draft::deleteForNamespace(
                    sprintf('ticket.%d.task', $ticket->getId()),
                    $thisstaff->getId());
            // Default form
Peter Rotich's avatar
Peter Rotich committed
            $form = TaskForm::getInstance();
            $form->setSource($_POST);
            // Internal form
            $iform = TaskForm::getInternalForm($_POST);
            $isvalid = true;
            if (!$iform->isValid())
                $isvalid = false;
            if (!$form->isValid())
                $isvalid = false;

            if ($isvalid) {
                $vars = $_POST;
                $vars['object_id'] = $ticket->getId();
                $vars['object_type'] = ObjectModel::OBJECT_TYPE_TICKET;
                $vars['default_formdata'] = $form->getClean();
                $vars['internal_formdata'] = $iform->getClean();
                $desc = $form->getField('description');
                if ($desc
                        && $desc->isAttachmentsEnabled()
                        && ($attachments=$desc->getWidget()->getAttachments()))
                    $vars['cannedattachments'] = $attachments->getClean();
                $vars['staffId'] = $thisstaff->getId();
                $vars['poster'] = $thisstaff;
                $vars['ip_address'] = $_SERVER['REMOTE_ADDR'];
                if (($task=Task::create($vars, $errors)))
                    Http::response(201, $task->getId());
            }

            $info['error'] = __('Error adding task - try again!');
        }

        $info['action'] = sprintf('#tickets/%d/add-task', $ticket->getId());
        $info['title'] = sprintf(
                __( 'Ticket #%1$s: %2$s'),
                $ticket->getNumber(),
                __('Add New Task')
                );

         include STAFFINC_DIR . 'templates/task.tmpl.php';
    }
Peter Rotich's avatar
Peter Rotich committed

    function task($tid, $id) {
        global $thisstaff;

        if (!($ticket=Ticket::lookup($tid))
                || !$ticket->checkStaffPerm($thisstaff))
            Http::response(404, 'Unknown ticket');

        // Lookup task and check access
        if (!($task=Task::lookup($id))
                || !$task->checkStaffPerm($thisstaff))
            Http::response(404, 'Unknown task');

Peter Rotich's avatar
Peter Rotich committed
        $info = $errors = array();
        $note_attachments_form = new SimpleForm(array(
Peter Rotich's avatar
Peter Rotich committed
            'attachments' => new FileUploadField(array('id'=>'attach',
Peter Rotich's avatar
Peter Rotich committed
                'name'=>'attach:note',
                'configuration' => array('extensions'=>'')))
        ));

        $reply_attachments_form = new SimpleForm(array(
            'attachments' => new FileUploadField(array('id'=>'attach',
                'name'=>'attach:reply',
                'configuration' => array('extensions'=>'')))
        ));
Peter Rotich's avatar
Peter Rotich committed

        if ($_POST) {
Peter Rotich's avatar
Peter Rotich committed
            $vars = $_POST;
Peter Rotich's avatar
Peter Rotich committed
            switch ($_POST['a']) {
            case 'postnote':
Peter Rotich's avatar
Peter Rotich committed
                $attachments = $note_attachments_form->getField('attachments')->getClean();
Peter Rotich's avatar
Peter Rotich committed
                $vars['cannedattachments'] = array_merge(
                    $vars['cannedattachments'] ?: array(), $attachments);
Peter Rotich's avatar
Peter Rotich committed
                if (($note=$task->postNote($vars, $errors, $thisstaff))) {
Peter Rotich's avatar
Peter Rotich committed
                    $msg=__('Note posted successfully');
                    // Clear attachment list
Peter Rotich's avatar
Peter Rotich committed
                    $note_attachments_form->setSource(array());
                    $note_attachments_form->getField('attachments')->reset();
Peter Rotich's avatar
Peter Rotich committed
                    Draft::deleteForNamespace('task.note.'.$task->getId(),
                            $thisstaff->getId());
                } else {
Peter Rotich's avatar
Peter Rotich committed
                    if (!$errors['err'])
Peter Rotich's avatar
Peter Rotich committed
                        $errors['err'] = __('Unable to post the note - missing or invalid data.');
                }
                break;
Peter Rotich's avatar
Peter Rotich committed
            case 'postreply':
                $attachments = $reply_attachments_form->getField('attachments')->getClean();
                $vars['cannedattachments'] = array_merge(
                    $vars['cannedattachments'] ?: array(), $attachments);
                if (($response=$task->postReply($vars, $errors))) {
                    $msg=__('Update posted successfully');
                    // Clear attachment list
                    $reply_attachments_form->setSource(array());
                    $reply_attachments_form->getField('attachments')->reset();
                    Draft::deleteForNamespace('task.reply.'.$task->getId(),
                            $thisstaff->getId());
                } else {
                    if (!$errors['err'])
                        $errors['err'] = __('Unable to post the reply - missing or invalid data.');
                }
                break;
Peter Rotich's avatar
Peter Rotich committed
            default:
                $errors['err'] = __('Unknown action');
            }
        }

        include STAFFINC_DIR . 'templates/task-view.tmpl.php';
    }