Merge bugfixes from 'upstream/master'

This commit is contained in:
Garret Wassermann 2020-11-12 02:17:10 -05:00
commit 4d83067826
8 changed files with 647 additions and 621 deletions

File diff suppressed because it is too large Load Diff

View File

@ -8,20 +8,22 @@ def clear_secret_keys(apps, schema_editor):
db_alias = schema_editor.connection.alias db_alias = schema_editor.connection.alias
for ticket in Ticket.objects.using(db_alias).all(): for ticket in Ticket.objects.using(db_alias).all():
ticket.secret_key='' ticket.secret_key = ''
ticket.save() ticket.save()
class Migration(migrations.Migration): class Migration(migrations.Migration):
dependencies = [ dependencies = [
('helpdesk', '0018_fix_migrations'), ('helpdesk', '0018_ticket_secret_key'),
] ]
operations = [ operations = [
migrations.AddField( migrations.AddField(
model_name='ticket', model_name='ticket',
name='secret_key', name='secret_key',
field=models.CharField(default=helpdesk.models.mk_secret, max_length=36, verbose_name='Secret key needed for viewing/editing ticket by non-logged in users'), field=models.CharField(default=helpdesk.models.mk_secret, max_length=36,
verbose_name='Secret key needed for viewing/editing ticket by non-logged in users'),
), ),
migrations.RunPython(clear_secret_keys), migrations.RunPython(clear_secret_keys),
] ]

View File

@ -7,6 +7,7 @@ def forwards_func(apps, schema_editor):
EmailTemplate = apps.get_model("helpdesk", "EmailTemplate") EmailTemplate = apps.get_model("helpdesk", "EmailTemplate")
db_alias = schema_editor.connection.alias db_alias = schema_editor.connection.alias
EmailTemplate.objects.using(db_alias).create( EmailTemplate.objects.using(db_alias).create(
id=EmailTemplate.objects.order_by('-id').first().id + 1, # because PG sequences are not reset
template_name='merged', template_name='merged',
subject='(Merged)', subject='(Merged)',
heading='Ticket merged', heading='Ticket merged',
@ -23,6 +24,7 @@ From now on, please answer on this ticket, or you can include the tag {{ ticket.
locale='en' locale='en'
) )
EmailTemplate.objects.using(db_alias).create( EmailTemplate.objects.using(db_alias).create(
id=EmailTemplate.objects.order_by('-id').first().id + 1, # because PG sequences are not reset
template_name='merged', template_name='merged',
subject='(Fusionné)', subject='(Fusionné)',
heading='Ticket Fusionné', heading='Ticket Fusionné',
@ -32,7 +34,7 @@ Ce courriel indicatif permet de vous prévenir que le ticket {{ ticket.ticket }
Veillez à répondre sur ce ticket dorénavant, ou bien inclure la balise {{ ticket.merged_to.ticket }} dans le sujet de votre réponse par mail.""", Veillez à répondre sur ce ticket dorénavant, ou bien inclure la balise {{ ticket.merged_to.ticket }} dans le sujet de votre réponse par mail.""",
html="""<p style="font-family: sans-serif; font-size: 1em;">Bonjour,</p> html="""<p style="font-family: sans-serif; font-size: 1em;">Bonjour,</p>
<p style="font-family: sans-serif; font-size: 1em;">Ce courriel indicatif permet de vous prévenir que le ticket <b>{{ ticket.ticket }}</b> (<em>{{ ticket.title }}</em>) par {{ ticket.submitter_email }} a été fusionné au ticket <a href="{{ ticket.merged_to.staff_url }}">{{ ticket.merged_to.ticket }}</a>.</p> <p style="font-family: sans-serif; font-size: 1em;">Ce courriel indicatif permet de vous prévenir que le ticket <b>{{ ticket.ticket }}</b> (<em>{{ ticket.title }}</em>) par {{ ticket.submitter_email }} a été fusionné au ticket <a href="{{ ticket.merged_to.staff_url }}">{{ ticket.merged_to.ticket }}</a>.</p>
<p style="font-family: sans-serif; font-size: 1em;">Veillez à répondre sur ce ticket dorénavant, ou bien inclure la balise <b>{{ ticket.merged_to.ticket }}</b> dans le sujet de votre réponse par mail.</p>""", <p style="font-family: sans-serif; font-size: 1em;">Veillez à répondre sur ce ticket dorénavant, ou bien inclure la balise <b>{{ ticket.merged_to.ticket }}</b> dans le sujet de votre réponse par mail.</p>""",

View File

@ -1,24 +1,24 @@
$(document).ready(function() { $(document).ready(function() {
$("#filterBuilderButton").click(function() { $("#filterBuilderButton").click(function() {
var boxName = "#filterBox" + $("#filterBuilderSelect").val(); const boxName = "#filterBox" + $("#filterBuilderSelect").val();
$(boxName).slideDown(); $(boxName).slideDown();
return false; return false;
}); });
$(".filterBuilderRemove").click(function() { $(".filterBuilderRemove").click(function() {
var boxName = "#" + $(this).parents(".filterBox").attr('id'); const boxName = "#" + $(this).parents(".filterBox").attr('id');
$(boxName).slideUp(); $(boxName).slideUp();
$(boxName).children("input:text").each(function() { $(boxName).find("input:text").each(function() {
$(this).val(""); $(this).val("");
}); });
$(boxName).children("input:checkbox").each(function() { $(boxName).find("input:checkbox").each(function() {
this.checked = false; this.checked = false;
}); });
$(boxName).children("select").each(function() { $(boxName).find("select").each(function() {
this.selectedIndex = -1; this.selectedIndex = -1;
}); });
var selectId = $(this).parents(".filterBox").attr('id'); let selectId = $(this).parents(".filterBox").attr('id');
var attr = selectId.replace("filterBox", ""); const attr = selectId.replace("filterBox", "");
$("#filterBuilderSelect-" + attr)[0].disabled = ""; $("#filterBuilderSelect-" + attr)[0].disabled = "";
return false; return false;
@ -31,9 +31,9 @@ $(document).ready(function() {
* *
* @param {string} val name of selected filter value * @param {string} val name of selected filter value
*/ */
var onFilterChange = function(val) { const onFilterChange = function(val) {
if (val) { if (val) {
var boxName = "#filterBox" + val; const boxName = "#filterBox" + val;
$(boxName).slideDown(); $(boxName).slideDown();
$(boxName)[0].style.display="block"; $(boxName)[0].style.display="block";

View File

@ -1,63 +1,66 @@
{% extends "helpdesk/base.html" %} {% extends "helpdesk/base.html" %}
{% load i18n humanize %}
{% load static %} {% load i18n humanize static in_list %}
{% block helpdesk_title %}{% trans "Tickets" %}{% endblock %} {% block helpdesk_title %}{% trans "Tickets" %}{% endblock %}
{% block helpdesk_head %} {% block helpdesk_head %}
<link title="timeline-styles" rel="stylesheet" href="https://cdn.knightlab.com/libs/timeline3/latest/css/timeline.css">
{% endblock %} {% endblock %}
{% block h1_title %}Tickets {% block h1_title %}Tickets
{% if from_saved_query %} [{{ saved_query.title }}]{% endif %} {% if from_saved_query %} [{{ saved_query.title }}]{% endif %}
{% endblock %} {% endblock %}
{% block helpdesk_breadcrumb %} {% block helpdesk_breadcrumb %}
<li class="breadcrumb-item"> <li class="breadcrumb-item">
<a href="{% url 'helpdesk:list' %}">{% trans "Tickets" %}</a> <a href="{% url 'helpdesk:list' %}">{% trans "Tickets" %}</a>
</li> </li>
{% if from_saved_query and saved_query.user == user %} {% if from_saved_query and saved_query.user == user %}
<li class="breadcrumb-item">{% trans "Saved Query" %}</li> <li class="breadcrumb-item">{% trans "Saved Query" %}</li>
<li class="breadcrumb-item active">{{ saved_query.title }}</li> <li class="breadcrumb-item active">{{ saved_query.title }}</li>
{% else %} {% else %}
<li class="breadcrumb-item active">{% trans "Overview" %}</li> <li class="breadcrumb-item active">{% trans "Overview" %}</li>
{% endif %} {% endif %}
{% endblock %} {% endblock %}
{% block helpdesk_body %} {% block helpdesk_body %}
{% load in_list %} <div class="card">
<div class="card-header">
<div class="card"> <ul class="nav nav-tabs">
<div class="card-header"> <li class="nav-item" style="width: 200px;">
<ul class="nav nav-tabs"> {% trans "Query Results" %}:
<li class="nav-item" style="width: 200px;"> </li>
{% trans "Query Results" %}: <li class="nav-item">
</li> <a class="nav-link active" href="#datatabletabcontents" id="datatabletabcontents-tab"
<li class="nav-item""> data-toggle="tab" role="tab" aria-controls="datatabletabcontents" aria-selected=true>
<a class="nav-link active" href="#datatabletabcontents" id="datatabletabcontents-tab" data-toggle="tab" role="tab" aria-controls="datatabletabcontents" aria-selected=true> <i class="fas fa-th-list"></i>
<i class="fas fa-th-list"></i> {% trans "Table" %}
{% trans "Table" %} </a>
</a> </li>
</li> <li class="nav-item">
<li class="nav-item"> <a class="nav-link" href="#timelinetabcontents" id="timelinetabcontents-tab" data-toggle="tab"
<a class="nav-link" href="#timelinetabcontents" id="timelinetabcontents-tab" data-toggle="tab" role="tab" aria-controls="timelinetabcontents" aria-selected=false> role="tab" aria-controls="timelinetabcontents" aria-selected=false>
<i class="fas fa-history"></i> <i class="fas fa-history"></i>
{% trans "Timeline" %} {% trans "Timeline" %}
</a> </a>
</li> </li>
</ul> </ul>
</div> </div>
<div class="card-body"> <div class="card-body">
{{ search_message|safe }} {{ search_message|safe }}
<div class="tab-content" id="myTabContent"> <div class="tab-content" id="myTabContent">
<div class="tab-pane fade show active" id="datatabletabcontents" role="tabpanel" aria-labelledby="datatabletabcontents-tab"> <div class="tab-pane fade show active" id="datatabletabcontents" role="tabpanel"
<form method='post' action='{% url 'helpdesk:mass_update' %}' id="ticket_mass_update"> aria-labelledby="datatabletabcontents-tab">
<table width="100%" class="table table-sm table-striped table-bordered table-hover" id="ticketTable" data-page-length='{{ default_tickets_per_page }}'> <form method='post' action='{% url 'helpdesk:mass_update' %}' id="ticket_mass_update">
<thead class="thead-light"> {% csrf_token %}
<table class="table table-sm table-striped table-bordered table-hover"
id="ticketTable" data-page-length='{{ default_tickets_per_page }}'>
<thead class="thead-light">
<tr> <tr>
<th>&nbsp;</th> <th></th>
<th>{% trans "Ticket" %}</th> <th>{% trans "Ticket" %}</th>
<th>{% trans "Priority" %}</th> <th>{% trans "Priority" %}</th>
<th>{% trans "Queue" %}</th> <th>{% trans "Queue" %}</th>
@ -69,357 +72,365 @@
<th>{% trans "Time Spent" %}</th> <th>{% trans "Time Spent" %}</th>
<th>{% trans "KB item" %}</th> <th>{% trans "KB item" %}</th>
</tr> </tr>
</thead> </thead>
</table> </table>
<p><label>{% trans "Select:" %} </label> <p>
<label>{% trans "Select:" %}</label>
<button id="select_all_btn" type="button" class="btn btn-primary btn-sm" /> <button id="select_all_btn" type="button" class="btn btn-primary btn-sm">
<i class="fas fa-check-circle"></i>&nbsp;{% trans "All" %} <i class="fas fa-check-circle"></i> {% trans "All" %}
</button> </button>
<button id='select_none_btn' type="button" class="btn btn-primary btn-sm"><i class="fas fa-times-circle"></i>&nbsp;{% trans "None" %}</button> <button id='select_none_btn' type="button" class="btn btn-primary btn-sm">
<i class="fas fa-times-circle"></i> {% trans "None" %}
</button>
<button id='select_inverse_btn' type="button" class="btn btn-primary btn-sm"><i class="fas fa-expand-arrows-alt"></i>&nbsp;{% trans "Invert" %}</button> <button id='select_inverse_btn' type="button" class="btn btn-primary btn-sm">
</p> <i class="fas fa-expand-arrows-alt"></i> {% trans "Invert" %}
</button>
</p>
<p> <p>
<label for='id_mass_action'>{% trans "With Selected Tickets:" %}</label> <label for='id_mass_action'>{% trans "With Selected Tickets:" %}</label>
<select name='action' id='id_mass_action'> <select name='action' id='id_mass_action'>
<option value='take'>{% trans "Take (Assign to me)" %}</option> <option value='take'>{% trans "Take (Assign to me)" %}</option>
<option value='delete'>{% trans "Delete" %}</option> <option value='delete'>{% trans "Delete" %}</option>
<option value='merge'>{% trans "Merge" %}</option> <option value='merge'>{% trans "Merge" %}</option>
<optgroup label='{% trans "Close" %}'> <optgroup label='{% trans "Close" %}'>
<option value='close'>{% trans "Close (Don't Send E-Mail)" %}</option> <option value='close'>{% trans "Close (Don't Send E-Mail)" %}</option>
<option value='close_public'>{% trans "Close (Send E-Mail)" %}</option> <option value='close_public'>{% trans "Close (Send E-Mail)" %}</option>
</optgroup> </optgroup>
<optgroup label='{% trans "Assign To" %}'> <optgroup label='{% trans "Assign To" %}'>
<option value='unassign'>{% trans "Nobody (Unassign)" %}</option> <option value='unassign'>{% trans "Nobody (Unassign)" %}</option>
{% for u in user_choices %}<option value='assign_{{ u.id }}'>{{ u.get_username }}</option>{% endfor %} {% for u in user_choices %}
</optgroup> <option value='assign_{{ u.id }}'>{{ u.get_username }}</option>
<optgroup label='{% trans "Set KB Item" %}'> {% endfor %}
<option value='kbitem_none'>{% trans "No KB Item" %}</option> </optgroup>
{% for kbi in kb_items %}<option value='kbitem_{{ kbi.id }}'>{{kbi.category.title}}: {{ kbi.title }}</option>{% endfor %} <optgroup label='{% trans "Set KB Item" %}'>
</optgroup> <option value='kbitem_none'>{% trans "No KB Item" %}</option>
</select> {% for kbi in kb_items %}
<button type="submit" class="btn btn-primary btn-sm"><i class="fas fa-arrow-circle-right"></i>&nbsp;{% trans "Go" %}</button> <option value='kbitem_{{ kbi.id }}'>{{ kbi.category.title }}: {{ kbi.title }}</option>
</p> {% endfor %}
{% csrf_token %}</form> </optgroup>
</select>
</div> <button type="submit" class="btn btn-primary btn-sm">
<div class="tab-pane fade" id="timelinetabcontents" role="tabpanel" aria-labelledby="timelinetabcontents-tab"> <i class="fas fa-arrow-circle-right"></i> {% trans "Go" %}
<div id='timeline-embed' style="width: 100%; height: 80vh"></div> </button>
<!-- 1 --> </p>
<link title="timeline-styles" rel="stylesheet" href="https://cdn.knightlab.com/libs/timeline3/latest/css/timeline.css">
<!-- 2 -->
<script src="https://cdn.knightlab.com/libs/timeline3/latest/js/timeline.js"></script>
<!-- 3 -->
<script type="text/javascript">
// The TL.Timeline constructor takes at least two arguments:
// the id of the Timeline container (no '#'), and
// the URL to your JSON data file or Google spreadsheet.
// the id must refer to an element "above" this code,
// and the element must have CSS styling to give it width and height
// optionally, a third argument with configuration options can be passed.
// See below for more about options.
var timeline_loaded = false;
$(function () {
$('#timelinetabcontents-tab').on('shown.bs.tab', function (e) {
if(!timeline_loaded){
timeline = new TL.Timeline(
'timeline-embed',
'{% url 'helpdesk:timeline_ticket_list' urlsafe_query %}'
);
timeline_loaded = true;
}
});
})
</script>
</div>
</div>
</div>
<!-- /.panel-body -->
</div>
<!-- /.panel -->
<div class="card mb-3">
<div class="card-header">
<i class="fas fa-hand-pointer"></i>
{% trans "Query Selection" %}
</div>
<div class="card-body">
<!-- start accordion -->
<div class="accordion" id="queryAccordion">
<div class="card">
<div class="card-header" id="headingOne">
<h5 class="mb-0">
<button class="btn btn-link btn-sm" type="button" data-toggle="collapse" data-target="#collapseOne" aria-expanded="true" aria-controls="collapseOne">
<i class="fas fa-filter"></i>
{% trans "Filters" %}
</button>
</h5>
</div>
<div id="collapseOne" class="collapse show" aria-labelledby="headingOne" data-parent="#queryAccordion">
<div class="card-body">
<form>
<div class="form-group float-right">
<label for="filterBuilderSelect">{% trans "Add filter" %}:</label>
<select class="custom-select custom-select-sm mb-0" aria-describedby="select-description" name="select" id="filterBuilderSelect"
onChange="onFilterChange(this.value)">
<option value="">--</option>
<option id="filterBuilderSelect-Sort" value="Sort">{% trans "Sorting" %}</option>
<option id="filterBuilderSelect-Owner" value="Owner">{% trans "Owner" %}</option>
<option id="filterBuilderSelect-Queue" value="Queue">{% trans "Queue" %}</option>
<option id="filterBuilderSelect-Status" value="Status">{% trans "Status" %}</option>
<option id="filterBuilderSelect-Keywords" value="Keywords">{% trans "Keywords" %}</option>
<option id="filterBuilderSelect-Dates" value="Dates">{% trans "Date Range" %}</option>
<option id="filterBuilderSelect-KBItems" value="KBItems">{% trans "Knowledge base items" %}</option>
</select>
{% csrf_token %}
</form>
</div>
<form method='get' action='./'>
<ul class="list-group list-group-flush">
<li id="filterBoxSort" class="filterBox{% if query_params.sorting %} filterBoxShow{% endif %} list-group-item" id="filterBoxSort">
{% include './filters/sorting.html' %}
</li>
<li class="filterBox{% if query_params.filtering.assigned_to__id__in %} filterBoxShow{% endif %} list-group-item" id=filterBoxOwner>
{% include './filters/owner.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.queue__id__in %} filterBoxShow{% endif %}" id="filterBoxQueue">
{% include './filters/queue.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.status__in %} filterBoxShow{% endif %}" id="filterBoxStatus">
{% include './filters/status.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.created__gte or query_params.filtering.created__lte %} filterBoxShow{% endif %}" id='filterBoxDates'>
{% include './filters/date.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.search_string %} filterBoxShow{% endif %}" id="filterBoxKeywords">
{% include './filters/keywords.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.kbitem__in %} filterBoxShow{% endif %}" id="filterBoxKBItems">
{% include './filters/kbitems.html' %}
</li>
</ul>
<input class="btn btn-primary btn-sm" type='submit' value='{% trans "Apply Filters" %}' />
{% if from_saved_query and saved_query.user == user %}
<p>{% blocktrans with saved_query.title as query_name %}You are currently viewing saved query <strong>"{{ query_name }}"</strong>.{% endblocktrans %} <a href='{% url 'helpdesk:delete_query' saved_query.id %}'>{% trans "Delete Saved Query" %}</a></p>
{% endif %}
{% if from_saved_query %}
<p>{% blocktrans with saved_query.id as query_id %}<a href='../reports/?saved_query={{ query_id }}'>Run a report</a> on this query to see stats and charts for the data listed below.{% endblocktrans %}</p>
{% endif %}
{% csrf_token %}
</form> </form>
</div> </div>
<div class="tab-pane fade" id="timelinetabcontents" role="tabpanel" aria-labelledby="timelinetabcontents-tab">
<div id='timeline-embed' style="width: 100%; height: 80vh"></div>
</div> </div>
</div> <!-- end card --> </div>
<div class="card">
<div class="card-header" id="headingTwo">
<h5 class="mb-0">
<button class="btn btn-link collapsed btn-sm" type="button" data-toggle="collapse" data-target="#collapseTwo" aria-expanded="false" aria-controls="collapseTwo">
<i class="fas fa-save"></i>
{% trans "Save Query" %}
</button>
</h5>
</div>
<div id="collapseTwo" class="collapse" aria-labelledby="headingTwo" data-parent="#queryAccordion">
<div class="card-body">
<form method='post' action='{% url 'helpdesk:savequery' %}'>
<input type='hidden' name='query_encoded' value='{{ urlsafe_query }}' />
<dl>
<dt><label for='id_title'>{% trans "Query Name" %}</label></dt>
<dd><input type='text' name='title' id='id_title' /></dd>
<dd class='form_help_text'>{% trans "This name appears in the drop-down list of saved queries. If you share your query, other users will see this name, so choose something clear and descriptive!" %}</dd>
<dt><label for='id_shared'>{% trans "Shared?" %}</label></dt>
<dd><input type='checkbox' name='shared' id='id_shared' /> {% trans "Yes, share this query with other users." %}</dd>
<dd class='form_help_text'>{% trans "If you share this query, it will be visible by <em>all</em> other logged-in users." %}</dd>
</dl>
<div class='buttons'>
<input class="btn btn-primary" type='submit' value='{% trans "Save Query" %}'>
</div>
{% csrf_token %}</form>
</div>
</div>
</div> <!-- end card -->
<div class="card">
<div class="card-header" id="headingThree">
<h5 class="mb-0">
<button class="btn btn-link collapsed btn-sm" type="button" data-toggle="collapse" data-target="#collapseThree" aria-expanded="false" aria-controls="collapseThree">
<i class="fas fa-clipboard-check"></i>
{% trans "Use Saved Query" %}
</button>
</h5>
</div>
<div id="collapseThree" class="collapse" aria-labelledby="headingThree" data-parent="#queryAccordion">
<div class="card-body">
<form method='get' action='{% url 'helpdesk:list' %}'>
<p><label for='id_query_selector'>{% trans "Query" %}</label><select name='saved_query' id='id_query_selector'>
{% for q in user_saved_queries %}
<option value='{{ q.id }}'>{{ q.title }}{% if q.shared %} (Shared{% ifnotequal user q.user %} by {{ q.user.get_username }}{% endifnotequal %}){% endif %}</option>
{% endfor %}
</select></p>
<input class="btn btn-primary" type='submit' value='{% trans "Run Query" %}'>
{% csrf_token %}</form>
</div>
</div>
</div> <!-- end card -->
</div> </div>
<!-- end accordion --> <!-- /.panel-body -->
</div> </div>
<!-- end card-body --> <!-- /.panel -->
</div>
<!-- end top card --> <div class="card mb-3">
<div class="card-header">
<i class="fas fa-hand-pointer"></i>
{% trans "Query Selection" %}
</div>
<div class="card-body">
<!-- start accordion -->
<div class="accordion" id="queryAccordion">
<div class="card">
<div class="card-header" id="headingOne">
<h5 class="mb-0">
<button class="btn btn-link btn-sm" type="button" data-toggle="collapse"
data-target="#collapseOne" aria-expanded="true" aria-controls="collapseOne">
<i class="fas fa-filter"></i>
{% trans "Filters" %}
</button>
</h5>
</div>
<div id="collapseOne" class="collapse show" aria-labelledby="headingOne"
data-parent="#queryAccordion">
<div class="card-body">
<form method="get">
<div class="form-group float-right">
<label for="filterBuilderSelect">{% trans "Add filter" %}:</label>
<select class="custom-select custom-select-sm mb-0"
aria-describedby="select-description" name="select" id="filterBuilderSelect"
onChange="onFilterChange(this.value)">
<option value="">--</option>
<option id="filterBuilderSelect-Sort" value="Sort"{% if query_params.sorting %} disabled{% endif %}>
{% trans "Sorting" %}
</option>
<option id="filterBuilderSelect-Owner" value="Owner"{% if query_params.filtering.assigned_to__id__in %} disabled{% endif %}>
{% trans "Owner" %}
</option>
<option id="filterBuilderSelect-Queue" value="Queue"{% if query_params.filtering.queue__id__in %} disabled{% endif %}>
{% trans "Queue" %}
</option>
<option id="filterBuilderSelect-Status" value="Status"{% if query_params.filtering.status__in %} disabled{% endif %}>
{% trans "Status" %}
</option>
<option id="filterBuilderSelect-Keywords" value="Keywords"{% if query_params.search_string %} disabled{% endif %}>
{% trans "Keywords" %}
</option>
<option id="filterBuilderSelect-Dates" value="Dates"{% if query_params.filtering.created__gte or query_params.filtering.created__lte %} disabled{% endif %}>
{% trans "Date Range" %}
</option>
<option id="filterBuilderSelect-KBItems" value="KBItems"{% if query_params.filtering.kbitem__in %} disabled{% endif %}>
{% trans "Knowledge base items" %}
</option>
</select>
</div>
</form>
</div>
<form method="get">
<ul class="list-group list-group-flush">
<li id="filterBoxSort"
class="filterBox{% if query_params.sorting %} filterBoxShow{% endif %} list-group-item"
id="filterBoxSort">
{% include 'helpdesk/filters/sorting.html' %}
</li>
<li class="filterBox{% if query_params.filtering.assigned_to__id__in %} filterBoxShow{% endif %} list-group-item"
id=filterBoxOwner>
{% include 'helpdesk/filters/owner.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.queue__id__in %} filterBoxShow{% endif %}"
id="filterBoxQueue">
{% include 'helpdesk/filters/queue.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.status__in %} filterBoxShow{% endif %}"
id="filterBoxStatus">
{% include 'helpdesk/filters/status.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.created__gte or query_params.filtering.created__lte %} filterBoxShow{% endif %}"
id='filterBoxDates'>
{% include 'helpdesk/filters/date.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.search_string %} filterBoxShow{% endif %}"
id="filterBoxKeywords">
{% include 'helpdesk/filters/keywords.html' %}
</li>
<li class="list-group-item filterBox{% if query_params.filtering.kbitem__in %} filterBoxShow{% endif %}"
id="filterBoxKBItems">
{% include 'helpdesk/filters/kbitems.html' %}
</li>
<li class="list-group-item">
<input class="btn btn-primary btn-sm" type='submit' value='{% trans "Apply Filters" %}'/>
</li>
{% if from_saved_query and saved_query.user == user %}
<li class="list-group-item">
{% blocktrans with saved_query.title as query_name %}You are currently viewing saved query <strong>"{{ query_name }}"</strong>.{% endblocktrans %}
<a href='{% url 'helpdesk:delete_query' saved_query.id %}'>
{% trans "Delete Saved Query" %}
</a>
</li>
{% endif %}
{% if from_saved_query %}
<li class="list-group-item">
{% blocktrans with saved_query.id as query_id %}<a href='../reports/?saved_query={{ query_id }}'>Run a report</a> on this query to see stats and charts for the data listed below.{% endblocktrans %}
</li>
{% endif %}
</ul>
</form>
</div>
</div> <!-- end card -->
<div class="card">
<div class="card-header" id="headingTwo">
<h5 class="mb-0">
<button class="btn btn-link collapsed btn-sm" type="button" data-toggle="collapse"
data-target="#collapseTwo" aria-expanded="false" aria-controls="collapseTwo">
<i class="fas fa-save"></i>
{% trans "Save Query" %}
</button>
</h5>
</div>
<div id="collapseTwo" class="collapse" aria-labelledby="headingTwo" data-parent="#queryAccordion">
<div class="card-body">
<form method='post' action='{% url 'helpdesk:savequery' %}'>
{% csrf_token %}
<input type='hidden' name='query_encoded' value='{{ urlsafe_query }}'/>
<dl>
<dt><label for='id_title'>{% trans "Query Name" %}</label></dt>
<dd><input type='text' name='title' id='id_title'/></dd>
<dd class='form_help_text'>{% trans "This name appears in the drop-down list of saved queries. If you share your query, other users will see this name, so choose something clear and descriptive!" %}</dd>
<dt><label for='id_shared'>{% trans "Shared?" %}</label></dt>
<dd><input type='checkbox' name='shared'
id='id_shared'/> {% trans "Yes, share this query with other users." %}
</dd>
<dd class='form_help_text'>{% trans "If you share this query, it will be visible by <em>all</em> other logged-in users." %}</dd>
</dl>
<div class='buttons'>
<input class="btn btn-primary" type='submit' value='{% trans "Save Query" %}'>
</div>
</form>
</div>
</div>
</div> <!-- end card -->
{% if user_saved_queries %}
<div class="card">
<div class="card-header" id="headingThree">
<h5 class="mb-0">
<button class="btn btn-link collapsed btn-sm" type="button" data-toggle="collapse"
data-target="#collapseThree" aria-expanded="false" aria-controls="collapseThree">
<i class="fas fa-clipboard-check"></i>
{% trans "Use Saved Query" %}
</button>
</h5>
</div>
<div id="collapseThree" class="collapse" aria-labelledby="headingThree"
data-parent="#queryAccordion">
<div class="card-body">
<form action='{% url 'helpdesk:list' %}'>
<p>
<label for='id_query_selector'>{% trans "Query" %}</label>
<select name='saved_query' id='id_query_selector'>
{% for q in user_saved_queries %}
<option value='{{ q.id }}'>
{{ q.title }}{% if q.shared %}
(Shared{% if user != q.user %} by {{ q.user.get_username }}{% endif %})
{% endif %}
</option>
{% endfor %}
</select></p>
<input class="btn btn-primary" type='submit' value='{% trans "Run Query" %}'>
</form>
</div>
</div>
</div> <!-- end card -->
{% endif %}
</div>
<!-- end accordion -->
</div>
<!-- end card-body -->
</div>
<!-- end top card -->
{% endblock %} {% endblock %}
{% block helpdesk_js %} {% block helpdesk_js %}
<script src='{% static "helpdesk/filter.js" %}'></script> <script src='{% static "helpdesk/filter.js" %}'></script>
<script> <script src="https://cdn.knightlab.com/libs/timeline3/latest/js/timeline.js"></script>
function get_url(row) <script>
{ function get_url(row) {
return "{% url 'helpdesk:view' 1234 %}".replace(/1234/, row.id.toString()); return "{% url 'helpdesk:view' 1234 %}".replace(/1234/, row.id.toString());
} }
$(document).ready(function()
{
//DataTables Initialization
let tasks_table = $('#ticketTable').DataTable({
"language": {
"emptyTable": "{% trans 'No Tickets Match Your Selection' %}"
},
"processing": true,
"serverSide": true,
"ajax": {
"url": "{% url 'helpdesk:datatables_ticket_list' urlsafe_query %}",
"type": "GET",
},
createdRow: function( row, data, dataIndex )
{
$( row ).addClass(data.row_class);
},
dom: 'ltBp',
buttons: ["colvis"],
"columns": [ $(document).ready(function () {
{"data": "id", // Ticket DataTable Initialization
"orderable": false, $('#ticketTable').DataTable({
"render": function(data, type, row, meta) language: {
{ "emptyTable": "{% trans 'No Tickets Match Your Selection' %}"
var pk = data; },
if(type === 'display'){ processing: true,
data = "<input type='checkbox' name='ticket_id' value='"+pk+"'"+ "class='ticket_multi_select' />" serverSide: true,
} ajax: {
return data "url": "{% url 'helpdesk:datatables_ticket_list' urlsafe_query %}",
} "type": "GET",
}, },
{"data": "ticket", createdRow: function (row, data, dataIndex) {
"render": function (data, type, row, meta) $(row).addClass(data.row_class);
{ },
var id = data.split(" ")[0]; dom: 'ltBp',
var name = data.split(" ")[1]; buttons: ["colvis"],
if (type === 'display') columns: [
{ {
data = '<div class="tickettitle"><a href="' + get_url(row) + '" >' + data: "id",
row.id + '. ' + orderable: false,
row.title + '</a></div>'; render: function (data, type, row, meta) {
} const pk = data;
return data if (type === 'display') {
} data = "<input type='checkbox' name='ticket_id' value='" + pk + "' class='ticket_multi_select' />"
}, }
{"data": "priority", return data
"render": function (data, type, row, meta) { }
var priority = "success"; },
if (data == 4 ) { {
priority = "warning"; data: "ticket",
} else if (data == 5) { render: function (data, type, row, meta) {
priority = "danger"; if (type === 'display') {
} data = '<div class="tickettitle"><a href="' + get_url(row) + '" >' +
return '<p class="text-'+priority+'">'+data+'</p>'; row.id + '. ' +
}, row.title + '</a></div>';
"visible": false, }
}, return data
{"data": "queue", }
"render": function(data, type, row, meta) { },
return data.title; {
}, data: "priority",
"visible": false, render: function (data, type, row, meta) {
}, let priority = "success";
{"data": "status"}, if (data === 4) {
{"data": "created"}, priority = "warning";
{"data": "due_date", "visible": false}, } else if (data === 5) {
{"data": "assigned_to", priority = "danger";
"render": function(data, type, row, meta) { }
if (data != "None") { return '<p class="text-' + priority + '">' + data + '</p>';
return data; },
} visible: false,
else { },
return ""; {
} data: "queue",
} render: function (data, type, row, meta) {
}, return data.title;
{"data": "submitter"}, },
{"data": "time_spent", "visible": false}, visible: false,
{"data": "kbitem"}, },
] {data: "status"},
}); {data: "created"},
}) {data: "due_date", "visible": false},
{
data: "assigned_to",
render: function (data, type, row, meta) {
if (data !== "None") {
return data;
}
return "";
}
},
{data: "submitter"},
{data: "time_spent", "visible": false},
{data: "kbitem"},
]
});
$(document).ready(function() {# Timeline initialization when tab is displayed #}
{ // The TL.Timeline constructor takes at least two arguments:
$("#select_all_btn").click(function() { // the id of the Timeline container (no '#'), and
$(".ticket_multi_select").prop('checked', true); // the URL to your JSON data file or Google spreadsheet.
}); // the id must refer to an element "above" this code,
$("#select_none_btn").click(function() { // and the element must have CSS styling to give it width and height
$(".ticket_multi_select").prop('checked', false); // optionally, a third argument with configuration options can be passed.
}); // See below for more about options.
$("#select_inverse_btn").click(function() { let timeline_loaded = false;
$(".ticket_multi_select").each(function() { $('#timelinetabcontents-tab').on('shown.bs.tab', function (e) {
$(this).prop('checked', !$(this).prop('checked')); if (!timeline_loaded) {
}); new TL.Timeline(
}); 'timeline-embed',
}) '{% url 'helpdesk:timeline_ticket_list' urlsafe_query %}'
);
timeline_loaded = true;
}
});
/** {# Shortcuts to select/unselect multiple tickets #}
* Disable active filtering options in filter select menu $("#select_all_btn").click(function () {
*/ $(".ticket_multi_select").prop('checked', true);
$(document).ready(function() { });
{% if query_params.sorting %} $("#select_none_btn").click(function () {
$("#filterBuilderSelect-Sort")[0].disabled = "disabled"; $(".ticket_multi_select").prop('checked', false);
{% endif %} });
{% if query_params.filtering.assigned_to__id__in %} $("#select_inverse_btn").click(function () {
$("#filterBuilderSelect-Owner")[0].disabled = "disabled"; $(".ticket_multi_select").each(function () {
{% endif %} $(this).prop('checked', !$(this).prop('checked'));
{% if query_params.filtering.queue__id__in %} });
$("#filterBuilderSelect-Queue")[0].disabled = "disabled"; });
{% endif %} })
{% if query_params.filtering.status__in %} </script>
$("#filterBuilderSelect-Status")[0].disabled = "disabled";
{% endif %}
{% if query_params.filtering.created__gte or query_params.filtering.created__lte %}
$("#filterBuilderSelect-Dates")[0].disabled = "disabled";
{% endif %}
{% if query_params.search_string %}
$("#filterBuilderSelect-Keywords")[0].disabled = "disabled";
{% endif %}
{% if query_params.filtering.kbitem__in %}
$("#filterBuilderSelect-KBItems")[0].disabled = "disabled";
{% endif %}
});
{% for f in query_params.filtering %}
{% endfor %}
</script>
{% endblock %} {% endblock %}