<?xml version="1.0" encoding="utf-8" ?>
<otrs_package version="1.1">
    <Name>HotfixMention</Name>
    <Version>6.4.3</Version>
    <Vendor>maxence business consulting gmbh</Vendor>
    <URL>http://www.maxence.de/</URL>
    <License>GNU AFFERO GENERAL PUBLIC LICENSE Version 3, November 2007</License>
    <ChangeLog Date="2022-09-28 10:00:00" Version="6.4.3">New package.</ChangeLog>
    <Description Lang="en">Fixes the removal of the mention in the overview.</Description>
    <Description Lang="de">Fixed das entfernen von Mentions aus der Übersicht.</Description>
    <Framework>6.4.3</Framework>
    <OS>linux</OS>
    <IntroInstall Lang="en" Title="Install Information" Type="pre">
        &lt;br/ &gt;
        &lt;br/ &gt;
        You are about to install the maxence HotfixMention package.
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;Strong&gt;Required OTRS packages&lt;/Strong&gt;
        &lt;br/ &gt;
        &lt;ul&gt;
                &lt;li&gt;none&lt;/li&gt;
        &lt;/ul&gt;
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;p&gt;
        Your maxence team!
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;a href=&quot;http://www.maxence.de/otrs/&quot; target=&quot;_blank&quot;&gt;
        &lt;img src=&quot;data:image/png;base64,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&quot;&gt;&lt;/img&gt;
        &lt;/a&gt;
        &lt;br/ &gt;
        &lt;a href=&quot;http://www.maxence.de/otrs/&quot; target=&quot;_blank&quot;&gt;http://www.maxence.de/otrs/&lt;/a&gt;
        &lt;/p&gt;
        &lt;br/ &gt;
    </IntroInstall>
    <IntroInstall Lang="de" Title="Installation Information" Type="pre">
        &lt;br/ &gt;
        &lt;br/ &gt;
        Sie sind im Begriff das maxence HotfixMention Paket zu installieren.
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;Strong&gt;Benötigte OTRS-Pakete&lt;/Strong&gt;
        &lt;br/ &gt;
        &lt;ul&gt;
                &lt;li&gt;keine Abhängigkeiten&lt;/li&gt;
        &lt;/ul&gt;
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;p&gt;
        Ihr maxence Team!
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;br/ &gt;
        &lt;a href=&quot;http://www.maxence.de/otrs/&quot; target=&quot;_blank&quot;&gt;
        &lt;img src=&quot;data:image/png;base64,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&quot;&gt;&lt;/img&gt;
        &lt;/a&gt;
        &lt;br/ &gt;
        &lt;a href=&quot;http://www.maxence.de/otrs/&quot; target=&quot;_blank&quot;&gt;http://www.maxence.de/otrs/&lt;/a&gt;
        &lt;/p&gt;
        &lt;br/ &gt;
    </IntroInstall>
    <BuildDate>2022-09-28 08:13:36</BuildDate>
    <BuildHost>maxence.de</BuildHost>
    <Filelist>
        <File Location="Custom/Kernel/Output/HTML/Templates/Standard/AgentTicketOverviewNavBar.tt" Permission="644" Encode="Base64"># --
# Copyright (C) 2001-2021 OTRS AG, https://otrs.com/
# Copyright (C) 2021-2022 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (GPL). If you
# did not receive this file, see https://www.gnu.org/licenses/gpl-3.0.txt.
# --

[% RenderBlockStart("OverviewNavBar") %]
<div class="OverviewBox ARIARoleMain [% Data.View | html %]">
    <h1>[% Translate(Data.TitleName) | html %]: [% IF Env("Action") == 'AgentTicketQueue' %][% Data.TitleValue | html %][% ELSE %][% Translate(Data.TitleValue) | html %][% END %]</h1>

    [% Data.DataInTheMiddle %]

    <div class="OverviewControl" id="OverviewControl">
        <div>
            <div class="ControlRow">
[% RenderBlockStart("OverviewNavBarFilter") %]
                <ul class="Tabs">
[% RenderBlockStart("OverviewNavBarFilterItem") %]
[% RenderBlockStart("OverviewNavBarFilterItemSelected") %]
                    <li class="Active [% Data.CSS | html %]">
                        <a name="OverviewControl" href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkFilter %]Filter=[% Data.Filter | uri %]">[% Translate(Data.Name) | html %]<span>[% Data.Count | html %]</span></a>
                    </li>
[% RenderBlockEnd("OverviewNavBarFilterItemSelected") %]
[% RenderBlockStart("OverviewNavBarFilterItemSelectedNot") %]
                    <li class="[% Data.CSS | html %]">
                        <a name="OverviewControl" href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkFilter %]Filter=[% Data.Filter | uri %]">[% Translate(Data.Name) | html %]<span>[% Data.Count | html %]</span></a>
                    </li>
[% RenderBlockEnd("OverviewNavBarFilterItemSelectedNot") %]
[% RenderBlockStart("OverviewNavBarFilterItemSplit") %]
                    -
[% RenderBlockEnd("OverviewNavBarFilterItemSplit") %]
[% RenderBlockEnd("OverviewNavBarFilterItem") %]
                </ul>
[% RenderBlockEnd("OverviewNavBarFilter") %]
[% RenderBlockStart("OverviewNavBarPageBack") %]
                <ul class="Tabs">
                    <li>
                        <a name="OverviewControl" id="TicketSearch" href="[% Env("Baselink") %]Action=[% Env("Action") %];[% Data.LinkBack %]">[% Translate("Change search options") | html %] [% Data.ProfileName | html %]</a>
                    </li>
                </ul>
[% RenderBlockEnd("OverviewNavBarPageBack") %]

[% RenderBlockStart("ContextSettings") %]

                <ul class="ContextFunctions">

[% RenderBlockStart("DocumentActionRowRemoveColumnFilters") %]
                    <li class="[% Data.CSS | html %]">
                        <a href="[% Env("Baselink") %]Action=[% Env("Action") %];Filter=[% Data.Filter %];View=[% Data.View %];[% Data.LinkFilter %];DeleteFilters=DeleteFilters" title="[% Translate("Remove active filters for this screen.") | html %]">
                            <i class="fa fa-trash-o"></i>
                        </a>
                    </li>
[% RenderBlockEnd("DocumentActionRowRemoveColumnFilters") %]
[% IF Env('Action') == 'AgentTicketMentionView' %]
                    <li class="RemoveMention ContextSettings">
                        <a href="#" id="RemoveMention" data-user-id="[% Env('UserID') | html %]" title="[% Translate("Remove mention") | html %]">
                            <i class="fa fa-eye-slash"></i>
                            <span>[% Translate("Remove mention") | html %]</span>
                        </a>
                    </li>
[% END %]
                    <li class="ContextSettings">
                        <a href="#" id="ShowContextSettingsDialog" title="[% Translate("Settings") | html %]">
                            <i class="fa fa-cog"></i>
                            <span>[% Translate("Settings") | html %]</span>
                        </a>
                    </li>
                </ul>

                <div id="ContextSettingsDialogContainer" class="Hidden">
                    <form action="[% Env("CGIHandle") %]" method="post" id="ContextSettingsDialogOverview[% Data.View | html %]" class="WidgetSettingsForm">
                        <input type="hidden" name="Action" value="AgentPreferences"/>
                        <input type="hidden" name="Subaction" value="Update"/>
                        <input type="hidden" name="Group" value="[% Data.Group | html %]"/>
                        <input type="hidden" name="RedirectURL" value="[% Data.RequestedURL | html %]"/>
                        <fieldset class="TableLike FixedLabelSmall">
                            <label for="[% Data.PreferencesKey | html %]">[% Translate("Tickets per page") | html %]:</label>
                            <div class="Field">
                                [% Data.PageShownString %]
                            </div>
                            <div class="Clear"></div>

[% RenderBlockStart("FilterColumnSettings") %]
                            <input type="hidden" class="ColumnsEnabledJSON" value="[% Data.ColumnsEnabled | html %]" />
                            <input type="hidden" class="ColumnsAvailableJSON" value="[% Data.ColumnsAvailable | html %]" />
                            <input type="hidden" name="Group" value="[% Data.GroupName | html %]"/>
                            <input type="hidden" name="FilterAction" value="[% Data.Name | html %]"/>
                            <label class="DontPrint" for="[% Data.NamePref | html %]">[% Translate(Data.Desc) | html %]:</label>
                            <div class="Field DontPrint LayoutGrid AllocationListContainer">
                                <div class="Size1of2">
                                    <ul class="Tablelike">
                                        <li class="Header">[% Translate("Available Columns") | html %]</li>
                                    </ul>
                                    <div class="FilterInputContainer">
                                        <input type="text" class="FilterAvailableFields" name="FilterAvailableFields" value="" title="[% Translate("Filter available fields") | html %]" placeholder="[% Translate("Filter available fields") | html %]..." />
                                    </div>
                                    <ul class="AllocationList Tablelike AvailableFields SettingsOverview" id="AvailableField-Dashboard[% Data.Name | html %]"></ul>
                                </div>
                                <div class="Size1of2">
                                    <ul class="Tablelike">
                                        <li class="Header">[% Translate("Visible Columns (order by drag & drop)") | html %]</li>
                                    </ul>
                                    <ul class="AllocationList Tablelike AssignedFields OrderNumbers SettingsOverview" id="AssignedFields-Dashboard[% Data.Name | html %]"></ul>
                                </div>
                            </div>
                            <div class="Clear"></div>

[% RenderBlockEnd("FilterColumnSettings") %]

                        </fieldset>
                    </form>
                </div>

[% RenderBlockEnd("ContextSettings") %]

            </div>
            <div class="ActionRow">
                <div class="OverviewActions">
                    [% Data.ActionRow %]
                    <ul class="OverviewZoom">
[% RenderBlockStart("OverviewNavBarViewMode") %]
[% RenderBlockStart("OverviewNavBarViewModeSelected") %]
                        <li class="Active">
                            <a class="[% Data.Name | html %]" name="OverviewControl" href="[% Env("Baselink") %]Action=[% Env("Action") %];Filter=[% Data.Filter | uri %];View=[% Data.View | uri %];[% Data.LinkFilter %]" title="[% Translate(Data.Name) | html %]">
                                [% Translate(Data.NameShort) | html %]
                            </a>
                        </li>
[% RenderBlockEnd("OverviewNavBarViewModeSelected") %]
[% RenderBlockStart("OverviewNavBarViewModeNotSelected") %]
                        <li>
                            <a class="[% Data.Name | html %]" name="OverviewControl" href="[% Env("Baselink") %]Action=[% Env("Action") %];Filter=[% Data.Filter | uri %];View=[% Data.View | uri %];[% Data.LinkFilter %]" title="[% Translate(Data.Name) | html %]">
                                [% Translate(Data.NameShort) | html %]
                            </a>
                        </li>
[% RenderBlockEnd("OverviewNavBarViewModeNotSelected") %]
[% RenderBlockEnd("OverviewNavBarViewMode") %]
                    </ul>
                    <span class="Pagination">
[% RenderBlockStart("OverviewNavBarPageNavBar") %]
                        [% Data.ResultLong %]
                        [% Data.SiteNavBarLong %]
[% RenderBlockEnd("OverviewNavBarPageNavBar") %]
                    </span>
                    <div class="ClearLeft"></div>
                </div>
# sort order bar if needed
                [% Data.SortOrderBar %]
            </div>
        </div>
    </div>
</div>

[% RenderBlockEnd("OverviewNavBar") %]</File>
        <File Location="Custom/Kernel/System/Mention.pm" Permission="644" Encode="Base64"># --
# Copyright (C) 2021-2022 Znuny GmbH, https://znuny.org/
# --
# This software comes with ABSOLUTELY NO WARRANTY. For details, see
# the enclosed file COPYING for license information (AGPL). If you
# did not receive this file, see http://www.gnu.org/licenses/agpl.txt.
# --

package Kernel::System::Mention;

use strict;
use warnings;

use Kernel::Language qw(Translatable);
use Kernel::System::VariableCheck qw(:all);

use parent qw(Kernel::System::EventHandler);

our @ObjectDependencies = (
    'Kernel::Config',
    'Kernel::System::DB',
    'Kernel::System::Group',
    'Kernel::System::Log',
    'Kernel::System::Ticket',
    'Kernel::System::Ticket::Article',
    'Kernel::System::User',
);

=head1 NAME

Kernel::System::Mention

=head1 DESCRIPTION

Support for mentioning users.

=head1 PUBLIC INTERFACE

=head2 new()

Don't use the constructor directly, use the ObjectManager instead:

    my $MentionObject = $Kernel::OM->Get('Kernel::System::Mention');

=cut

sub new {
    my ( $Type, %Param ) = @_;

    my $Self = {};
    bless( $Self, $Type );

    $Self->EventHandlerInit(
        Config => 'Ticket::EventModulePost',
    );

    return $Self;
}

=head2 AddMention()

Adds a mention and triggers event "UserMention" to send a notification.

    my $Success = $MentionObject->AddMention(
        TicketID        => 3252,
        ArticleID       => 6538,
        MentionedUserID => 5,
        UserID          => 1,
    );

    Returns true value on success.

=cut

sub AddMention {
    my ( $Self, %Param ) = @_;

    my $LogObject     = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject      = $Kernel::OM->Get('Kernel::System::DB');
    my $ConfigObject  = $Kernel::OM->Get('Kernel::Config');
    my $ArticleObject = $Kernel::OM->Get('Kernel::System::Ticket::Article');
    my $TicketObject  = $Kernel::OM->Get('Kernel::System::Ticket');

    NEEDED:
    for my $Needed (qw(TicketID ArticleID MentionedUserID UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $TicketID        = $Param{TicketID};
    my $ArticleID       = $Param{ArticleID};
    my $MentionedUserID = $Param{MentionedUserID};

    my $Mentions = $Self->GetTicketMentions(
        TicketID => $TicketID,
    );
    return if ref $Mentions ne 'ARRAY';

    my $MentionExists = grep {
        $_->{UserID} == $MentionedUserID
            && $_->{TicketID} == $TicketID
            && $_->{ArticleID} == $ArticleID
    } @{$Mentions};
    return 1 if $MentionExists;

    my $TicketFlagSet = $TicketObject->TicketFlagSet(
        TicketID => $TicketID,
        Key      => 'MentionSeen',
        Value    => 0,
        UserID   => $MentionedUserID,
    );
    return if !$TicketFlagSet;

    return if !$DBObject->Do(
        SQL => '
            INSERT INTO
                mention (user_id, ticket_id, article_id, create_time)
                VALUES (?, ?, ?, current_timestamp)
        ',
        Bind => [
            \$MentionedUserID,
            \$TicketID,
            \$ArticleID,
        ],
    );

    my $NotificationsConfig = $ConfigObject->Get('Mentions')->{Notifications};
    if ( $NotificationsConfig && $NotificationsConfig eq 'Ticket' ) {
        my $IsUserMentionedTicket = grep {
            $_->{UserID} == $MentionedUserID
                && $_->{TicketID} == $TicketID
        } @{$Mentions};

        return if $IsUserMentionedTicket;
    }

    $Self->EventHandler(
        Event => 'UserMention',
        Data  => {
            TicketID   => $TicketID,
            ArticleID  => $ArticleID,
            Recipients => [ $MentionedUserID, ],
        },
        UserID => $Param{UserID},
    );

    return 1;
}

=head2 CanUserRemoveMention()

Checks if the given user is allowed to remove the given mention.

    my $UserCanRemoveMention = $MentionObject->CanUserRemoveMention(
        TicketID        => 3252,
        MentionedUserID => 5,
        UserID          => 1, # user who wants to remove the mention
    );

    Returns true value if the user is allowed to remove the mention.

=cut

sub CanUserRemoveMention {
    my ( $Self, %Param ) = @_;

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $TicketObject = $Kernel::OM->Get('Kernel::System::Ticket');

    NEEDED:
    for my $Needed (qw(TicketID MentionedUserID UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    # User can remove his own mention.
    return 1 if $Param{MentionedUserID} == $Param{UserID};

    # User can remove any mention from a ticket he owns.
    my %Ticket = $TicketObject->TicketGet(
        TicketID => $Param{TicketID},
        UserID   => $Param{UserID},
    );
    return   if !%Ticket;
    return 1 if $Ticket{OwnerID} == $Param{UserID};

    return;
}

=head2 RemoveMention()

Removes all mentions of a ticket for a specific user ID.

    my $Success = $MentionObject->RemoveMention(
        TicketID        => 3252,
        MentionedUserID => 5,
        UserID          => 1, # user who wants to remove the mention
    );

    Returns true value on success.

=cut

sub RemoveMention {
    my ( $Self, %Param ) = @_;

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    NEEDED:
    for my $Needed (qw(TicketID MentionedUserID UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $UserCanRemoveMention = $Self->CanUserRemoveMention(
        TicketID        => $Param{TicketID},
        MentionedUserID => $Param{MentionedUserID},
        UserID          => $Param{UserID},
    );
    if ( !$UserCanRemoveMention ) {
        $LogObject->Log(
            Priority => 'error',
            Message =>
                "User with ID $Param{UserID} is not allowed to remove mention of user with ID $Param{MentionedUserID} from ticket with ID $Param{TicketID}.",
        );

        return;
    }

    return if !$DBObject->Do(
        SQL => '
            DELETE FROM mention
            WHERE       ticket_id = ?
                        AND user_id = ?
        ',
        Bind => [
            \$Param{TicketID},
            \$Param{MentionedUserID},
        ],
    );

    return 1;
}

=head2 GetTicketMentions()

Retrieves all mentions of a ticket.

    my $Mentions = $MentionObject->GetTicketMentions(
        TicketID  => 3252,
        OrderBy   => 'create_time', # optional; default
        SortOrder => 'ASC', # or 'DESC', optional; default
    );

    Returns:

    my $Mentions = [
        {
            UserID     => 5,
            TicketID   => 76,
            ArticleID  => 89,
            CreateTime => '2022-07-20 12:34:23',
        },
        # ...
    ];

=cut

sub GetTicketMentions {
    my ( $Self, %Param ) = @_;

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    NEEDED:
    for my $Needed (qw(TicketID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $OrderByQuery = $Self->_AssembleOrderByQuery(
        OrderBy   => $Param{OrderBy},
        SortOrder => $Param{SortOrder},
    );
    return if !$OrderByQuery;

    return if !$DBObject->Prepare(
        SQL => "
            SELECT   user_id, ticket_id, article_id, create_time
            FROM     mention
            WHERE    ticket_id = ?
            ORDER BY $OrderByQuery
        ",
        Bind => [
            \$Param{TicketID},
        ],
    );

    my @Mentions;
    while ( my @Row = $DBObject->FetchrowArray() ) {
        my %Mention = (
            UserID     => $Row[0],
            TicketID   => $Row[1],
            ArticleID  => $Row[2],
            CreateTime => $Row[3],
        );

        push @Mentions, \%Mention;
    }

    return \@Mentions;
}

=head2 GetUserMentions()

Retrieves all mentions of a user.

    my $Mentions = $MentionObject->GetUserMentions(
        UserID     => 87,

        # optional, defaults to 0 and then means that all mentions of all articles per ticket are
        # counted as one combined mention;
        # if set to 1, all mentions of every article count separately
        PerArticle => 0,

        OrderBy    => 'create_time', # optional; default
        SortOrder  => 'ASC', # or 'DESC', optional; default
    );

    Returns:

    my $Mentions = [
        {
            UserID   => 5,
            TicketID => 76,

            # the following two entries will only be returned if parameter PerArticle is set
            ArticleID  => 89,
            CreateTime => '2022-07-20 12:34:23',
        },
        # ...
    ];

=cut

sub GetUserMentions {
    my ( $Self, %Param ) = @_;

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');
    my $DBObject  = $Kernel::OM->Get('Kernel::System::DB');

    NEEDED:
    for my $Needed (qw(UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $OrderByQuery = $Self->_AssembleOrderByQuery(
        OrderBy   => $Param{OrderBy},
        SortOrder => $Param{SortOrder},
    );
    return if !$OrderByQuery;

    $DBObject->Prepare(
        SQL => "
            SELECT   user_id, ticket_id, article_id, create_time
            FROM     mention
            WHERE    user_id = ?
            ORDER BY $OrderByQuery
        ",
        Bind => [
            \$Param{UserID},
        ],
    );

    my $PerArticle = $Param{PerArticle} // 0;
    my %MentionedTicketIDs;

    my @Mentions;
    ROW:
    while ( my @Row = $DBObject->FetchrowArray() ) {
        my $TicketID = $Row[1];
        if ( !$PerArticle ) {
            next ROW if $MentionedTicketIDs{$TicketID};

            $MentionedTicketIDs{$TicketID} = 1;
        }

        my %Mention = (
            UserID   => $Row[0],
            TicketID => $TicketID,
        );

        if ($PerArticle) {
            $Mention{ArticleID}  = $Row[2];
            $Mention{CreateTime} = $Row[3];
        }

        push @Mentions, \%Mention;
    }

    return \@Mentions;
}

=head2 GetDashboardWidgetTicketData()

Returns data for dashboard widgets that output information about mentions.

    my $Data = $MentionObject->GetDashboardWidgetTicketData(
        UserID => 37,
    );

    Returns:

    my $Data = (
        TicketIDs     => [ 5, 27, 382, ],
        CustomColumns => {
            5 => {
                LastMention => '2022-07-03 10:32:42',
            },
            27 => {
                LastMention => '2022-07-08 14:56:20',
            },
            382 => {
                LastMention => '2022-07-25 16:09:12',
            },
        },
    );

=cut

sub GetDashboardWidgetTicketData {
    my ( $Self, %Param ) = @_;

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(UserID)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $Mentions = $Self->GetUserMentions(
        UserID     => $Param{UserID},
        PerArticle => 1,
        OrderBy    => 'create_time',
        SortOrder  => 'ASC',
    );
    return if !IsArrayRefWithData($Mentions);

    my %TicketIDs;
    my %CustomColumns;
    for my $Mention ( @{$Mentions} ) {
        $TicketIDs{ $Mention->{TicketID} } = 1;

        # Note: This will only use the creation time of the newest article
        # per ticket.
        $CustomColumns{ $Mention->{TicketID} }->{ Translatable('LastMention') } = $Mention->{CreateTime};
    }

    my %Data = (
        TicketIDs     => [ sort keys %TicketIDs ],
        CustomColumns => \%CustomColumns,
    );

    return \%Data;
}

=head2 GetMentionedUserIDsFromString()

    Parses HTML string and returns the IDs of found mentioned users.

    my $MentionedUserIDs = $MentionObject->GetMentionedUserIDsFromString(
        HTMLString => '...<a class="Mention" href="..." target="...">@root@localhost<\/a>...',

        # optional
        # plain text string must be given if mentions in quoted text should be ignored.
        # they are not reliably parsable from the HTML string.
        PlainTextString => '...@root@localhost...',

        # optional
        # Limit for number of returned user IDs. The rest will silently be ignored.
        Limit => 5,
    );

    Returns:
    my $MentionedUserIDs = [ 1, 5, ],

=cut

sub GetMentionedUserIDsFromString {
    my ( $Self, %Param ) = @_;

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    NEEDED:
    for my $Needed (qw(HTMLString)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $MentionsRichtTextEditorConfig = $ConfigObject->Get('Mentions::RichTextEditor') // {};
    my $MentionsTriggerConfig         = $MentionsRichtTextEditorConfig->{Triggers};
    return [] if !IsHashRefWithData($MentionsTriggerConfig);

    my @MentionedUsers = (
        $Param{HTMLString}
            =~ m{<a\b[^>]*?\bclass="Mention"[^>]*?>\Q$MentionsTriggerConfig->{User}\E(.*?)<\/a>}sg
    );

    my @MentionedGroups = (
        $Param{HTMLString}
            =~ m{<a\b[^>]*?\bclass="GroupMention"[^>]*?>\Q$MentionsTriggerConfig->{Group}\E(.*?)<\/a>}sg
    );

    # If plain text has additionally been given, use it to remove quoted text (lines starting
    # with characters configured in Ticket::Frontend::Quote) and match the remaining
    # contained mentions with those of the given HTML string.
    #
    # This avoids notification for mentions contained in quoted text.
    #
    # Mentions cannot be removed from quotations in HTML string because
    # parsing is not reliably possible.
    my $QuoteMarker = $ConfigObject->Get('Ticket::Frontend::Quote');
    if (
        IsStringWithData( $Param{PlainTextString} )
        && IsStringWithData($QuoteMarker)
        )
    {
        # Remove every line that starts with a quote marker.
        ( my $PlainTextStringWithoutQuote = $Param{PlainTextString} ) =~ s{^\Q$QuoteMarker\E.*}{}mig;

        # Drop found mentioned users that are not part of the plain text without quotes.
        if ( defined $PlainTextStringWithoutQuote ) {
            @MentionedUsers = grep { $PlainTextStringWithoutQuote =~ m{\[\d+\]\Q$MentionsTriggerConfig->{User}\E$_\b}m }
                @MentionedUsers;
        }

        # Drop found mentioned groups that are not part of the plain text without quotes.
        if ( defined $PlainTextStringWithoutQuote ) {
            @MentionedGroups
                = grep { $PlainTextStringWithoutQuote =~ m{\[\d+\]\Q$MentionsTriggerConfig->{Group}\E$_\b}m }
                @MentionedGroups;
        }
    }

    # Filter out blocked groups
    @MentionedGroups = grep { !$Self->IsGroupBlocked( Group => $_ ) } @MentionedGroups;

    if (@MentionedGroups) {
        my $GroupUsers = $Self->_GetUsersOfGroups(
            Groups => \@MentionedGroups,
        );
        if ( IsArrayRefWithData($GroupUsers) ) {
            push @MentionedUsers, @{$GroupUsers};
        }
    }

    return [] if !@MentionedUsers;

    # Remove duplicate users but keep their order because of possible configured limit.
    # Note that the order of users within a group is arbitrary.
    my %UniqueMentionedUsers;
    my @UniqueMentionedUsers;

    MENTIONEDUSER:
    for my $MentionedUser (@MentionedUsers) {
        next MENTIONEDUSER if $UniqueMentionedUsers{$MentionedUser};

        $UniqueMentionedUsers{$MentionedUser} = 1;
        push @UniqueMentionedUsers, $MentionedUser;
    }

    if ( $Param{Limit} && @UniqueMentionedUsers > $Param{Limit} ) {
        @UniqueMentionedUsers = @UniqueMentionedUsers[ 0 .. $Param{Limit} - 1 ];
    }

    my $MentionedUserIDs = $Self->_GetMentionedUserIDs(
        MentionedUsers => \@UniqueMentionedUsers,
    );

    return $MentionedUserIDs;
}

=head2 IsGroupBlocked()

Checks if the given group is blocked for mentioning by SysConfig option 'Mentions###BlockedGroups'.

    my $GroupIsBlocked = $MentionObject->IsGroupBlocked(
        Group => 'users',
    );

    Returns true value of group is blocked.

=cut

sub IsGroupBlocked {
    my ( $Self, %Param ) = @_;

    my $LogObject    = $Kernel::OM->Get('Kernel::System::Log');
    my $ConfigObject = $Kernel::OM->Get('Kernel::Config');

    NEEDED:
    for my $Needed (qw(Group)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    my $MentionsConfig = $ConfigObject->Get('Mentions') // {};

    my $BlockedGroups = $MentionsConfig->{BlockedGroups} // [];
    my %BlockedGroups = map { $_ => 1 } @{$BlockedGroups};
    return if !%BlockedGroups;
    return if !$BlockedGroups{ $Param{Group} };

    return 1;
}

sub _GetUsersOfGroups {
    my ( $Self, %Param ) = @_;

    my $GroupObject = $Kernel::OM->Get('Kernel::System::Group');
    my $LogObject   = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(Groups)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    return if !IsArrayRefWithData( $Param{Groups} );
    my %Groups = map { $_ => 1 } @{ $Param{Groups} };

    my %Users;
    GROUP:
    for my $Group ( sort keys %Groups ) {
        my $GroupID = $GroupObject->GroupLookup(
            Group => $Group,
        );
        next GROUP if !$GroupID;

        my %GroupUsers = $GroupObject->PermissionGroupUserGet(
            GroupID => $GroupID,
            Type    => 'ro',
        );
        next GROUP if !%GroupUsers;

        %Users = ( %Users, %GroupUsers );
    }

    my @Users = values %Users;

    return \@Users;
}

sub _GetMentionedUserIDs {
    my ( $Self, %Param ) = @_;

    my $UserObject = $Kernel::OM->Get('Kernel::System::User');
    my $LogObject  = $Kernel::OM->Get('Kernel::System::Log');

    NEEDED:
    for my $Needed (qw(MentionedUsers)) {
        next NEEDED if defined $Param{$Needed};

        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter '$Needed' is needed!",
        );
        return;
    }

    return if !IsArrayRefWithData( $Param{MentionedUsers} );

    my %MentionedUserIDs;
    USER:
    for my $User ( @{ $Param{MentionedUsers} } ) {
        my $UserID = $UserObject->UserLookup(
            UserLogin => $User,
        );
        next USER if !$UserID;

        $MentionedUserIDs{$UserID} = 1;
    }

    my @MentionedUserIDs = sort keys %MentionedUserIDs;

    return \@MentionedUserIDs;
}

sub _AssembleOrderByQuery {
    my ( $Self, %Param ) = @_;

    my $LogObject = $Kernel::OM->Get('Kernel::System::Log');

    my $OrderBy      = $Param{OrderBy} // 'create_time';
    my %ValidOrderBy = (
        user_id     => 1,
        ticket_id   => 1,
        article_id  => 1,
        create_time => 1,
    );
    if ( !$ValidOrderBy{$OrderBy} ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter 'OrderBy' contains invalid column name!",
        );
        return;
    }

    my $SortOrder = $Param{SortOrder} // 'ASC';
    if ( $SortOrder !~ m{\A(?:ASC|DESC)\z}i ) {
        $LogObject->Log(
            Priority => 'error',
            Message  => "Parameter 'SortOrder' contains invalid value!",
        );
        return;
    }

    my $OrderByQuery = "$OrderBy $SortOrder";

    return $OrderByQuery;
}

1;</File>
        <File Location="var/httpd/htdocs/js/Znuny.Agent.TicketMention.js" Permission="644" Encode="Base64">Ly8gLS0KLy8gQ29weXJpZ2h0IChDKSAyMDIxLTIwMjIgWm51bnkgR21iSCwgaHR0cHM6Ly96bnVueS5vcmcvCi8vIC0tCi8vIFRoaXMgc29mdHdhcmUgY29tZXMgd2l0aCBBQlNPTFVURUxZIE5PIFdBUlJBTlRZLiBGb3IgZGV0YWlscywgc2VlCi8vIHRoZSBlbmNsb3NlZCBmaWxlIENPUFlJTkcgZm9yIGxpY2Vuc2UgaW5mb3JtYXRpb24gKEFHUEwpLiBJZiB5b3UKLy8gZGlkIG5vdCByZWNlaXZlIHRoaXMgZmlsZSwgc2VlIGh0dHA6Ly93d3cuZ251Lm9yZy9saWNlbnNlcy9hZ3BsLnR4dC4KLy8gLS0KCiJ1c2Ugc3RyaWN0IjsKCnZhciBDb3JlID0gQ29yZSB8fCB7fSwKICAgIFpudW55ID0gWm51bnkgfHwge307CgpabnVueS5BZ2VudCA9IFpudW55LkFnZW50IHx8IHt9OwoKWm51bnkuQWdlbnQuVGlja2V0TWVudGlvbiA9IChmdW5jdGlvbiAoVGFyZ2V0TlMpIHsKCiAgICBUYXJnZXROUy5Jbml0ID0gZnVuY3Rpb24gKCkgewogICAgICAgIHZhciBEYXRhID0gewogICAgICAgICAgICBBY3Rpb246ICAgICdNZW50aW9ucycsCiAgICAgICAgICAgIFN1YmFjdGlvbjogJ1JlbW92ZScsCiAgICAgICAgICAgIE1lbnRpb25lZFVzZXJJRDogJCgnI1JlbW92ZU1lbnRpb24nKS5kYXRhKCd1c2VyLWlkJyksCiAgICAgICAgfTsKCiAgICAgICAgJCgnI1JlbW92ZU1lbnRpb24nKS5vbignY2xpY2snLGZ1bmN0aW9uKEV2ZW50KXsKICAgICAgICAgICAgRXZlbnQucHJldmVudERlZmF1bHQoKTsKCiAgICAgICAgICAgICQoJy5NYXN0ZXJBY3Rpb24nKS5lYWNoKGZ1bmN0aW9uKCl7CiAgICAgICAgICAgICAgICB2YXIgVGlja2V0SUQgID0gJCh0aGlzKS5hdHRyKCdpZCcpLnJlcGxhY2UoJ1RpY2tldElEXycsICcnKTsKICAgICAgICAgICAgICAgIERhdGEuVGlja2V0SUQgPSBUaWNrZXRJRDsKCiAgICAgICAgICAgICAgICBpZigkKHRoaXMpLmNoaWxkcmVuKCkuZmlyc3QoKS5jaGlsZHJlbigpLmZpcnN0KCkucHJvcCgnY2hlY2tlZCcpKSB7CiAgICAgICAgICAgICAgICAgICAgQ29yZS5BSkFYLkZ1bmN0aW9uQ2FsbCgKICAgICAgICAgICAgICAgICAgICAgICAgQ29yZS5Db25maWcuR2V0KCdCYXNlbGluaycpLAogICAgICAgICAgICAgICAgICAgICAgICBEYXRhLAogICAgICAgICAgICAgICAgICAgICAgICBmdW5jdGlvbigpIHsKICAgICAgICAgICAgICAgICAgICAgICAgICAgIGxvY2F0aW9uLnJlbG9hZCgpOwogICAgICAgICAgICAgICAgICAgICAgICB9CiAgICAgICAgICAgICAgICAgICAgKQogICAgICAgICAgICAgICAgfQogICAgICAgICAgICB9KTsKICAgICAgICB9KTsKICAgIH07CgogICAgQ29yZS5Jbml0LlJlZ2lzdGVyTmFtZXNwYWNlKFRhcmdldE5TLCAnQVBQX01PRFVMRScpOwoKICAgIHJldHVybiBUYXJnZXROUzsKfShabnVueS5BZ2VudC5UaWNrZXRNZW50aW9uIHx8IHt9KSk7</File>
    </Filelist>
</otrs_package>