'notificationevent',
        'comment'               => 'commentevent',
        'comment_deleted'       => 'commentevent',
        'ack'                   => 'commentevent',
        'ack_deleted'           => 'commentevent',
        'dt_comment'            => 'commentevent',
        'dt_comment_deleted'    => 'commentevent',
        'flapping'              => 'flappingevent',
        'flapping_deleted'      => 'flappingevent',
        'hard_state'            => 'statechangeevent',
        'soft_state'            => 'statechangeevent',
        'dt_start'              => 'downtimeevent',
        'dt_end'                => 'downtimeevent'
    );
    /**
     * Cache for {@link time()}
     *
     * @var DateTimeZone
     */
    protected $timeZone;
    public function showAction()
    {
        $type = $this->params->shiftRequired('type');
        $id = $this->params->shiftRequired('id');
        if (! isset($this->dataViewsByType[$type])
            || $this->applyRestriction(
                'monitoring/filter/objects',
                $this->backend->select()->from('eventhistory', array('id'))->where('id', $id)
            )->fetchRow() === false
        ) {
            $this->httpNotFound($this->translate('Event not found'));
        }
        $event = $this->query($type, $id)->fetchRow();
        if ($event === false) {
            $this->httpNotFound($this->translate('Event not found'));
        }
        $this->view->object = $object = $event->service_description === null
            ? new Host($this->backend, $event->host_name)
            : new Service($this->backend, $event->host_name, $event->service_description);
        $object->fetch();
        list($icon, $label) = $this->getIconAndLabel($type);
        $this->view->details = array_merge(
            array(array($this->view->escape($this->translate('Type')), $label)),
            $this->getDetails($type, $event)
        );
        $this->getTabs()
            ->add('event', array(
                'title'     => $label,
                'label'     => $label,
                'icon'      => $icon,
                'url'       => Url::fromRequest(),
                'active'    => true
            ))
            ->extend(new OutputFormat())
            ->extend(new DashboardAction())
            ->extend(new MenuAction());
    }
    /**
     * Return translated and escaped 'Yes' if the given condition is true, 'No' otherwise, 'N/A' if NULL
     *
     * @param   bool|null   $condition
     *
     * @return  string
     */
    protected function yesOrNo($condition)
    {
        if ($condition === null) {
            return $this->view->escape($this->translate('N/A'));
        }
        return $this->view->escape($condition ? $this->translate('Yes') : $this->translate('No'));
    }
    /**
     * Render the given timestamp as human readable HTML in the user agent's timezone or 'N/A' if NULL
     *
     * @param   int|null    $stamp
     *
     * @return  string
     */
    protected function time($stamp)
    {
        if ($stamp === null) {
            return $this->view->escape($this->translate('N/A'));
        }
        if ($this->timeZone === null) {
            $timezoneDetect = new TimezoneDetect();
            $this->timeZone = new DateTimeZone(
                $timezoneDetect->success() ? $timezoneDetect->getTimezoneName() : date_default_timezone_get()
            );
        }
        return $this->view->escape(
            DateTime::createFromFormat('U', $stamp)->setTimezone($this->timeZone)->format('Y-m-d H:i:s')
        );
    }
    /**
     * Render the given duration in seconds as human readable HTML or 'N/A' if NULL
     *
     * @param   int|null    $seconds
     *
     * @return  string
     */
    protected function duration($seconds)
    {
        return $this->view->escape(
            $seconds === null ? $this->translate('N/A') : DateFormatter::formatDuration($seconds)
        );
    }
    /**
     * Render the given percent number as human readable HTML or 'N/A' if NULL
     *
     * @param   float|null  $percent
     *
     * @return  string
     */
    protected function percent($percent)
    {
        return $this->view->escape(
            $percent === null ? $this->translate('N/A') : sprintf($this->translate('%.2f%%'), $percent)
        );
    }
    /**
     * Render the given comment message as HTML or 'N/A' if NULL
     *
     * @param   string|null $message
     *
     * @return  string
     */
    protected function comment($message)
    {
        return $this->view->nl2br($this->view->createTicketLinks($this->view->escapeComment($message)));
    }
    /**
     * Render a link to the given contact or 'N/A' if NULL
     *
     * @param   string|null $name
     *
     * @return  string
     */
    protected function contact($name)
    {
        return $name === null
            ? $this->view->escape($this->translate('N/A'))
            : $this->view->qlink($name, Url::fromPath('monitoring/show/contact', array('contact_name' => $name)));
    }
    /**
     * Render the given monitored object state as human readable HTML or 'N/A' if NULL
     *
     * @param   bool        $isService
     * @param   int|null    $state
     *
     * @return  string
     */
    protected function state($isService, $state)
    {
        if ($state === null) {
            return $this->view->escape($this->translate('N/A'));
        }
        try {
            $stateText = $isService
                ? Service::getStateText($state, true)
                : Host::getStateText($state, true);
        } catch (InvalidArgumentException $e) {
            return $this->view->escape($this->translate('N/A'));
        }
        return ' ' . $this->view->escape($stateText) . '';
    }
    /**
     * Render the given plugin output as human readable HTML
     *
     * @param   string  $output
     *
     * @return  string
     */
    protected function pluginOutput($output)
    {
        return $this->view->getHelper('PluginOutput')->pluginOutput($output);
    }
    /**
     * Return the icon and the label for the given event type
     *
     * @param   string  $eventType
     *
     * @return  string[]
     */
    protected function getIconAndLabel($eventType)
    {
        switch ($eventType) {
            case 'notify':
                return array('bell', $this->translate('Notification', 'tooltip'));
            case 'comment':
                return array('comment-empty', $this->translate('Comment', 'tooltip'));
            case 'comment_deleted':
                return array('cancel', $this->translate('Comment removed', 'tooltip'));
            case 'ack':
                return array('ok', $this->translate('Acknowledged', 'tooltip'));
            case 'ack_deleted':
                return array('ok', $this->translate('Acknowledgement removed', 'tooltip'));
            case 'dt_comment':
                return array('plug', $this->translate('Downtime scheduled', 'tooltip'));
            case 'dt_comment_deleted':
                return array('plug', $this->translate('Downtime removed', 'tooltip'));
            case 'flapping':
                return array('flapping', $this->translate('Flapping started', 'tooltip'));
            case 'flapping_deleted':
                return array('flapping', $this->translate('Flapping stopped', 'tooltip'));
            case 'hard_state':
                return array('warning-empty', $this->translate('Hard state change'));
            case 'soft_state':
                return array('spinner', $this->translate('Soft state change'));
            case 'dt_start':
                return array('plug', $this->translate('Downtime started', 'tooltip'));
            case 'dt_end':
                return array('plug', $this->translate('Downtime ended', 'tooltip'));
        }
    }
    /**
     * Return a query for the given event ID of the given type
     *
     * @param   string  $type
     * @param   int     $id
     *
     * @return  Queryable
     */
    protected function query($type, $id)
    {
        switch ($this->dataViewsByType[$type]) {
            case 'downtimeevent':
                return $this->backend->select()
                    ->from('downtimeevent', array(
                        'entry_time'            => 'downtimeevent_entry_time',
                        'author_name'           => 'downtimeevent_author_name',
                        'comment_data'          => 'downtimeevent_comment_data',
                        'is_fixed'              => 'downtimeevent_is_fixed',
                        'scheduled_start_time'  => 'downtimeevent_scheduled_start_time',
                        'scheduled_end_time'    => 'downtimeevent_scheduled_end_time',
                        'was_started'           => 'downtimeevent_was_started',
                        'actual_start_time'     => 'downtimeevent_actual_start_time',
                        'actual_end_time'       => 'downtimeevent_actual_end_time',
                        'was_cancelled'         => 'downtimeevent_was_cancelled',
                        'is_in_effect'          => 'downtimeevent_is_in_effect',
                        'trigger_time'          => 'downtimeevent_trigger_time',
                        'host_name',
                        'service_description'
                    ))
                    ->where('downtimeevent_id', $id);
            case 'commentevent':
                return $this->backend->select()
                    ->from('commentevent', array(
                        'entry_type'            => 'commentevent_entry_type',
                        'comment_time'          => 'commentevent_comment_time',
                        'author_name'           => 'commentevent_author_name',
                        'comment_data'          => 'commentevent_comment_data',
                        'is_persistent'         => 'commentevent_is_persistent',
                        'comment_source'        => 'commentevent_comment_source',
                        'expires'               => 'commentevent_expires',
                        'expiration_time'       => 'commentevent_expiration_time',
                        'deletion_time'         => 'commentevent_deletion_time',
                        'host_name',
                        'service_description'
                    ))
                    ->where('commentevent_id', $id);
            case 'flappingevent':
                return $this->backend->select()
                    ->from('flappingevent', array(
                        'event_time'            => 'flappingevent_event_time',
                        'reason_type'           => 'flappingevent_reason_type',
                        'percent_state_change'  => 'flappingevent_percent_state_change',
                        'low_threshold'         => 'flappingevent_low_threshold',
                        'high_threshold'        => 'flappingevent_high_threshold',
                        'host_name',
                        'service_description'
                    ))
                    ->where('flappingevent_id', $id)
                    ->where('flappingevent_event_type', $type);
            case 'notificationevent':
                return $this->backend->select()
                    ->from('notificationevent', array(
                        'notification_reason'   => 'notificationevent_reason',
                        'start_time'            => 'notificationevent_start_time',
                        'end_time'              => 'notificationevent_end_time',
                        'state'                 => 'notificationevent_state',
                        'output'                => 'notificationevent_output',
                        'long_output'           => 'notificationevent_long_output',
                        'escalated'             => 'notificationevent_escalated',
                        'contacts_notified'     => 'notificationevent_contacts_notified',
                        'host_name',
                        'service_description'
                    ))
                    ->where('notificationevent_id', $id);
            case 'statechangeevent':
                return $this->backend->select()
                    ->from('statechangeevent', array(
                        'state_time'            => 'statechangeevent_state_time',
                        'state'                 => 'statechangeevent_state',
                        'current_check_attempt' => 'statechangeevent_current_check_attempt',
                        'max_check_attempts'    => 'statechangeevent_max_check_attempts',
                        'last_state'            => 'statechangeevent_last_state',
                        'last_hard_state'       => 'statechangeevent_last_hard_state',
                        'output'                => 'statechangeevent_output',
                        'long_output'           => 'statechangeevent_long_output',
                        'host_name',
                        'service_description'
                    ))
                    ->where('statechangeevent_id', $id)
                    ->where('statechangeevent_state_change', 1)
                    ->where('statechangeevent_state_type', $type);
        }
    }
    /**
     * Return the given event's data prepared for a name-value table
     *
     * @param   string      $type
     * @param   \stdClass   $event
     *
     * @return  string[][]
     */
    protected function getDetails($type, $event)
    {
        switch ($type) {
            case 'dt_start':
            case 'dt_end':
                $details = array(array(
                    array($this->translate('Entry time'), $this->time($event->entry_time)),
                    array($this->translate('Is fixed'), $this->yesOrNo($event->is_fixed)),
                    array($this->translate('Is in effect'), $this->yesOrNo($event->is_in_effect)),
                    array($this->translate('Was started'), $this->yesOrNo($event->was_started))
                ));
                if ($type === 'dt_end') {
                    $details[] = array(
                        array($this->translate('Was cancelled'), $this->yesOrNo($event->was_cancelled))
                    );
                }
                $details[] = array(
                    array($this->translate('Trigger time'), $this->time($event->trigger_time)),
                    array($this->translate('Scheduled start time'), $this->time($event->scheduled_start_time)),
                    array($this->translate('Actual start time'), $this->time($event->actual_start_time)),
                    array($this->translate('Scheduled end time'), $this->time($event->scheduled_end_time))
                );
                if ($type === 'dt_end') {
                    $details[] = array(
                        array($this->translate('Actual end time'), $this->time($event->actual_end_time)))
                    ;
                }
                $details[] = array(
                    array($this->translate('Author'), $this->contact($event->author_name)),
                    array($this->translate('Comment'), $this->comment($event->comment_data))
                );
                return call_user_func_array('array_merge', $details);
            case 'comment':
            case 'comment_deleted':
            case 'ack':
            case 'ack_deleted':
            case 'dt_comment':
            case 'dt_comment_deleted':
                switch ($event->entry_type) {
                    case 'comment':
                        $entryType = $this->translate('User comment');
                        break;
                    case 'downtime':
                        $entryType = $this->translate('Scheduled downtime');
                        break;
                    case 'flapping':
                        $entryType = $this->translate('Flapping');
                        break;
                    case 'ack':
                        $entryType = $this->translate('Acknowledgement');
                        break;
                    default:
                        $entryType = $this->translate('N/A');
                }
                switch ($event->comment_source) {
                    case 'icinga':
                        $commentSource = $this->translate('Icinga');
                        break;
                    case 'user':
                        $commentSource = $this->translate('User');
                        break;
                    default:
                        $commentSource = $this->translate('N/A');
                }
                return array(
                    array($this->translate('Time'), $this->time($event->comment_time)),
                    array($this->translate('Source'), $this->view->escape($commentSource)),
                    array($this->translate('Entry type'), $this->view->escape($entryType)),
                    array($this->translate('Author'), $this->contact($event->author_name)),
                    array($this->translate('Is persistent'), $this->yesOrNo($event->is_persistent)),
                    array($this->translate('Expires'), $this->yesOrNo($event->expires)),
                    array($this->translate('Expiration time'), $this->time($event->expiration_time)),
                    array($this->translate('Deletion time'), $this->time($event->deletion_time)),
                    array($this->translate('Message'), $this->comment($event->comment_data))
                );
            case 'flapping':
            case 'flapping_deleted':
                switch ($event->reason_type) {
                    case 'stopped':
                        $reasonType = $this->translate('Flapping stopped normally');
                        break;
                    case 'disabled':
                        $reasonType = $this->translate('Flapping was disabled');
                        break;
                    default:
                        $reasonType = $this->translate('N/A');
                }
                return array(
                    array($this->translate('Event time'), $this->time($event->event_time)),
                    array($this->translate('Reason'), $this->view->escape($reasonType)),
                    array($this->translate('State change'), $this->percent($event->percent_state_change)),
                    array($this->translate('Low threshold'), $this->percent($event->low_threshold)),
                    array($this->translate('High threshold'), $this->percent($event->high_threshold))
                );
            case 'notify':
                switch ($event->notification_reason) {
                    case 'normal_notification':
                        $notificationReason = $this->translate('Normal notification');
                        break;
                    case 'ack':
                        $notificationReason = $this->translate('Problem acknowledgement');
                        break;
                    case 'flapping_started':
                        $notificationReason = $this->translate('Flapping started');
                        break;
                    case 'flapping_stopped':
                        $notificationReason = $this->translate('Flapping stopped');
                        break;
                    case 'flapping_disabled':
                        $notificationReason = $this->translate('Flapping was disabled');
                        break;
                    case 'dt_start':
                        $notificationReason = $this->translate('Downtime started');
                        break;
                    case 'dt_end':
                        $notificationReason = $this->translate('Downtime ended');
                        break;
                    case 'dt_cancel':
                        $notificationReason = $this->translate('Downtime was cancelled');
                        break;
                    case 'custom_notification':
                        $notificationReason = $this->translate('Custom notification');
                        break;
                    default:
                        $notificationReason = $this->translate('N/A');
                }
                $details = array(
                    array($this->translate('Start time'), $this->time($event->start_time)),
                    array($this->translate('End time'), $this->time($event->end_time)),
                    array($this->translate('Reason'), $this->view->escape($notificationReason)),
                    array(
                        $this->translate('State'),
                        $this->state($event->service_description !== null, $event->state)
                    ),
                    array($this->translate('Escalated'), $this->yesOrNo($event->escalated)),
                    array($this->translate('Contacts notified'), (int) $event->contacts_notified),
                    array(
                        $this->translate('Output'),
                        $this->pluginOutput($event->output) .  $this->pluginOutput($event->long_output)
                    )
                );
                return $details;
            case 'hard_state':
            case 'soft_state':
                $isService = $event->service_description !== null;
                $details = array(
                    array($this->translate('State time'), $this->time($event->state_time)),
                    array($this->translate('State'), $this->state($isService, $event->state)),
                    array($this->translate('Check attempt'), $this->view->escape(sprintf(
                        $this->translate('%d of %d'),
                        (int) $event->current_check_attempt,
                        (int) $event->max_check_attempts
                    ))),
                    array($this->translate('Last state'), $this->state($isService, $event->last_state)),
                    array($this->translate('Last hard state'), $this->state($isService, $event->last_hard_state)),
                    array(
                        $this->translate('Output'),
                        $this->pluginOutput($event->output) .  $this->pluginOutput($event->long_output)
                    )
                );
                return $details;
        }
    }
}