Initial commit: OCA Technical packages (595 packages)
|
After Width: | Height: | Size: 9.2 KiB |
|
|
@ -0,0 +1,538 @@
|
|||
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
|
||||
<html xmlns="http://www.w3.org/1999/xhtml" xml:lang="en" lang="en">
|
||||
<head>
|
||||
<meta http-equiv="Content-Type" content="text/html; charset=utf-8" />
|
||||
<meta name="generator" content="Docutils: https://docutils.sourceforge.io/" />
|
||||
<title>Point Of Sale - Meal Voucher</title>
|
||||
<style type="text/css">
|
||||
|
||||
/*
|
||||
:Author: David Goodger (goodger@python.org)
|
||||
:Id: $Id: html4css1.css 9511 2024-01-13 09:50:07Z milde $
|
||||
:Copyright: This stylesheet has been placed in the public domain.
|
||||
|
||||
Default cascading style sheet for the HTML output of Docutils.
|
||||
Despite the name, some widely supported CSS2 features are used.
|
||||
|
||||
See https://docutils.sourceforge.io/docs/howto/html-stylesheets.html for how to
|
||||
customize this style sheet.
|
||||
*/
|
||||
|
||||
/* used to remove borders from tables and images */
|
||||
.borderless, table.borderless td, table.borderless th {
|
||||
border: 0 }
|
||||
|
||||
table.borderless td, table.borderless th {
|
||||
/* Override padding for "table.docutils td" with "! important".
|
||||
The right padding separates the table cells. */
|
||||
padding: 0 0.5em 0 0 ! important }
|
||||
|
||||
.first {
|
||||
/* Override more specific margin styles with "! important". */
|
||||
margin-top: 0 ! important }
|
||||
|
||||
.last, .with-subtitle {
|
||||
margin-bottom: 0 ! important }
|
||||
|
||||
.hidden {
|
||||
display: none }
|
||||
|
||||
.subscript {
|
||||
vertical-align: sub;
|
||||
font-size: smaller }
|
||||
|
||||
.superscript {
|
||||
vertical-align: super;
|
||||
font-size: smaller }
|
||||
|
||||
a.toc-backref {
|
||||
text-decoration: none ;
|
||||
color: black }
|
||||
|
||||
blockquote.epigraph {
|
||||
margin: 2em 5em ; }
|
||||
|
||||
dl.docutils dd {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
object[type="image/svg+xml"], object[type="application/x-shockwave-flash"] {
|
||||
overflow: hidden;
|
||||
}
|
||||
|
||||
/* Uncomment (and remove this text!) to get bold-faced definition list terms
|
||||
dl.docutils dt {
|
||||
font-weight: bold }
|
||||
*/
|
||||
|
||||
div.abstract {
|
||||
margin: 2em 5em }
|
||||
|
||||
div.abstract p.topic-title {
|
||||
font-weight: bold ;
|
||||
text-align: center }
|
||||
|
||||
div.admonition, div.attention, div.caution, div.danger, div.error,
|
||||
div.hint, div.important, div.note, div.tip, div.warning {
|
||||
margin: 2em ;
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.admonition p.admonition-title, div.hint p.admonition-title,
|
||||
div.important p.admonition-title, div.note p.admonition-title,
|
||||
div.tip p.admonition-title {
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
div.attention p.admonition-title, div.caution p.admonition-title,
|
||||
div.danger p.admonition-title, div.error p.admonition-title,
|
||||
div.warning p.admonition-title, .code .error {
|
||||
color: red ;
|
||||
font-weight: bold ;
|
||||
font-family: sans-serif }
|
||||
|
||||
/* Uncomment (and remove this text!) to get reduced vertical space in
|
||||
compound paragraphs.
|
||||
div.compound .compound-first, div.compound .compound-middle {
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
div.compound .compound-last, div.compound .compound-middle {
|
||||
margin-top: 0.5em }
|
||||
*/
|
||||
|
||||
div.dedication {
|
||||
margin: 2em 5em ;
|
||||
text-align: center ;
|
||||
font-style: italic }
|
||||
|
||||
div.dedication p.topic-title {
|
||||
font-weight: bold ;
|
||||
font-style: normal }
|
||||
|
||||
div.figure {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
div.footer, div.header {
|
||||
clear: both;
|
||||
font-size: smaller }
|
||||
|
||||
div.line-block {
|
||||
display: block ;
|
||||
margin-top: 1em ;
|
||||
margin-bottom: 1em }
|
||||
|
||||
div.line-block div.line-block {
|
||||
margin-top: 0 ;
|
||||
margin-bottom: 0 ;
|
||||
margin-left: 1.5em }
|
||||
|
||||
div.sidebar {
|
||||
margin: 0 0 0.5em 1em ;
|
||||
border: medium outset ;
|
||||
padding: 1em ;
|
||||
background-color: #ffffee ;
|
||||
width: 40% ;
|
||||
float: right ;
|
||||
clear: right }
|
||||
|
||||
div.sidebar p.rubric {
|
||||
font-family: sans-serif ;
|
||||
font-size: medium }
|
||||
|
||||
div.system-messages {
|
||||
margin: 5em }
|
||||
|
||||
div.system-messages h1 {
|
||||
color: red }
|
||||
|
||||
div.system-message {
|
||||
border: medium outset ;
|
||||
padding: 1em }
|
||||
|
||||
div.system-message p.system-message-title {
|
||||
color: red ;
|
||||
font-weight: bold }
|
||||
|
||||
div.topic {
|
||||
margin: 2em }
|
||||
|
||||
h1.section-subtitle, h2.section-subtitle, h3.section-subtitle,
|
||||
h4.section-subtitle, h5.section-subtitle, h6.section-subtitle {
|
||||
margin-top: 0.4em }
|
||||
|
||||
h1.title {
|
||||
text-align: center }
|
||||
|
||||
h2.subtitle {
|
||||
text-align: center }
|
||||
|
||||
hr.docutils {
|
||||
width: 75% }
|
||||
|
||||
img.align-left, .figure.align-left, object.align-left, table.align-left {
|
||||
clear: left ;
|
||||
float: left ;
|
||||
margin-right: 1em }
|
||||
|
||||
img.align-right, .figure.align-right, object.align-right, table.align-right {
|
||||
clear: right ;
|
||||
float: right ;
|
||||
margin-left: 1em }
|
||||
|
||||
img.align-center, .figure.align-center, object.align-center {
|
||||
display: block;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
table.align-center {
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
.align-left {
|
||||
text-align: left }
|
||||
|
||||
.align-center {
|
||||
clear: both ;
|
||||
text-align: center }
|
||||
|
||||
.align-right {
|
||||
text-align: right }
|
||||
|
||||
/* reset inner alignment in figures */
|
||||
div.align-right {
|
||||
text-align: inherit }
|
||||
|
||||
/* div.align-center * { */
|
||||
/* text-align: left } */
|
||||
|
||||
.align-top {
|
||||
vertical-align: top }
|
||||
|
||||
.align-middle {
|
||||
vertical-align: middle }
|
||||
|
||||
.align-bottom {
|
||||
vertical-align: bottom }
|
||||
|
||||
ol.simple, ul.simple {
|
||||
margin-bottom: 1em }
|
||||
|
||||
ol.arabic {
|
||||
list-style: decimal }
|
||||
|
||||
ol.loweralpha {
|
||||
list-style: lower-alpha }
|
||||
|
||||
ol.upperalpha {
|
||||
list-style: upper-alpha }
|
||||
|
||||
ol.lowerroman {
|
||||
list-style: lower-roman }
|
||||
|
||||
ol.upperroman {
|
||||
list-style: upper-roman }
|
||||
|
||||
p.attribution {
|
||||
text-align: right ;
|
||||
margin-left: 50% }
|
||||
|
||||
p.caption {
|
||||
font-style: italic }
|
||||
|
||||
p.credits {
|
||||
font-style: italic ;
|
||||
font-size: smaller }
|
||||
|
||||
p.label {
|
||||
white-space: nowrap }
|
||||
|
||||
p.rubric {
|
||||
font-weight: bold ;
|
||||
font-size: larger ;
|
||||
color: maroon ;
|
||||
text-align: center }
|
||||
|
||||
p.sidebar-title {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold ;
|
||||
font-size: larger }
|
||||
|
||||
p.sidebar-subtitle {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
p.topic-title {
|
||||
font-weight: bold }
|
||||
|
||||
pre.address {
|
||||
margin-bottom: 0 ;
|
||||
margin-top: 0 ;
|
||||
font: inherit }
|
||||
|
||||
pre.literal-block, pre.doctest-block, pre.math, pre.code {
|
||||
margin-left: 2em ;
|
||||
margin-right: 2em }
|
||||
|
||||
pre.code .ln { color: gray; } /* line numbers */
|
||||
pre.code, code { background-color: #eeeeee }
|
||||
pre.code .comment, code .comment { color: #5C6576 }
|
||||
pre.code .keyword, code .keyword { color: #3B0D06; font-weight: bold }
|
||||
pre.code .literal.string, code .literal.string { color: #0C5404 }
|
||||
pre.code .name.builtin, code .name.builtin { color: #352B84 }
|
||||
pre.code .deleted, code .deleted { background-color: #DEB0A1}
|
||||
pre.code .inserted, code .inserted { background-color: #A3D289}
|
||||
|
||||
span.classifier {
|
||||
font-family: sans-serif ;
|
||||
font-style: oblique }
|
||||
|
||||
span.classifier-delimiter {
|
||||
font-family: sans-serif ;
|
||||
font-weight: bold }
|
||||
|
||||
span.interpreted {
|
||||
font-family: sans-serif }
|
||||
|
||||
span.option {
|
||||
white-space: nowrap }
|
||||
|
||||
span.pre {
|
||||
white-space: pre }
|
||||
|
||||
span.problematic, pre.problematic {
|
||||
color: red }
|
||||
|
||||
span.section-subtitle {
|
||||
/* font-size relative to parent (h1..h6 element) */
|
||||
font-size: 80% }
|
||||
|
||||
table.citation {
|
||||
border-left: solid 1px gray;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docinfo {
|
||||
margin: 2em 4em }
|
||||
|
||||
table.docutils {
|
||||
margin-top: 0.5em ;
|
||||
margin-bottom: 0.5em }
|
||||
|
||||
table.footnote {
|
||||
border-left: solid 1px black;
|
||||
margin-left: 1px }
|
||||
|
||||
table.docutils td, table.docutils th,
|
||||
table.docinfo td, table.docinfo th {
|
||||
padding-left: 0.5em ;
|
||||
padding-right: 0.5em ;
|
||||
vertical-align: top }
|
||||
|
||||
table.docutils th.field-name, table.docinfo th.docinfo-name {
|
||||
font-weight: bold ;
|
||||
text-align: left ;
|
||||
white-space: nowrap ;
|
||||
padding-left: 0 }
|
||||
|
||||
/* "booktabs" style (no vertical lines) */
|
||||
table.docutils.booktabs {
|
||||
border: 0px;
|
||||
border-top: 2px solid;
|
||||
border-bottom: 2px solid;
|
||||
border-collapse: collapse;
|
||||
}
|
||||
table.docutils.booktabs * {
|
||||
border: 0px;
|
||||
}
|
||||
table.docutils.booktabs th {
|
||||
border-bottom: thin solid;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
h1 tt.docutils, h2 tt.docutils, h3 tt.docutils,
|
||||
h4 tt.docutils, h5 tt.docutils, h6 tt.docutils {
|
||||
font-size: 100% }
|
||||
|
||||
ul.auto-toc {
|
||||
list-style-type: none }
|
||||
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="document" id="point-of-sale-meal-voucher">
|
||||
<h1 class="title">Point Of Sale - Meal Voucher</h1>
|
||||
|
||||
<!-- !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! This file is generated by oca-gen-addon-readme !!
|
||||
!! changes will be overwritten. !!
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!
|
||||
!! source digest: sha256:6017c21d19f693f1b5e3ca796085eb3b752979671be3000968613478f38f7e20
|
||||
!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! -->
|
||||
<p><a class="reference external image-reference" href="https://odoo-community.org/page/development-status"><img alt="Beta" src="https://img.shields.io/badge/maturity-Beta-yellow.png" /></a> <a class="reference external image-reference" href="http://www.gnu.org/licenses/agpl-3.0-standalone.html"><img alt="License: AGPL-3" src="https://img.shields.io/badge/licence-AGPL--3-blue.png" /></a> <a class="reference external image-reference" href="https://github.com/OCA/pos/tree/16.0/pos_meal_voucher"><img alt="OCA/pos" src="https://img.shields.io/badge/github-OCA%2Fpos-lightgray.png?logo=github" /></a> <a class="reference external image-reference" href="https://translation.odoo-community.org/projects/pos-16-0/pos-16-0-pos_meal_voucher"><img alt="Translate me on Weblate" src="https://img.shields.io/badge/weblate-Translate%20me-F47D42.png" /></a> <a class="reference external image-reference" href="https://runboat.odoo-community.org/builds?repo=OCA/pos&target_branch=16.0"><img alt="Try me on Runboat" src="https://img.shields.io/badge/runboat-Try%20me-875A7B.png" /></a></p>
|
||||
<p>This module allows to handle meal vouchers in the point of sale.</p>
|
||||
<p>Meal vouchers are a payment method, available in some countries (France, Belgium, Romania,…) that allows customers to buy food products in grocery stores or pay in restaurants.</p>
|
||||
<p><strong>Table of contents</strong></p>
|
||||
<div class="contents local topic" id="contents">
|
||||
<ul class="simple">
|
||||
<li><a class="reference internal" href="#configuration" id="toc-entry-1">Configuration</a><ul>
|
||||
<li><a class="reference internal" href="#products" id="toc-entry-2">Products</a></li>
|
||||
<li><a class="reference internal" href="#point-of-sale-payment-methods" id="toc-entry-3">Point of Sale Payment Methods</a></li>
|
||||
<li><a class="reference internal" href="#settings" id="toc-entry-4">Settings</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
<li><a class="reference internal" href="#usage" id="toc-entry-5">Usage</a><ul>
|
||||
<li><a class="reference internal" href="#product-screen" id="toc-entry-6">Product Screen</a></li>
|
||||
<li><a class="reference internal" href="#payment-screen" id="toc-entry-7">Payment Screen</a></li>
|
||||
<li><a class="reference internal" href="#receipt" id="toc-entry-8">Receipt</a></li>
|
||||
<li><a class="reference internal" href="#barcodes" id="toc-entry-9">Barcodes</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
<li><a class="reference internal" href="#known-issues-roadmap" id="toc-entry-10">Known issues / Roadmap</a><ul>
|
||||
<li><a class="reference internal" href="#possible-features" id="toc-entry-11">Possible Features</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
<li><a class="reference internal" href="#bug-tracker" id="toc-entry-12">Bug Tracker</a></li>
|
||||
<li><a class="reference internal" href="#credits" id="toc-entry-13">Credits</a><ul>
|
||||
<li><a class="reference internal" href="#authors" id="toc-entry-14">Authors</a></li>
|
||||
<li><a class="reference internal" href="#contributors" id="toc-entry-15">Contributors</a></li>
|
||||
<li><a class="reference internal" href="#other-credits" id="toc-entry-16">Other credits</a></li>
|
||||
<li><a class="reference internal" href="#maintainers" id="toc-entry-17">Maintainers</a></li>
|
||||
</ul>
|
||||
</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="configuration">
|
||||
<h1><a class="toc-backref" href="#toc-entry-1">Configuration</a></h1>
|
||||
<div class="section" id="products">
|
||||
<h2><a class="toc-backref" href="#toc-entry-2">Products</a></h2>
|
||||
<p>On a product form, in the “Sales” tab, the “Can be Paid for by Meal Vouchers” checkbox controls whether the product can be paid for by meal vouchers.</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/product_product_form.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/product_product_form.png" />
|
||||
</div>
|
||||
<p>Product categories can be configured to have a default value for the “Can be Paid for by Meal Vouchers” field for its products.
|
||||
The “Apply to All Products” button allows to set the value on all products of the category.</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/product_category_form.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/product_category_form.png" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="point-of-sale-payment-methods">
|
||||
<h2><a class="toc-backref" href="#toc-entry-3">Point of Sale Payment Methods</a></h2>
|
||||
<p>Point of sale payment methods (Point of Sale > Configuration > Payment Methods) have a “Meal Voucher Type” field that defines what type of meal voucher payment method they are:</p>
|
||||
<ul class="simple">
|
||||
<li>(empty): The payment method is not a meal voucher payment method.</li>
|
||||
<li><strong>Paper</strong>: The payment method will be used when scanning meal voucher barcodes.</li>
|
||||
<li><strong>Electronic</strong>: The payment method will be used for electronic meal vouchers.</li>
|
||||
</ul>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_method_form.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_method_form.png" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="settings">
|
||||
<h2><a class="toc-backref" href="#toc-entry-4">Settings</a></h2>
|
||||
<p>This module adds a “Meal Vouchers” section in the point of sale settings (Point of Sale > Configuration > Settings, or Settings > Point of Sale) with several options:</p>
|
||||
<ul class="simple">
|
||||
<li><strong>Maximum Amount</strong>: Optional maximum amount per order that can be paid by meal vouchers. Set to 0 to disable.</li>
|
||||
<li><strong>Icon on Order Lines</strong>: Whether to display an icon on point of sale order lines (on the product screen) for products that can be paid for by meal vouchers.</li>
|
||||
<li><strong>Information on Receipt</strong>: Whether to display an asterisk (*) on receipts before each product that can be paid for by meal vouchers as well as the total eligible amount.</li>
|
||||
</ul>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_settings.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_settings.png" />
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="usage">
|
||||
<h1><a class="toc-backref" href="#toc-entry-5">Usage</a></h1>
|
||||
<div class="section" id="product-screen">
|
||||
<h2><a class="toc-backref" href="#toc-entry-6">Product Screen</a></h2>
|
||||
<p>On the product screen, the products that can be paid for by meal vouchers are (optionally) identified with an icon and the total amount of those products is displayed.</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_order_screen.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_order_screen.png" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="payment-screen">
|
||||
<h2><a class="toc-backref" href="#toc-entry-7">Payment Screen</a></h2>
|
||||
<p>On the payment screen, a meal voucher summary is displayed:</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_screen_meal_vouchers.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_screen_meal_vouchers.png" />
|
||||
</div>
|
||||
<p>If the received amount is too high, a warning icon is displayed:</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_screen_warning.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/pos_payment_screen_warning.png" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="receipt">
|
||||
<h2><a class="toc-backref" href="#toc-entry-8">Receipt</a></h2>
|
||||
<p>The receipts can optionally contain information about the products that can be paid for by meal vouchers and the total amount of those products:</p>
|
||||
<div class="figure">
|
||||
<img alt="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/receipt_information.png" src="https://raw.githubusercontent.com/OCA/pos/16.0/pos_meal_voucher/static/description/receipt_information.png" />
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="barcodes">
|
||||
<h2><a class="toc-backref" href="#toc-entry-9">Barcodes</a></h2>
|
||||
<p>A new barcode rule is defined for paper meal vouchers (with 24 characters):</p>
|
||||
<p><tt class="docutils literal"><span class="pre">...........{NNNDD}........</span></tt></p>
|
||||
<p>If you scan the following barcode: <tt class="docutils literal">052566641320080017000000</tt>, a new payment line with an amount of ¤8.00 (<tt class="docutils literal">00800</tt>) will be added.</p>
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="known-issues-roadmap">
|
||||
<h1><a class="toc-backref" href="#toc-entry-10">Known issues / Roadmap</a></h1>
|
||||
<div class="section" id="possible-features">
|
||||
<h2><a class="toc-backref" href="#toc-entry-11">Possible Features</a></h2>
|
||||
<ul class="simple">
|
||||
<li>Introduce the Meal Voucher Issuer model.</li>
|
||||
<li>Deduce the issuer when scanning a meal voucher barcode.</li>
|
||||
<li>Add a report to easily compute the total deposit of meal vouchers (per issuer).</li>
|
||||
<li>Prevent to scan the same meal voucher barcode twice.</li>
|
||||
</ul>
|
||||
</div>
|
||||
</div>
|
||||
<div class="section" id="bug-tracker">
|
||||
<h1><a class="toc-backref" href="#toc-entry-12">Bug Tracker</a></h1>
|
||||
<p>Bugs are tracked on <a class="reference external" href="https://github.com/OCA/pos/issues">GitHub Issues</a>.
|
||||
In case of trouble, please check there if your issue has already been reported.
|
||||
If you spotted it first, help us to smash it by providing a detailed and welcomed
|
||||
<a class="reference external" href="https://github.com/OCA/pos/issues/new?body=module:%20pos_meal_voucher%0Aversion:%2016.0%0A%0A**Steps%20to%20reproduce**%0A-%20...%0A%0A**Current%20behavior**%0A%0A**Expected%20behavior**">feedback</a>.</p>
|
||||
<p>Do not contact contributors directly about support or help with technical issues.</p>
|
||||
</div>
|
||||
<div class="section" id="credits">
|
||||
<h1><a class="toc-backref" href="#toc-entry-13">Credits</a></h1>
|
||||
<div class="section" id="authors">
|
||||
<h2><a class="toc-backref" href="#toc-entry-14">Authors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>GRAP</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="contributors">
|
||||
<h2><a class="toc-backref" href="#toc-entry-15">Contributors</a></h2>
|
||||
<ul class="simple">
|
||||
<li>Sylvain LE GAL <<a class="reference external" href="https://twitter.com/legalsylvain">https://twitter.com/legalsylvain</a>></li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="other-credits">
|
||||
<h2><a class="toc-backref" href="#toc-entry-16">Other credits</a></h2>
|
||||
<p>The development of this module has been financially supported by:</p>
|
||||
<ul class="simple">
|
||||
<li>Vracoop (<a class="reference external" href="https://portail.vracoop.fr/">https://portail.vracoop.fr/</a>)</li>
|
||||
<li>Demain Supermarché (<a class="reference external" href="http://www.demainsupermarche.org/">http://www.demainsupermarche.org/</a>)</li>
|
||||
<li>Le Drive tout nu (<a class="reference external" href="https://ledrivetoutnu.com/">https://ledrivetoutnu.com/</a>)</li>
|
||||
</ul>
|
||||
</div>
|
||||
<div class="section" id="maintainers">
|
||||
<h2><a class="toc-backref" href="#toc-entry-17">Maintainers</a></h2>
|
||||
<p>This module is maintained by the OCA.</p>
|
||||
<a class="reference external image-reference" href="https://odoo-community.org">
|
||||
<img alt="Odoo Community Association" src="https://odoo-community.org/logo.png" />
|
||||
</a>
|
||||
<p>OCA, or the Odoo Community Association, is a nonprofit organization whose
|
||||
mission is to support the collaborative development of Odoo features and
|
||||
promote its widespread use.</p>
|
||||
<p>Current <a class="reference external" href="https://odoo-community.org/page/maintainer-role">maintainer</a>:</p>
|
||||
<p><a class="reference external image-reference" href="https://github.com/legalsylvain"><img alt="legalsylvain" src="https://github.com/legalsylvain.png?size=40px" /></a></p>
|
||||
<p>This module is part of the <a class="reference external" href="https://github.com/OCA/pos/tree/16.0/pos_meal_voucher">OCA/pos</a> project on GitHub.</p>
|
||||
<p>You are welcome to contribute. To learn how please visit <a class="reference external" href="https://odoo-community.org/page/Contribute">https://odoo-community.org/page/Contribute</a>.</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
||||
|
After Width: | Height: | Size: 25 KiB |
|
After Width: | Height: | Size: 66 KiB |
|
After Width: | Height: | Size: 15 KiB |
|
After Width: | Height: | Size: 5.1 KiB |
|
After Width: | Height: | Size: 22 KiB |
|
After Width: | Height: | Size: 17 KiB |
|
After Width: | Height: | Size: 54 KiB |
|
After Width: | Height: | Size: 8.8 KiB |
|
After Width: | Height: | Size: 33 KiB |
|
After Width: | Height: | Size: 14 KiB |
|
After Width: | Height: | Size: 31 KiB |
|
After Width: | Height: | Size: 44 KiB |
|
|
@ -0,0 +1,42 @@
|
|||
/** @odoo-module **/
|
||||
// SPDX-FileCopyrightText: 2025 Coop IT Easy SC
|
||||
//
|
||||
// SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
|
||||
import {Component} from "point_of_sale.Registries";
|
||||
import PaymentScreen from "point_of_sale.PaymentScreen";
|
||||
import {useBarcodeReader} from "point_of_sale.custom_hooks";
|
||||
|
||||
const MealVoucherPaymentScreen = (OriginalPaymentScreen) =>
|
||||
class extends OriginalPaymentScreen {
|
||||
setup() {
|
||||
super.setup();
|
||||
if (this.env.pos.paper_meal_voucher_payment_method !== null) {
|
||||
useBarcodeReader({
|
||||
meal_voucher_payment: this._barcodeMealVoucherAction,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
async _barcodeMealVoucherAction(code) {
|
||||
return this.currentOrder.handle_meal_voucher_barcode(code);
|
||||
}
|
||||
|
||||
get hasMealVoucherPaymentMethod() {
|
||||
return this.env.pos.config.has_meal_voucher_payment_method;
|
||||
}
|
||||
|
||||
get mealVoucherEligibleAmount() {
|
||||
return this.currentOrder.get_total_meal_voucher_eligible();
|
||||
}
|
||||
|
||||
get mealVoucherReceivedAmount() {
|
||||
return this.currentOrder.get_total_meal_voucher_received();
|
||||
}
|
||||
|
||||
get maxMealVoucherAmount() {
|
||||
return this.env.pos.config.max_meal_voucher_amount;
|
||||
}
|
||||
};
|
||||
|
||||
Component.extend(PaymentScreen, MealVoucherPaymentScreen);
|
||||
|
|
@ -0,0 +1,16 @@
|
|||
/** @odoo-module **/
|
||||
// SPDX-FileCopyrightText: 2025 Coop IT Easy SC
|
||||
//
|
||||
// SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
|
||||
import {Component} from "point_of_sale.Registries";
|
||||
import PaymentScreenPaymentLines from "point_of_sale.PaymentScreenPaymentLines";
|
||||
|
||||
const MealVoucherPaymentScreenPaymentLines = (OriginalPaymentScreenPaymentLines) =>
|
||||
class extends OriginalPaymentScreenPaymentLines {
|
||||
isMealVoucher(line) {
|
||||
return line.payment_method.meal_voucher_type !== false;
|
||||
}
|
||||
};
|
||||
|
||||
Component.extend(PaymentScreenPaymentLines, MealVoucherPaymentScreenPaymentLines);
|
||||
|
|
@ -0,0 +1,19 @@
|
|||
/** @odoo-module **/
|
||||
// SPDX-FileCopyrightText: 2025 Coop IT Easy SC
|
||||
//
|
||||
// SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
|
||||
import {Component} from "point_of_sale.Registries";
|
||||
import Orderline from "point_of_sale.Orderline";
|
||||
|
||||
const MealVoucherOrderline = (OriginalOrderline) =>
|
||||
class extends OriginalOrderline {
|
||||
get displayMealVoucherIcon() {
|
||||
return (
|
||||
this.env.pos.config.enable_meal_voucher_order_lines_icon &&
|
||||
this.props.line.get_product().meal_voucher_ok
|
||||
);
|
||||
}
|
||||
};
|
||||
|
||||
Component.extend(Orderline, MealVoucherOrderline);
|
||||
|
|
@ -0,0 +1,28 @@
|
|||
/** @odoo-module **/
|
||||
// SPDX-FileCopyrightText: 2025 Coop IT Easy SC
|
||||
//
|
||||
// SPDX-License-Identifier: AGPL-3.0-or-later
|
||||
|
||||
import {Component} from "point_of_sale.Registries";
|
||||
import ProductScreen from "point_of_sale.ProductScreen";
|
||||
import {useBarcodeReader} from "point_of_sale.custom_hooks";
|
||||
|
||||
const MealVoucherProductScreen = (OriginalProductScreen) =>
|
||||
class extends OriginalProductScreen {
|
||||
setup() {
|
||||
super.setup();
|
||||
if (this.env.pos.paper_meal_voucher_payment_method !== null) {
|
||||
useBarcodeReader({
|
||||
meal_voucher_payment: this._barcodeMealVoucherAction,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
async _barcodeMealVoucherAction(code) {
|
||||
// Display the payment screen, if it is not the current one.
|
||||
this.showScreen("PaymentScreen");
|
||||
return this.currentOrder.handle_meal_voucher_barcode(code);
|
||||
}
|
||||
};
|
||||
|
||||
Component.extend(ProductScreen, MealVoucherProductScreen);
|
||||
|
|
@ -0,0 +1,168 @@
|
|||
/** @odoo-module **/
|
||||
// Copyright (C) 2020 - Today: GRAP (http://www.grap.coop)
|
||||
// @author: Sylvain LE GAL (https://twitter.com/legalsylvain)
|
||||
// License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html).
|
||||
|
||||
import {Order, Orderline, Payment, PosGlobalState} from "point_of_sale.models";
|
||||
import {Gui} from "point_of_sale.Gui";
|
||||
import {Model} from "point_of_sale.Registries";
|
||||
import {_t} from "web.core";
|
||||
import {round_precision as round_pr} from "web.utils";
|
||||
|
||||
const MealVoucherOrder = (OriginalOrder) =>
|
||||
class extends OriginalOrder {
|
||||
get_total_meal_voucher_eligible() {
|
||||
return round_pr(
|
||||
this.orderlines.reduce(function (sum, orderLine) {
|
||||
if (orderLine.product.meal_voucher_ok) {
|
||||
return sum + orderLine.get_price_with_tax();
|
||||
}
|
||||
return sum;
|
||||
}, 0),
|
||||
this.pos.currency.rounding
|
||||
);
|
||||
}
|
||||
|
||||
get_total_meal_voucher_non_eligible() {
|
||||
return round_pr(
|
||||
this.orderlines.reduce(function (sum, orderLine) {
|
||||
if (!orderLine.product.meal_voucher_ok) {
|
||||
return sum + orderLine.get_price_with_tax();
|
||||
}
|
||||
return sum;
|
||||
}, 0),
|
||||
this.pos.currency.rounding
|
||||
);
|
||||
}
|
||||
|
||||
get_total_meal_voucher_received() {
|
||||
return round_pr(
|
||||
this.paymentlines.reduce(function (sum, paymentLine) {
|
||||
if (paymentLine.is_meal_voucher()) {
|
||||
return sum + paymentLine.get_amount();
|
||||
}
|
||||
return sum;
|
||||
}, 0),
|
||||
this.pos.currency.rounding
|
||||
);
|
||||
}
|
||||
|
||||
async _meal_voucher_is_valid(code) {
|
||||
for (const payment_line of this.paymentlines) {
|
||||
if (payment_line.payment_note === code) {
|
||||
await Gui.showPopup("ErrorPopup", {
|
||||
title: _t("Invalid Meal Voucher"),
|
||||
body: _.str.sprintf(
|
||||
_t(
|
||||
'The paper meal voucher with code "%s" has already been scanned.'
|
||||
),
|
||||
code
|
||||
),
|
||||
});
|
||||
return false;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
async handle_meal_voucher_barcode(code) {
|
||||
if (!(await this._meal_voucher_is_valid(code.code))) {
|
||||
return;
|
||||
}
|
||||
// Add new payment line with the amount found in the barcode.
|
||||
const payment_line = this.add_paymentline(
|
||||
this.pos.paper_meal_voucher_payment_method
|
||||
);
|
||||
payment_line.set_amount(code.value);
|
||||
payment_line.payment_note = code.code;
|
||||
}
|
||||
};
|
||||
|
||||
Model.extend(Order, MealVoucherOrder);
|
||||
|
||||
const RECEIPT_ORDER_LINE_PREFIX = "(*) ";
|
||||
|
||||
const MealVoucherOrderline = (OriginalOrderline) =>
|
||||
class extends OriginalOrderline {
|
||||
generate_wrapped_product_name() {
|
||||
if (
|
||||
!this.get_product().meal_voucher_ok ||
|
||||
!this.pos.config.enable_meal_voucher_receipt_info
|
||||
) {
|
||||
return super.generate_wrapped_product_name(...arguments);
|
||||
}
|
||||
// Temporarily change the product name to add a prefix on the
|
||||
// receipt.
|
||||
//
|
||||
// .generate_wrapped_product_name() calls
|
||||
// .get_full_product_name(), and it has a different behavior
|
||||
// depending on whether this.full_product_name is set or not. both
|
||||
// behaviors must be handled, because one is used when generating
|
||||
// a receipt during an order, while the order is used when
|
||||
// retrieving a receipt from a previous order.
|
||||
// .get_full_product_name() cannot be overridden because its
|
||||
// result is also used for display in the product screen and
|
||||
// its result is stored, which would result in the prefix being
|
||||
// added each time the pos interface is reloaded.
|
||||
const originalFullProductName = this.full_product_name;
|
||||
const originalDisplayName = this.product.display_name;
|
||||
if (originalFullProductName) {
|
||||
this.full_product_name =
|
||||
RECEIPT_ORDER_LINE_PREFIX + originalFullProductName;
|
||||
} else {
|
||||
this.product.display_name =
|
||||
RECEIPT_ORDER_LINE_PREFIX + originalDisplayName;
|
||||
}
|
||||
const res = super.generate_wrapped_product_name(...arguments);
|
||||
if (originalFullProductName) {
|
||||
this.full_product_name = originalFullProductName;
|
||||
} else {
|
||||
this.product.display_name = originalDisplayName;
|
||||
}
|
||||
return res;
|
||||
}
|
||||
};
|
||||
|
||||
Model.extend(Orderline, MealVoucherOrderline);
|
||||
|
||||
const MealVoucherPayment = (OriginalPayment) =>
|
||||
class extends OriginalPayment {
|
||||
initialize() {
|
||||
super.initialize(...arguments);
|
||||
this.payment_note = null;
|
||||
}
|
||||
|
||||
init_from_JSON(json) {
|
||||
super.init_from_JSON(...arguments);
|
||||
this.payment_note = json.payment_note;
|
||||
}
|
||||
|
||||
export_as_JSON() {
|
||||
const res = super.export_as_JSON(...arguments);
|
||||
res.payment_note = this.payment_note;
|
||||
return res;
|
||||
}
|
||||
|
||||
is_meal_voucher() {
|
||||
return this.payment_method.meal_voucher_type !== false;
|
||||
}
|
||||
};
|
||||
|
||||
Model.extend(Payment, MealVoucherPayment);
|
||||
|
||||
const MealVoucherPosGlobalState = (OriginalPosGlobalState) =>
|
||||
class extends OriginalPosGlobalState {
|
||||
async load_server_data() {
|
||||
await super.load_server_data(...arguments);
|
||||
this.paper_meal_voucher_payment_method = null;
|
||||
for (const payment_method_id of this.config.payment_method_ids) {
|
||||
const payment_method = this.payment_methods_by_id[payment_method_id];
|
||||
if (payment_method.meal_voucher_type === "paper") {
|
||||
this.paper_meal_voucher_payment_method = payment_method;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
};
|
||||
|
||||
Model.extend(PosGlobalState, MealVoucherPosGlobalState);
|
||||
|
|
@ -0,0 +1,56 @@
|
|||
.pos .paymentline .is-meal-voucher {
|
||||
text-align: center;
|
||||
padding: map-get($spacers, 3);
|
||||
}
|
||||
|
||||
.pos .paymentline .is-meal-voucher .fa-cutlery {
|
||||
border-radius: 8px;
|
||||
border: 1px solid rgba($gray-700, 0.5);
|
||||
width: 24px;
|
||||
line-height: 24px;
|
||||
}
|
||||
|
||||
.payment-screen div.meal-voucher-summary .fa-warning {
|
||||
color: red;
|
||||
}
|
||||
|
||||
.pos .meal-voucher-summary {
|
||||
// same as .pos .paymentlines
|
||||
display: flex;
|
||||
flex-direction: column-reverse;
|
||||
border-spacing: 0px 10px;
|
||||
border-collapse: inherit;
|
||||
}
|
||||
|
||||
.pos .meal-voucher-summary-container {
|
||||
padding: map-get($spacers, 2) 0;
|
||||
}
|
||||
|
||||
.pos .meal-voucher-summary-line {
|
||||
// same as .pos .paymentline, with some changes
|
||||
display: flex;
|
||||
flex-grow: 1;
|
||||
flex-basis: 100%;
|
||||
background: #fff;
|
||||
font-size: 16px;
|
||||
border-top-width: 0px;
|
||||
padding: map-get($spacers, 2) 0;
|
||||
}
|
||||
|
||||
.meal-voucher-summary-line .meal-voucher-summary-line-name {
|
||||
flex-grow: 1;
|
||||
padding: 0 map-get($spacers, 3);
|
||||
}
|
||||
|
||||
.meal-voucher-summary-line .meal-voucher-summary-line-amount {
|
||||
padding: 0 map-get($spacers, 3);
|
||||
}
|
||||
|
||||
.meal-voucher-summary-line .meal-voucher-summary-line-warning {
|
||||
font-size: 20px;
|
||||
margin-bottom: -(map-get($spacers, 2));
|
||||
}
|
||||
|
||||
.pos-receipt-meal-voucher-totals {
|
||||
font-size: 75%;
|
||||
}
|
||||
|
|
@ -0,0 +1,64 @@
|
|||
<?xml version="1.0" encoding="UTF-8" ?>
|
||||
<templates id="template" xml:space="preserve">
|
||||
|
||||
<t
|
||||
t-name="MealVoucherPaymentScreen"
|
||||
t-inherit="point_of_sale.PaymentScreen"
|
||||
t-inherit-mode="extension"
|
||||
owl="1"
|
||||
>
|
||||
<xpath expr="//div[hasclass('paymentmethods')]" position="after">
|
||||
<div t-if="hasMealVoucherPaymentMethod" class="meal-voucher-summary">
|
||||
<p class="title-category">Meal Vouchers</p>
|
||||
<div class="meal-voucher-summary-container">
|
||||
<div class="meal-voucher-summary-line">
|
||||
<div class="meal-voucher-summary-line-name">Total Eligible</div>
|
||||
<div
|
||||
t-if="mealVoucherReceivedAmount > mealVoucherEligibleAmount"
|
||||
class="meal-voucher-summary-line-warning"
|
||||
>
|
||||
<i
|
||||
id="meal-voucher-eligible-warning"
|
||||
class="fa fa-warning"
|
||||
/>
|
||||
</div>
|
||||
<div class="meal-voucher-summary-line-amount">
|
||||
<span
|
||||
id="meal-voucher-eligible-amount"
|
||||
t-out="env.pos.format_currency(mealVoucherEligibleAmount)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div
|
||||
t-if="maxMealVoucherAmount !== 0"
|
||||
class="meal-voucher-summary-line"
|
||||
>
|
||||
<div class="meal-voucher-summary-line-name">Max Amount</div>
|
||||
<div
|
||||
t-if="mealVoucherReceivedAmount > maxMealVoucherAmount"
|
||||
class="meal-voucher-summary-line-warning"
|
||||
>
|
||||
<i id="meal-voucher-max-warning" class="fa fa-warning" />
|
||||
</div>
|
||||
<div class="meal-voucher-summary-line-amount">
|
||||
<span
|
||||
id="meal-voucher-max-amount"
|
||||
t-out="env.pos.format_currency(maxMealVoucherAmount)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
<div class="meal-voucher-summary-line">
|
||||
<div class="meal-voucher-summary-line-name">Total Received</div>
|
||||
<div class="meal-voucher-summary-line-amount">
|
||||
<span
|
||||
id="meal-voucher-received-amount"
|
||||
t-out="env.pos.format_currency(mealVoucherReceivedAmount)"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</xpath>
|
||||
</t>
|
||||
|
||||
</templates>
|
||||
|
|
@ -0,0 +1,28 @@
|
|||
<?xml version="1.0" encoding="UTF-8" ?>
|
||||
<templates id="template" xml:space="preserve">
|
||||
|
||||
<t
|
||||
t-name="MealVoucherPaymentScreenPaymentLines"
|
||||
t-inherit="point_of_sale.PaymentScreenPaymentLines"
|
||||
t-inherit-mode="extension"
|
||||
owl="1"
|
||||
>
|
||||
<xpath
|
||||
expr="//t[@t-if='line.selected']/div/div[hasclass('payment-amount')]"
|
||||
position="before"
|
||||
>
|
||||
<div t-if="isMealVoucher(line)" class="is-meal-voucher">
|
||||
<i class="fa fa-cutlery" />
|
||||
</div>
|
||||
</xpath>
|
||||
<xpath
|
||||
expr="//t[@t-foreach='props.paymentLines']/t[@t-else='']/div/div[hasclass('payment-amount')]"
|
||||
position="before"
|
||||
>
|
||||
<div t-if="isMealVoucher(line)" class="is-meal-voucher">
|
||||
<i class="fa fa-cutlery" />
|
||||
</div>
|
||||
</xpath>
|
||||
</t>
|
||||
|
||||
</templates>
|
||||
|
|
@ -0,0 +1,26 @@
|
|||
<?xml version="1.0" encoding="UTF-8" ?>
|
||||
<templates id="template" xml:space="preserve">
|
||||
|
||||
<t
|
||||
t-name="MealVoucherOrderSummary"
|
||||
t-inherit="point_of_sale.OrderSummary"
|
||||
t-inherit-mode="extension"
|
||||
owl="1"
|
||||
>
|
||||
<xpath expr="//div[hasclass('summary')][1]" position="after">
|
||||
<t t-if="env.pos.config.has_meal_voucher_payment_method">
|
||||
<div class="summary clearfix">
|
||||
<div class="line">
|
||||
<div class="subentry meal-voucher">
|
||||
Meal Voucher: <span
|
||||
class="value"
|
||||
t-out="env.pos.format_currency(props.order.get_total_meal_voucher_eligible())"
|
||||
/>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</t>
|
||||
</xpath>
|
||||
</t>
|
||||
|
||||
</templates>
|
||||
|
|
@ -0,0 +1,19 @@
|
|||
<?xml version="1.0" encoding="UTF-8" ?>
|
||||
<templates id="template" xml:space="preserve">
|
||||
|
||||
<t
|
||||
t-name="MealVoucherOrderline"
|
||||
t-inherit="point_of_sale.Orderline"
|
||||
t-inherit-mode="extension"
|
||||
owl="1"
|
||||
>
|
||||
<!--
|
||||
the space after the <i> element is needed to separate the icon
|
||||
from the product name.
|
||||
-->
|
||||
<xpath expr="//span[hasclass('product-name')]/t[1]" position="before">
|
||||
<t t-if="displayMealVoucherIcon"><i class="fa fa-cutlery" /> </t>
|
||||
</xpath>
|
||||
</t>
|
||||
|
||||
</templates>
|
||||
|
|
@ -0,0 +1,32 @@
|
|||
<?xml version="1.0" encoding="UTF-8" ?>
|
||||
<templates id="template" xml:space="preserve">
|
||||
|
||||
<t
|
||||
t-name="MealVoucherOrderReceipt"
|
||||
t-inherit="point_of_sale.OrderReceipt"
|
||||
t-inherit-mode="extension"
|
||||
owl="1"
|
||||
>
|
||||
<xpath expr="//t/div[hasclass('pos-receipt-amount')]/.." position="after">
|
||||
<t t-if="env.pos.config.enable_meal_voucher_receipt_info">
|
||||
<br />
|
||||
<div class="pos-receipt-meal-voucher-totals">
|
||||
Products marked with an asterisk (*) can be paid for by meal vouchers.
|
||||
<br />
|
||||
Eligible Total:
|
||||
<span
|
||||
t-out="env.pos.format_currency(props.order.get_total_meal_voucher_eligible())"
|
||||
class="pos-receipt-right-align"
|
||||
/>
|
||||
<br />
|
||||
Non-Eligible Total:
|
||||
<span
|
||||
t-out="env.pos.format_currency(props.order.get_total_meal_voucher_non_eligible())"
|
||||
class="pos-receipt-right-align"
|
||||
/>
|
||||
</div>
|
||||
</t>
|
||||
</xpath>
|
||||
</t>
|
||||
|
||||
</templates>
|
||||