0
0
Fork 0
mirror of https://github.com/kevinpapst/kimai2.git synced 2025-01-10 19:47:35 +00:00
kevinpapst_kimai2/templates/invoice/index.html.twig
Kevin Papst cca3fa6b8a
Improve inline stats (#3865)
* deactivate timesheet stats if no filter was chosen (otherwise a full table scan would be triggered for each listing display)
* hide duration stat if duration = 0 (timesheet, invoice, export)
2023-02-22 11:46:58 +01:00

260 lines
15 KiB
Twig

{% extends 'base.html.twig' %}
{% import "macros/widgets.html.twig" as widgets %}
{% import "macros/datatables.html.twig" as tables %}
{% set totalDuration = 0 %}
{% set totalRates = {} %}
{% set showEmpty = false %}
{% if searched %}
{% set showEmpty = true %}
{% for model in models %}
{% if showEmpty %}
{% set showEmpty = model.calculator is empty or model.calculator.entries is empty %}
{% endif %}
{% set totalDuration = totalDuration + model.calculator.timeWorked %}
{% set customerCurrency = model.customer.currency %}
{% if totalRates[customerCurrency] is not defined %}
{% set totalRates = totalRates|merge({(customerCurrency): 0.00}) %}
{% endif %}
{% set totalRates = totalRates|merge({(customerCurrency): totalRates[customerCurrency] + model.calculator.total}) %}
{% endfor %}
{% endif %}
{% set columns = {
'date': {'class': 'alwaysVisible w-min', 'orderBy': false},
'project': {'class': 'd-none d-md-table-cell', 'orderBy': false},
'description': {'class': 'd-none d-xl-table-cell', 'orderBy': false},
'user': {'class': 'd-none d-xl-table-cell w-min', 'orderBy': false},
'unit_price': {'class': 'd-none d-sm-table-cell text-end w-min', 'orderBy': false},
'amount': {'class': 'd-none d-sm-table-cell text-end w-min', 'orderBy': false},
'duration': {'class': 'text-end w-min', 'orderBy': false},
'total_rate': {'class': 'text-end alwaysVisible w-min', 'orderBy': false},
} %}
{% set tableName = 'invoice_create' %}
{% block status %}
{% from "macros/status.html.twig" import status_duration, status_money %}
{% if totalDuration > 0 %}
{{ status_duration(totalDuration|duration) }}
{% endif %}
{% for totalCurrency, totalRate in totalRates %}
{{ status_money(totalRate|money(totalCurrency)) }}
{% endfor %}
{% endblock %}
{% block main %}
{% embed '@theme/embeds/card.html.twig' %}
{% import "macros/search.html.twig" as search %}
{% form_theme form 'form/horizontal.html.twig' %}
{% block box_title %}{{ 'invoice.filter'|trans }}{% endblock %}
{% block box_before %}{{ form_start(form) }}{% endblock %}
{% block box_body %}
{{ form_errors(form) }}
{% if form.searchTerm is defined %}
{{ form_row(form.searchTerm, {'row_attr': {'class': 'mb-3 invoice_search_form_row_searchTerm'}}) }}
{% endif %}
{{ form_row(form.daterange, {'row_attr': {'class': 'mb-3 invoice_search_form_row_daterange'}}) }}
{{ form_row(form.customers, {'row_attr': {'class': 'mb-3 invoice_search_form_row_customers'}}) }}
{{ form_row(form.projects, {'row_attr': {'class': 'mb-3 invoice_search_form_row_projects'}}) }}
{% if form.activities is defined %}
{{ form_row(form.activities, {'row_attr': {'class': 'mb-3 invoice_search_form_row_activities'}}) }}
{% endif %}
{% if form.tags is defined %}
{{ form_row(form.tags, {'row_attr': {'class': 'mb-3 invoice_search_form_row_tags'}}) }}
{% endif %}
{% if form.users is defined %}
{{ form_row(form.users, {'row_attr': {'class': 'mb-3 invoice_search_form_row_users'}}) }}
{% endif %}
{% if form.teams is defined %}
{{ form_row(form.teams, {'row_attr': {'class': 'mb-3 invoice_search_form_row_teams'}}) }}
{% endif %}
{% if form.exported is defined %}
{{ form_row(form.exported, {'row_attr': {'class': 'mb-3 invoice_search_form_row_exported'}}) }}
{% endif %}
<div class="d-none">
{{ form_row(form.template, {'row_attr': {'class': 'mb-3 invoice_search_form_row_template'}}) }}
{{ form_row(form.invoiceDate, {'row_attr': {'class': 'mb-3 invoice_search_form_row_invoiceDate'}}) }}
</div>
{% endblock %}
{% block box_footer%}
{{ search.searchButton(form) }}
{% endblock %}
{% block box_after %}{{ form_end(form) }}{% endblock %}
{% endembed %}
{% if showEmpty %}
{{ widgets.nothing_found() }}
{% endif %}
{% if models|length > 0 %}
{% embed '@theme/embeds/card.html.twig' %}
{% import "macros/widgets.html.twig" as widgets %}
{% import "macros/datatables.html.twig" as tables %}
{% block box_title %}
{{ 'preview'|trans }}
{% endblock %}
{% block box_attributes %}id="invoice_customer_forms"{% endblock %}
{% block box_body_class %}p-0{% endblock %}
{% block box_body %}
<table class="table dataTable">
<thead>
<tr>
<th class="d-sm-table-cell d-none"></th>
<th>{{ 'customer'|trans }}</th>
<th class="w-min text-end d-sm-table-cell d-none">{{ 'duration'|trans }}</th>
<th class="w-min text-end d-sm-table-cell d-none">{{ 'total_rate'|trans }}</th>
<th class="w-min text-center actions alwaysVisible"></th>
</tr>
</thead>
<tbody>
{% for model in models %}
{% set isDecimal = true %}
{% set currency = model.currency %}
{% set isEmptyModel = model.calculator is empty or model.calculator.entries is empty %}
{% set customerForm = forms[loop.index0] %}
<tr>
<td class="w-min d-sm-table-cell d-none">
{% if not isEmptyModel %}
{{ widgets.card_tool_button('collapse', {'onclick': "this.parentElement.parentElement.nextElementSibling.classList.toggle('d-none');return false;"}) }}
{% endif %}
</td>
<td>
{% if is_granted('view', model.customer) %}
<a class="text-reset" href="{{ path('customer_details', {id: model.customer.id}) }}">{{ widgets.label_customer(model.customer) }}</a>
{% else %}
{{ widgets.label_customer(model.customer) }}
{% endif %}
</td>
<td class="w-min text-end d-sm-table-cell d-none">
{{ model.calculator.timeWorked|duration(isDecimal) }}
</td>
<td class="w-min text-end d-sm-table-cell d-none">
<span title="{{ 'vat'|trans }}: {{ model.calculator.tax|money(currency) }}" data-toggle="tooltip">{{ model.calculator.total|money(currency) }}</span>
</td>
<td class="w-min text-center actions alwaysVisible">
{% for attrs in [{'class': 'd-inline-flex d-md-none'}, {'icon': false, 'class': 'd-none d-md-inline-flex'}] %}
{{ widgets.action_button('print', {'url': '#', 'onclick': 'return singleInvoice(this, true)', 'title': 'preview'|trans, 'target': '_blank', 'attr': {'data-customer': model.customer.id, 'data-href': path('invoice_preview', {'customer': model.customer.id, 'token': csrf_token('invoice.preview')})}}|merge(attrs)) }}
{{ widgets.action_button('invoice', {'url': '#', 'onclick': 'return singleInvoice(this, false)', 'title': 'action.save'|trans, 'attr': {'data-customer': model.customer.id, 'data-href': path('invoice_create', {'customer': model.customer.id, 'token': csrf_token('invoice.create')})}}|merge(attrs), 'success') }}
{% endfor %}
</td>
</tr>
<tr class="d-none">
<td colspan="5" class="p-2">
{% if not isEmptyModel %}
{% form_theme customerForm 'form/horizontal.html.twig' %}
{{ form_start(customerForm) }}
{{ form_rest(customerForm) }}
{{ form_end(customerForm) }}
{% set isDecimal = true %}
{% set entries = model.calculator.entries %}
{% set currency = model.currency %}
{% set itemsAmount = entries|length %}
{{ tables.datatable_header(tableName, columns, model.query, {'boxClass': ''}) }}
{% if limit_preview %}
{% set entries = entries|slice(0, 100) %}
{% endif %}
{% for entry in entries %}
{% set amount = entry.amount %}
{% set duration = entry.duration|duration(isDecimal) %}
{% set rate = 0 %}
{% if entry.fixedRate is not null %}
{% set rate = entry.fixedRate %}
{% elseif entry.hourlyRate is not null %}
{% set rate = entry.hourlyRate %}
{% endif %}
<tr>
<td class="{{ tables.data_table_column_class(tableName, columns, 'date') }}">{{ entry.begin|date_short }}</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'project') }}">
{% if is_granted('view', entry.project) %}
<a class="text-reset" href="{{ path('project_details', {id: entry.project.id}) }}">{{ widgets.label_project(entry.project) }}</a>
{% else %}
{{ widgets.label_project(entry.project) }}
{% endif %}
{% if entry.activity is not null %}
<br>
<small>
{% if is_granted('view', entry.activity) %}
<a class="text-reset" href="{{ path('activity_details', {id: entry.activity.id}) }}">{{ widgets.label_activity(entry.activity) }}</a>
{% else %}
{{ widgets.label_activity(entry.activity) }}
{% endif %}
</small>
{% endif %}
</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'description') }} timesheet-description">
{% if entry.description is not empty %}
{{ entry.description|desc2html }}
{% endif %}
</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'user') }}">{{ widgets.label_user(entry.user) }}</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'unit_price') }}">{{ rate|money(currency) }}</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'amount') }}">{{ amount }}</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'duration') }}" data-duration="{{ entry.duration }}">{{ duration }}</td>
<td class="{{ tables.data_table_column_class(tableName, columns, 'total_rate') }}">{{ entry.rate|money(currency) }}</td>
</tr>
{% endfor %}
{% if limit_preview and itemsAmount > 100 %}
<tr class="warning">
<td colspan="8">&raquo; {{ 'preview.skipped_rows'|trans({'%rows%': (itemsAmount - 100)}) }}</td>
</tr>
{% endif %}
{{ tables.data_table_footer(entries) }}
{% endif %}
</td>
</tr>
{% endfor %}
</tbody>
</table>
<div style="display: none" id="create-token" data-value="{{ csrf_token('invoice.create') }}"></div>
<div style="display: none" id="preview-token" data-value="{{ csrf_token('invoice.preview') }}"></div>
{% endblock %}
{% endembed %}
{% endif %}
{% endblock %}
{% block javascripts %}
{{ parent() }}
{% set formId = form.vars.attr.id %}
{% set formSelector = 'form#' ~ formId %}
<script type="text/javascript">
function singleInvoice(link, preview)
{
const formName = 'customer_' + link.dataset['customer'];
const template = document.getElementById(formName + '_template').value;
const date = document.getElementById(formName + '_invoiceDate').value;
const formPlugin = kimai.getPlugin('form');
const overwrites = {'customers[]': link.dataset['customer'], 'template': template, 'invoiceDate': date};
const uri = formPlugin.convertFormDataToQueryString(document.getElementById('{{ formId }}'), overwrites);
const invoiceUrl = link.dataset['href'] + '?' + uri;
if (!preview) {
link.classList.toggle('disabled');
link.href = 'javascript: void(0)';
document.location = invoiceUrl;
return false;
}
link.href = invoiceUrl;
return true;
}
/** @type CustomEvent event */
document.addEventListener('kimai.initialized', function(event) {
KimaiReloadPageWidget.create('kimai.systemConfigUpdate', true);
{% if models|length > 0 %}
document.getElementById('{{ form.template.vars.id }}').addEventListener('change', function(event) {
document.getElementById('{{ form.vars.attr.id }}').submit();
});
{% endif %}
});
</script>
{% endblock %}