OCA-git-bot
5 years ago
11 changed files with 315 additions and 2 deletions
-
10report_xlsx/README.rst
-
6report_xlsx/__manifest__.py
-
1report_xlsx/models/__init__.py
-
85report_xlsx/models/header_footer.py
-
6report_xlsx/models/ir_report.py
-
15report_xlsx/report/report_xlsx.py
-
3report_xlsx/security/ir.model.access.csv
-
5report_xlsx/tests/__init__.py
-
102report_xlsx/tests/test_header_footer.py
-
64report_xlsx/views/header_footer.xml
-
20report_xlsx/views/ir_report.xml
@ -0,0 +1,85 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2019 ACSONE SA/NV (<http://acsone.eu>) |
||||
|
# License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
import io |
||||
|
import ast |
||||
|
from odoo import fields, models, api, _ |
||||
|
from odoo.exceptions import ValidationError |
||||
|
|
||||
|
|
||||
|
class ReportHeaderFooter(models.Model): |
||||
|
_name = 'report.xlsx.hf' |
||||
|
|
||||
|
name = fields.Char(string="Name", required=True) |
||||
|
hf_type = fields.Selection( |
||||
|
[('header', 'Header'), ('footer', 'Footer')], |
||||
|
string="Type", |
||||
|
required=True) |
||||
|
value = fields.Char(string="Value") |
||||
|
manual_options = fields.Char(string="Options") |
||||
|
image_left = fields.Binary(string='Image left') |
||||
|
image_left_name = fields.Char('File Name') |
||||
|
image_center = fields.Binary(string='Image center') |
||||
|
image_center_name = fields.Char('File Name') |
||||
|
image_right = fields.Binary(string='Image right') |
||||
|
image_right_name = fields.Char('File Name') |
||||
|
header_report_ids = fields.One2many( |
||||
|
'ir.actions.report.xml', |
||||
|
'header_id', |
||||
|
string="Associated report(s)") |
||||
|
footer_report_ids = fields.One2many( |
||||
|
'ir.actions.report.xml', |
||||
|
'footer_id', |
||||
|
string="Associated report(s)") |
||||
|
|
||||
|
@api.multi |
||||
|
@api.constrains('manual_options') |
||||
|
def _check_manual_options(self): |
||||
|
for rec in self: |
||||
|
if rec.manual_options: |
||||
|
options = ast.literal_eval(rec.manual_options) |
||||
|
if not isinstance(options, dict): |
||||
|
raise ValidationError( |
||||
|
_('The Header/Footer is not configured properly.\ |
||||
|
Options must be a dictionary.')) |
||||
|
|
||||
|
@api.multi |
||||
|
@api.constrains('image_left', 'image_center', 'image_right') |
||||
|
def _check_images(self): |
||||
|
for rec in self: |
||||
|
error = "" |
||||
|
if rec.image_left and ("&L&G" not in rec.value |
||||
|
and "&L&[Picture]" not in rec.value): |
||||
|
error += _('You must specify the control character &L&G or \ |
||||
|
&L&[Picture] in the "Value" when you add an "Image left".\n') |
||||
|
if rec.image_center and ("&C&G" not in rec.value |
||||
|
and "&C&[Picture]" not in rec.value): |
||||
|
error += _('You must specify the control character &C&G or \ |
||||
|
&C&[Picture] in the "Value" when you add an "Image center".\n') |
||||
|
if rec.image_right and ("&R&G" not in rec.value |
||||
|
and "&R&[Picture]" not in rec.value): |
||||
|
error += _('You must specify the control character &R&G or \ |
||||
|
&R&[Picture] in the "Value" when you add an "Image right".\n') |
||||
|
if error: |
||||
|
raise ValidationError(error) |
||||
|
|
||||
|
@api.multi |
||||
|
def get_options(self): |
||||
|
self.ensure_one() |
||||
|
options = {} |
||||
|
if self.manual_options: |
||||
|
options = ast.literal_eval(self.manual_options) |
||||
|
if self.image_left: |
||||
|
options['image_left'] = self.image_left_name |
||||
|
options['image_data_left'] = io.BytesIO( |
||||
|
self.image_left.decode('base64')) |
||||
|
if self.image_center: |
||||
|
options['image_center'] = self.image_center_name |
||||
|
options['image_data_center'] = io.BytesIO( |
||||
|
self.image_center.decode('base64')) |
||||
|
if self.image_right: |
||||
|
options['image_right'] = self.image_right_name |
||||
|
options['image_data_right'] = io.BytesIO( |
||||
|
self.image_right.decode('base64')) |
||||
|
return options |
@ -0,0 +1,3 @@ |
|||||
|
"id","name","model_id:id","group_id:id","perm_read","perm_write","perm_create","perm_unlink" |
||||
|
"access_report_xlsx_hf_all","report_xlsx_hf","model_report_xlsx_hf",,1,0,0,0 |
||||
|
"access_report_xlsx_hf_group_system","report_xlsx_hf_group_system","model_report_xlsx_hf","base.group_system",1,1,1,1 |
@ -0,0 +1,5 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2019 ACSONE SA/NV (<http://acsone.eu>) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
from . import test_header_footer |
@ -0,0 +1,102 @@ |
|||||
|
# -*- coding: utf-8 -*- |
||||
|
# Copyright 2019 ACSONE SA/NV (<http://acsone.eu>) |
||||
|
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html). |
||||
|
|
||||
|
import odoo.tests.common as common |
||||
|
from odoo.exceptions import ValidationError |
||||
|
from cStringIO import StringIO |
||||
|
from odoo.addons.report_xlsx.report.report_xlsx import ReportXlsx |
||||
|
|
||||
|
|
||||
|
class PartnerXlsx(ReportXlsx): |
||||
|
|
||||
|
def generate_xlsx_report(self, workbook, data, partners): |
||||
|
sheet = workbook.add_worksheet('sheet') |
||||
|
sheet.write(0, 0, 'test') |
||||
|
|
||||
|
|
||||
|
class TestHeaderFooter(common.TransactionCase): |
||||
|
|
||||
|
@classmethod |
||||
|
def setUpClass(cls): |
||||
|
super(TestHeaderFooter, cls).setUpClass() |
||||
|
cls.report = PartnerXlsx('report.res.partner.xlsx', |
||||
|
'res.partner') |
||||
|
|
||||
|
def setUp(self): |
||||
|
super(TestHeaderFooter, self).setUp() |
||||
|
|
||||
|
# Create Header |
||||
|
self.header_001 = self.env['report.xlsx.hf'].create({ |
||||
|
'name': 'Header 001', |
||||
|
'hf_type': 'header', |
||||
|
'value': '&LPage &P of &N &CFilename: &F &RSheetname: &A', |
||||
|
}) |
||||
|
# Create Footer |
||||
|
self.footer_001 = self.env['report.xlsx.hf'].create({ |
||||
|
'name': 'Footer 001', |
||||
|
'hf_type': 'footer', |
||||
|
'value': '&LCurrent date: &D &RCurrent time: &T', |
||||
|
}) |
||||
|
# Create Report |
||||
|
self.report_xlsx = self.env['ir.actions.report.xml'].create({ |
||||
|
'report_name': 'res.partner.xlsx', |
||||
|
'name': 'XLSX report', |
||||
|
'report_type': 'xlsx', |
||||
|
'model': 'res.partner', |
||||
|
'header_id': self.header_001.id, |
||||
|
'footer_id': self.footer_001.id, |
||||
|
}) |
||||
|
|
||||
|
def test_header_footer(self): |
||||
|
""" |
||||
|
Check that the header and footer have been added to the worksheets |
||||
|
""" |
||||
|
file_data = StringIO() |
||||
|
partner = self.env['res.partner'].browse([1]) |
||||
|
workbook = self.report.create_workbook( |
||||
|
file_data, {}, partner, self.report_xlsx) |
||||
|
header = u'&LPage &P of &N &CFilename: &F &RSheetname: &A' |
||||
|
footer = u'&LCurrent date: &D &RCurrent time: &T' |
||||
|
|
||||
|
for sheet in workbook.worksheets(): |
||||
|
self.assertEqual(header, sheet.header) |
||||
|
self.assertEqual(footer, sheet.footer) |
||||
|
|
||||
|
def test_wrong_options(self): |
||||
|
""" |
||||
|
Check that options must be a dict |
||||
|
""" |
||||
|
with self.assertRaises(ValidationError): |
||||
|
self.env['report.xlsx.hf'].create({ |
||||
|
'name': 'Header ERROR', |
||||
|
'hf_type': 'header', |
||||
|
'value': '&LPage &P of &N &CFilename: &F &RSheetname: &A', |
||||
|
'manual_options': "1234", |
||||
|
}) |
||||
|
|
||||
|
def test_image_options(self): |
||||
|
""" |
||||
|
Check that, adding image, modify the options |
||||
|
""" |
||||
|
header = self.env['report.xlsx.hf'].create({ |
||||
|
'name': 'Header IMAGE', |
||||
|
'hf_type': 'header', |
||||
|
'value': '&L&G &C&G &R&G', |
||||
|
'image_left': |
||||
|
'R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==', |
||||
|
'image_left_name': 'image_left.jpg', |
||||
|
'image_center': |
||||
|
'R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==', |
||||
|
'image_center_name': 'image_center.jpg', |
||||
|
'image_right': |
||||
|
'R0lGODlhAQABAAAAACH5BAEKAAEALAAAAAABAAEAAAICTAEAOw==', |
||||
|
'image_right_name': 'image_right.jpg', |
||||
|
}) |
||||
|
options = header.get_options() |
||||
|
self.assertEqual(options.get('image_left'), 'image_left.jpg') |
||||
|
self.assertEqual(options.get('image_center'), 'image_center.jpg') |
||||
|
self.assertEqual(options.get('image_right'), 'image_right.jpg') |
||||
|
self.assertTrue(options.get('image_data_left')) |
||||
|
self.assertTrue(options.get('image_data_center')) |
||||
|
self.assertTrue(options.get('image_data_right')) |
@ -0,0 +1,64 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
|
||||
|
<record id="report_xlsx_hf_tree" model="ir.ui.view"> |
||||
|
<field name="name">report_xlsx_hf Tree</field> |
||||
|
<field name="model">report.xlsx.hf</field> |
||||
|
<field name="type">tree</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<tree> |
||||
|
<field name="name"/> |
||||
|
<field name="hf_type"/> |
||||
|
<field name="value"/> |
||||
|
</tree> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="report_xlsx_hf_form" model="ir.ui.view"> |
||||
|
<field name="name">report_xlsx_hf Form</field> |
||||
|
<field name="model">report.xlsx.hf</field> |
||||
|
<field name="type">form</field> |
||||
|
<field name="arch" type="xml"> |
||||
|
<form> |
||||
|
<sheet> |
||||
|
<group> |
||||
|
<field name="name"/> |
||||
|
<field name="hf_type"/> |
||||
|
</group> |
||||
|
<div colspan="2"><i class="fa fa-info-circle" aria-hidden="true"/> Search in the <a href="https://xlsxwriter.readthedocs.io/page_setup.html#set_header">documentation</a> for set_header and set_footer to find the proper syntax to fill "Value" and "Options".</div> |
||||
|
<group> |
||||
|
<field name="value" placeholder="&LHello &CWorld &R!!!"/> |
||||
|
<field name="manual_options" placeholder="{'margin': 0.3}"/> |
||||
|
</group> |
||||
|
<separator string="Images"/> |
||||
|
<div><i class="fa fa-info-circle" aria-hidden="true"/> |
||||
|
If you select an image, you don't have to specify any image options (image_left, image_data_left, ...) but you need to add the control character <span style="background-color: #ebecec;">&G</span> in the value.</div> |
||||
|
<div> |
||||
|
If you add the three images, the value must contain <span style="background-color: #ebecec;">&L&G &C&G &R&G</span> or <span style="background-color: #ebecec;">&L&[Picture] &C&[Picture] &R&[Picture]</span>.</div> |
||||
|
<group> |
||||
|
<field name="image_left_name" invisible="1"/> |
||||
|
<field name="image_left" filename="image_left_name"/> |
||||
|
<field name="image_center_name" invisible="1"/> |
||||
|
<field name="image_center" filename="image_center_name"/> |
||||
|
<field name="image_right_name" invisible="1"/> |
||||
|
<field name="image_right" filename="image_right_name"/> |
||||
|
</group> |
||||
|
<separator string="Associated report(s)"/> |
||||
|
<field name="header_report_ids" attrs="{'invisible': [('hf_type', '!=', 'header')]}" nolabel="1" colspan="2"/> |
||||
|
<field name="footer_report_ids" attrs="{'invisible': [('hf_type', '!=', 'footer')]}" nolabel="1" colspan="2"/> |
||||
|
</sheet> |
||||
|
</form> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
<record id="report_xlsx_hf_action" model="ir.actions.act_window"> |
||||
|
<field name="name">XLSX Header/Footer</field> |
||||
|
<field name="res_model">report.xlsx.hf</field> |
||||
|
<field name="view_type">form</field> |
||||
|
<field name="view_mode">tree,form</field> |
||||
|
</record> |
||||
|
|
||||
|
<menuitem id="report_xlsx_hf_menu" parent="report.reporting_menuitem" |
||||
|
name="XLSX Header/Footer" action="report_xlsx_hf_action" sequence="100"/> |
||||
|
|
||||
|
</odoo> |
@ -0,0 +1,20 @@ |
|||||
|
<?xml version="1.0" encoding="utf-8"?> |
||||
|
<odoo> |
||||
|
|
||||
|
<record id="act_report_xml_view_inherit" model="ir.ui.view"> |
||||
|
<field name="name">ir.actions.report.xml (xlsx header footer)</field> |
||||
|
<field name="model">ir.actions.report.xml</field> |
||||
|
<field name="inherit_id" ref="base.act_report_xml_view"/> |
||||
|
<field name="arch" type="xml"> |
||||
|
<xpath expr="//notebook" position="inside"> |
||||
|
<page string="Header/Footer" attrs="{'invisible': [('report_type', '!=', 'xlsx')]}"> |
||||
|
<group> |
||||
|
<field name="header_id" context="{'default_hf_type': 'header'}"/> |
||||
|
<field name="footer_id" context="{'default_hf_type': 'footer'}"/> |
||||
|
</group> |
||||
|
</page> |
||||
|
</xpath> |
||||
|
</field> |
||||
|
</record> |
||||
|
|
||||
|
</odoo> |
Write
Preview
Loading…
Cancel
Save
Reference in new issue