%# BEGIN BPS TAGGED BLOCK {{{ %# %# COPYRIGHT: %# %# This software is Copyright (c) 1996-2015 Best Practical Solutions, LLC %# %# %# (Except where explicitly superseded by other copyright notices) %# %# %# LICENSE: %# %# This work is made available to you under the terms of Version 2 of %# the GNU General Public License. A copy of that license should have %# been provided with this software, but in any event can be snarfed %# from www.gnu.org. %# %# This work is distributed in the hope that it will be useful, but %# WITHOUT ANY WARRANTY; without even the implied warranty of %# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU %# General Public License for more details. %# %# You should have received a copy of the GNU General Public License %# along with this program; if not, write to the Free Software %# Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA %# 02110-1301 or visit their web page on the internet at %# http://www.gnu.org/licenses/old-licenses/gpl-2.0.html. %# %# %# CONTRIBUTION SUBMISSION POLICY: %# %# (The following paragraph is not intended to limit the rights granted %# to you to modify and distribute this software under the terms of %# the GNU General Public License and is only of importance to you if %# you choose to contribute your changes and enhancements to the %# community by submitting them to Best Practical Solutions, LLC.) %# %# By intentionally submitting any modifications, corrections or %# derivatives to this work, or any other work intended for use with %# Request Tracker, to Best Practical Solutions, LLC, you confirm that %# you are the copyright holder for those contributions and you grant %# Best Practical Solutions, LLC a nonexclusive, worldwide, irrevocable, %# royalty-free, perpetual, license to use, copy, create derivative %# works based on those contributions, and sublicense and distribute %# those contributions and any derivatives thereof. %# %# END BPS TAGGED BLOCK }}} <& /Elements/Header, Title => $title, onload => "function () { hide('Ticket-Create-details') }" &> <& /Elements/Tabs &> <& /Elements/ListActions, actions => \@results &>
% $m->callback( CallbackName => 'FormStart', QueueObj => $QueueObj, ARGSRef => \%ARGS ); % if ($gnupg_widget) { <& /Elements/Crypt/SignEncryptWidget:ShowIssues, self => $gnupg_widget &> % }
<&| /Widgets/TitleBox, title => loc("Basics"), class=>'ticket-info-basics' &> <& /Ticket/Elements/EditBasics, InTable => 1, fields => [ { name => 'Queue', comp => '/Elements/SelectQueue', args => { Name => 'Queue', Default => $QueueObj->Name, QueueObj => $QueueObj, ShowNullOption => 0, ShowAllQueues => 0, OnChange => "document.getElementsByName('id')[0].value = 'refresh'; form.submit()", }, }, { name => 'Status', comp => '/Ticket/Elements/SelectStatus', args => { Name => "Status", QueueObj => $QueueObj, }, }, { name => 'Owner', comp => '/Elements/SelectOwner', args => { Name => "Owner", Default => $ARGS{Owner} || RT->Nobody->Id, DefaultValue => 0, QueueObj => $QueueObj, }, } ] &> % $m->callback( CallbackName => 'AfterOwner', ARGSRef => \%ARGS ); <& /Elements/EditCustomFields, %ARGS, Object => $ticket, CustomFields => $QueueObj->TicketCustomFields, Grouping => 'Basics', InTable => 1, KeepValue => 1, &> <& /Ticket/Elements/EditTransactionCustomFields, %ARGS, QueueObj => $QueueObj, InTable => 1, KeepValue => 1, &>
% $m->callback( CallbackName => 'AfterBasics', QueueObj => $QueueObj, ARGSRef => \%ARGS ); <& /Elements/EditCustomFieldCustomGroupings, %ARGS, Object => $ticket, CustomFieldGenerator => sub { $QueueObj->TicketCustomFields }, KeepValue => 1, &>
<&| /Widgets/TitleBox, title => $title, class => 'messagedetails' &> <& /Elements/EditCustomFields, %ARGS, Object => $ticket, CustomFields => $QueueObj->TicketCustomFields, Grouping => 'People', InTable => 1, KeepValue => 1, &> % if ( $gnupg_widget ) { % } <& /Ticket/Elements/AddAttachments, %ARGS, QueueObj => $QueueObj &>
<&|/l&>Requestors: <& /Elements/EmailInput, Name => 'Requestors', Size => undef, Default => $ARGS{Requestors} // $session{CurrentUser}->EmailAddress, AutocompleteMultiple => 1 &> % $m->callback( CallbackName => 'AfterRequestors', QueueObj => $QueueObj, ARGSRef => \%ARGS );
<&|/l&>Cc: <& /Elements/EmailInput, Name => 'Cc', Size => undef, Default => $ARGS{Cc}, AutocompleteMultiple => 1 &>
  <&|/l&>(Sends a carbon-copy of this update to a comma-delimited list of email addresses. These people will receive future updates.)
<&|/l&>Admin Cc: <& /Elements/EmailInput, Name => 'AdminCc', Size => undef, Default => $ARGS{AdminCc}, AutocompleteMultiple => 1 &>
  <&|/l&>(Sends a carbon-copy of this update to a comma-delimited list of administrative email addresses. These people will receive future updates.)
<&|/l&>Subject: % $m->callback( %ARGS, CallbackName => 'AfterSubject' );
  <& /Elements/Crypt/SignEncryptWidget, self => $gnupg_widget, QueueObj => $QueueObj &>
<&|/l&>Describe the issue below:
% if ( RT->Config->Get('ArticleOnTicketCreate')) { <& /Articles/Elements/BeforeMessageBox, %ARGS, QueueObj => $QueueObj &> % } % $m->callback( %ARGS, QueueObj => $QueueObj, CallbackName => 'BeforeMessageBox' ); % if (exists $ARGS{Content}) { <& /Elements/MessageBox, Default => $ARGS{Content}, IncludeSignature => 0 &> % } else { <& /Elements/MessageBox, QuoteTransaction => $QuoteTransaction &> %} % $m->callback( %ARGS, QueueObj => $QueueObj, CallbackName => 'AfterMessageBox' );
<& /Elements/Submit, Label => loc("Create"), id => 'SubmitTicket' &>
<&| /Widgets/TitleBox, title => loc('The Basics'), title_class=> 'inverse', color => "#993333" &>
<&|/l&>Priority: <& /Elements/SelectPriority, Name => "InitialPriority", Default => $ARGS{InitialPriority} ? $ARGS{InitialPriority} : $QueueObj->InitialPriority, &>
<&|/l&>Final Priority: <& /Elements/SelectPriority, Name => "FinalPriority", Default => $ARGS{FinalPriority} ? $ARGS{FinalPriority} : $QueueObj->FinalPriority, &>
<&|/l&>Time Estimated: <& /Elements/EditTimeValue, Name => 'TimeEstimated', Default => $ARGS{TimeEstimated} || '' &>
<&|/l&>Time Worked: <& /Elements/EditTimeValue, Name => 'TimeWorked', Default => $ARGS{TimeWorked} || '' &>
<&|/l&>Time Left: <& /Elements/EditTimeValue, Name => 'TimeLeft', Default => $ARGS{TimeLeft} || '' &>

<&|/Widgets/TitleBox, title => loc("Dates"), title_class=> 'inverse', color => "#663366" &> <& /Elements/EditCustomFields, %ARGS, Object => $ticket, CustomFields => $QueueObj->TicketCustomFields, Grouping => 'Dates', InTable => 1, KeepValue => 1, &>
<&|/l&>Starts:<& /Elements/SelectDate, Name => "Starts", Default => $ARGS{Starts} || '' &>
<&|/l&>Due:<& /Elements/SelectDate, Name => "Due", Default => $ARGS{Due} || '' &>


<& /Elements/Submit, Label => loc("Create") &>
<%INIT> $m->callback( CallbackName => "Init", ARGSRef => \%ARGS ); my $Queue = $ARGS{Queue}; $session{DefaultQueue} = $Queue; my $current_user = $session{'CurrentUser'}; if ($CloneTicket) { my $CloneTicketObj = RT::Ticket->new( $session{CurrentUser} ); $CloneTicketObj->Load($CloneTicket) or Abort( loc("Ticket could not be loaded") ); my $clone = { Requestors => join( ',', $CloneTicketObj->RequestorAddresses ), Cc => join( ',', $CloneTicketObj->CcAddresses ), AdminCc => join( ',', $CloneTicketObj->AdminCcAddresses ), InitialPriority => $CloneTicketObj->Priority, }; $clone->{$_} = $CloneTicketObj->$_() for qw/Owner Subject FinalPriority Status/; # not TimeWorked, TimeEstimated, or TimeLeft $clone->{$_} = $CloneTicketObj->$_->AsString for grep { $CloneTicketObj->$_->IsSet } map { $_ . "Obj" } qw/Starts Started Due Resolved/; my $get_link_value = sub { my ($link, $type) = @_; my $uri_method = $type . 'URI'; my $local_method = 'Local' . $type; my $uri = $link->$uri_method; return if $uri->IsLocal and $uri->Object and $uri->Object->isa('RT::Ticket') and $uri->Object->Type eq 'reminder'; return $link->$local_method || $uri->URI; }; my (@refers, @refers_by); my $refers = $CloneTicketObj->RefersTo; while ( my $refer = $refers->Next ) { my $refer_value = $get_link_value->($refer, 'Target'); push @refers, $refer_value if defined $refer_value; } $clone->{'new-RefersTo'} = join ' ', @refers; my $refers_by = $CloneTicketObj->ReferredToBy; while ( my $refer_by = $refers_by->Next ) { my $refer_by_value = $get_link_value->($refer_by, 'Base'); push @refers_by, $refer_by_value if defined $refer_by_value; } $clone->{'RefersTo-new'} = join ' ', @refers_by; my $cfs = $CloneTicketObj->QueueObj->TicketCustomFields(); while ( my $cf = $cfs->Next ) { next if $cf->FirstAttribute('NoClone'); my $cf_id = $cf->id; my $cf_values = $CloneTicketObj->CustomFieldValues( $cf->id ); my @cf_values; while ( my $cf_value = $cf_values->Next ) { push @cf_values, $cf_value->Content; } if ( @cf_values > 1 && $cf->Type eq 'Select' ) { $clone->{GetCustomFieldInputName( CustomField => $cf )} = \@cf_values; } else { $clone->{GetCustomFieldInputName( CustomField => $cf )} = join "\n", @cf_values; } } # Pass customer links along (even though cloning of parent links # in general is disabled). my $customers = $CloneTicketObj->Customers; my @customers; while ( my $customer = $customers->Next ) { my ($custnum) = $customer->Target =~ /cust_main\/(\d+)$/; push @customers, $custnum if $custnum; } $clone->{'new-Customer'} = join(' ', @customers); for ( keys %$clone ) { $ARGS{$_} = $clone->{$_} if not defined $ARGS{$_}; } } my @results; my $title = loc("Create a new ticket"); my $QueueObj = RT::Queue->new($current_user); $QueueObj->Load($Queue) || Abort(loc("Queue [_1] could not be loaded.", $Queue||'')); $m->callback( QueueObj => $QueueObj, title => \$title, results => \@results, ARGSRef => \%ARGS ); $m->scomp( '/Articles/Elements/SubjectOverride', ARGSRef => \%ARGS, QueueObj => $QueueObj, results => \@results ); $QueueObj->Disabled && Abort(loc("Cannot create tickets in a disabled queue.")); my $ticket = RT::Ticket->new($current_user); # empty ticket object ProcessAttachments(ARGSRef => \%ARGS); my $checks_failure = 0; { my ($status, @msg) = $m->comp( '/Elements/ValidateCustomFields', CustomFields => $QueueObj->TicketCustomFields, ARGSRef => \%ARGS ); unless ($status) { $checks_failure = 1; push @results, @msg; } } my $gnupg_widget = $m->comp('/Elements/Crypt/SignEncryptWidget:new', Arguments => \%ARGS ); $m->comp( '/Elements/Crypt/SignEncryptWidget:Process', self => $gnupg_widget, QueueObj => $QueueObj, ); if ( !exists $ARGS{'AddMoreAttach'} && ($ARGS{'id'}||'') eq 'new' ) { my $status = $m->comp('/Elements/Crypt/SignEncryptWidget:Check', self => $gnupg_widget, Operation => 'Create', QueueObj => $QueueObj, ); $checks_failure = 1 unless $status; } # check email addresses for RT's { foreach my $field ( qw(Requestors Cc AdminCc) ) { my $value = $ARGS{ $field }; next unless defined $value && length $value; my @emails = Email::Address->parse( $value ); foreach my $email ( grep RT::EmailParser->IsRTAddress($_->address), @emails ) { push @results, loc("[_1] is an address RT receives mail at. Adding it as a '[_2]' would create a mail loop", $email->format, loc($field =~ /^(.*?)s?$/) ); $checks_failure = 1; $email = undef; } $ARGS{ $field } = join ', ', map $_->format, grep defined, @emails; } } my $skip_create = 0; $m->callback( CallbackName => 'BeforeCreate', ARGSRef => \%ARGS, skip_create => \$skip_create, checks_failure => $checks_failure, results => \@results ); $m->comp( '/Articles/Elements/CheckSkipCreate', ARGSRef => \%ARGS, skip_create => \$skip_create, checks_failure => $checks_failure, results => \@results ); if ((!exists $ARGS{'AddMoreAttach'}) and (defined($ARGS{'id'}) and $ARGS{'id'} eq 'new')) { # new ticket? if ( !$checks_failure && !$skip_create ) { # CREATE THE TICKET. # For some reason it's done by a Mason component named "Display.html" # and the call is buried in obscure error-handling stuff. # This comment exists to make it more visually obvious. # ************************************************************ $m->comp('Display.html', %ARGS); # ************************************************************ # Execution should not continue here. Display.html calls # Redirect() which does an $m->abort. We only get here if the # code dies before then, hence "$@". $RT::Logger->crit("After display call; error is $@"); $m->abort(); } } PageMenu->child( basics => raw_html => q[] . loc('Basics') . q[]); PageMenu->child( details => raw_html => q[] . loc('Details') . q[]); <%ARGS> $DependsOn => undef $DependedOnBy => undef $MemberOf => undef $QuoteTransaction => undef $CloneTicket => undef