[14.0][MIG] excel_import_export, excel_import_export_demo
parent
354890e57a
commit
0421635d7a
|
@ -4,7 +4,7 @@
|
|||
{
|
||||
"name": "Excel Import/Export/Report",
|
||||
"summary": "Base module for developing Excel import/export/report",
|
||||
"version": "13.0.1.0.0",
|
||||
"version": "14.0.1.0.0",
|
||||
"author": "Ecosoft,Odoo Community Association (OCA)",
|
||||
"license": "AGPL-3",
|
||||
"website": "https://github.com/OCA/server-tools",
|
||||
|
|
|
@ -3,7 +3,6 @@
|
|||
|
||||
import base64
|
||||
import json
|
||||
import time
|
||||
|
||||
from odoo.http import content_disposition, request, route
|
||||
from odoo.tools.safe_eval import safe_eval
|
||||
|
@ -36,9 +35,7 @@ class ReportController(report.ReportController):
|
|||
if report.print_report_name and not len(docids) > 1:
|
||||
obj = request.env[report.model].browse(docids[0])
|
||||
file_ext = report_name.split(".")[-1:].pop()
|
||||
report_name = safe_eval(
|
||||
report.print_report_name, {"object": obj, "time": time}
|
||||
)
|
||||
report_name = safe_eval(report.print_report_name, {"object": obj})
|
||||
report_name = "{}.{}".format(report_name, file_ext)
|
||||
excelhttpheaders = [
|
||||
(
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
# Copyright 2019 Ecosoft Co., Ltd (http://ecosoft.co.th/)
|
||||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html)
|
||||
|
||||
import base64
|
||||
import csv
|
||||
import itertools
|
||||
import logging
|
||||
|
@ -226,8 +225,7 @@ def str_to_number(input_val):
|
|||
|
||||
|
||||
def csv_from_excel(excel_content, delimiter, quote):
|
||||
decoded_data = base64.decodestring(excel_content)
|
||||
wb = xlrd.open_workbook(file_contents=decoded_data)
|
||||
wb = xlrd.open_workbook(file_contents=excel_content)
|
||||
sh = wb.sheet_by_index(0)
|
||||
content = StringIO()
|
||||
quoting = csv.QUOTE_ALL
|
||||
|
@ -250,7 +248,7 @@ def csv_from_excel(excel_content, delimiter, quote):
|
|||
row.append(x)
|
||||
wr.writerow(row)
|
||||
content.seek(0) # Set index to 0, and start reading
|
||||
out_file = base64.b64encode(content.getvalue().encode("utf-8"))
|
||||
out_file = content.getvalue().encode("utf-8")
|
||||
return out_file
|
||||
|
||||
|
||||
|
|
|
@ -8,7 +8,9 @@ from odoo.exceptions import UserError
|
|||
class ReportAction(models.Model):
|
||||
_inherit = "ir.actions.report"
|
||||
|
||||
report_type = fields.Selection(selection_add=[("excel", "Excel")])
|
||||
report_type = fields.Selection(
|
||||
selection_add=[("excel", "Excel")], ondelete={"excel": "cascade"}
|
||||
)
|
||||
|
||||
@api.model
|
||||
def render_excel(self, docids, data):
|
||||
|
|
|
@ -4,7 +4,7 @@
|
|||
import base64
|
||||
import logging
|
||||
import os
|
||||
import time
|
||||
import zipfile
|
||||
from datetime import date, datetime as dt
|
||||
from io import BytesIO
|
||||
|
||||
|
@ -31,7 +31,6 @@ class XLSXExport(models.AbstractModel):
|
|||
def get_eval_context(self, model, record, value):
|
||||
eval_context = {
|
||||
"float_compare": float_compare,
|
||||
"time": time,
|
||||
"datetime": dt,
|
||||
"date": date,
|
||||
"value": value,
|
||||
|
@ -225,7 +224,7 @@ class XLSXExport(models.AbstractModel):
|
|||
return
|
||||
|
||||
@api.model
|
||||
def export_xlsx(self, template, res_model, res_id):
|
||||
def export_xlsx(self, template, res_model, res_ids):
|
||||
if template.res_model != res_model:
|
||||
raise ValidationError(_("Template's model mismatch"))
|
||||
data_dict = co.literal_eval(template.instruction.strip())
|
||||
|
@ -241,33 +240,50 @@ class XLSXExport(models.AbstractModel):
|
|||
ptemp = ConfParam.get_param("path_temp_file") or "/tmp"
|
||||
stamp = dt.utcnow().strftime("%H%M%S%f")[:-3]
|
||||
ftemp = "{}/temp{}.xlsx".format(ptemp, stamp)
|
||||
f = open(ftemp, "wb")
|
||||
f.write(decoded_data)
|
||||
f.seek(0)
|
||||
f.close()
|
||||
# Workbook created, temp file removed
|
||||
wb = load_workbook(ftemp)
|
||||
os.remove(ftemp)
|
||||
# Start working with workbook
|
||||
record = res_model and self.env[res_model].browse(res_id) or False
|
||||
self._fill_workbook_data(wb, record, export_dict)
|
||||
# Return file as .xlsx
|
||||
content = BytesIO()
|
||||
wb.save(content)
|
||||
content.seek(0) # Set index to 0, and start reading
|
||||
out_file = base64.encodebytes(content.read())
|
||||
if record and "name" in record and record.name:
|
||||
out_name = record.name.replace(" ", "").replace("/", "")
|
||||
records = res_model and self.env[res_model].browse(res_ids) or False
|
||||
outputs = []
|
||||
for record in records:
|
||||
f = open(ftemp, "wb")
|
||||
f.write(decoded_data)
|
||||
f.seek(0)
|
||||
f.close()
|
||||
# Workbook created, temp file removed
|
||||
wb = load_workbook(ftemp)
|
||||
os.remove(ftemp)
|
||||
self._fill_workbook_data(wb, record, export_dict)
|
||||
# Return file as .xlsx
|
||||
content = BytesIO()
|
||||
wb.save(content)
|
||||
content.seek(0) # Set index to 0, and start reading
|
||||
out_file = content.read()
|
||||
if record and "name" in record and record.name:
|
||||
out_name = record.name.replace(" ", "").replace("/", "")
|
||||
else:
|
||||
fname = out_name.replace(" ", "").replace("/", "")
|
||||
ts = fields.Datetime.context_timestamp(self, dt.now())
|
||||
out_name = "{}_{}".format(fname, ts.strftime("%Y%m%d_%H%M%S"))
|
||||
if not out_name or len(out_name) == 0:
|
||||
out_name = "noname"
|
||||
out_ext = "xlsx"
|
||||
# CSV (convert only on 1st sheet)
|
||||
if template.to_csv:
|
||||
delimiter = template.csv_delimiter
|
||||
out_file = co.csv_from_excel(out_file, delimiter, template.csv_quote)
|
||||
out_ext = template.csv_extension
|
||||
outputs.append((out_file, "{}.{}".format(out_name, out_ext)))
|
||||
# If outputs > 1 files, zip it
|
||||
if len(outputs) > 1:
|
||||
zip_buffer = BytesIO()
|
||||
with zipfile.ZipFile(
|
||||
zip_buffer, "a", zipfile.ZIP_DEFLATED, False
|
||||
) as zip_file:
|
||||
for data, file_name in outputs:
|
||||
zip_file.writestr(file_name, data)
|
||||
zip_buffer.seek(0)
|
||||
out_file = base64.encodebytes(zip_buffer.read())
|
||||
out_name = "files.zip"
|
||||
return (out_file, out_name)
|
||||
else:
|
||||
fname = out_name.replace(" ", "").replace("/", "")
|
||||
ts = fields.Datetime.context_timestamp(self, dt.now())
|
||||
out_name = "{}_{}".format(fname, ts.strftime("%Y%m%d_%H%M%S"))
|
||||
if not out_name or len(out_name) == 0:
|
||||
out_name = "noname"
|
||||
out_ext = "xlsx"
|
||||
# CSV (convert only on 1st sheet)
|
||||
if template.to_csv:
|
||||
delimiter = template.csv_delimiter
|
||||
out_file = co.csv_from_excel(out_file, delimiter, template.csv_quote)
|
||||
out_ext = template.csv_extension
|
||||
return (out_file, "{}.{}".format(out_name, out_ext))
|
||||
(out_file, out_name) = outputs[0]
|
||||
return (base64.encodebytes(out_file), out_name)
|
||||
|
|
|
@ -2,7 +2,6 @@
|
|||
# License AGPL-3.0 or later (https://www.gnu.org/licenses/agpl.html)
|
||||
|
||||
import base64
|
||||
import time
|
||||
import uuid
|
||||
from ast import literal_eval
|
||||
from datetime import date, datetime as dt
|
||||
|
@ -27,7 +26,6 @@ class XLSXImport(models.AbstractModel):
|
|||
def get_eval_context(self, model=False, value=False):
|
||||
eval_context = {
|
||||
"float_compare": float_compare,
|
||||
"time": time,
|
||||
"datetime": dt,
|
||||
"date": date,
|
||||
"env": self.env,
|
||||
|
|
|
@ -1,37 +0,0 @@
|
|||
13.0.1.0.0 (2020-08-23)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Migration to Odoo 13
|
||||
* Enhancement on usability, to be more configurable instead of coding
|
||||
* Import/Export now have "Add/Remove Export Action", "Add/Remove Import Action"
|
||||
* "Easy Report" option which allow user to create export directly by config only.
|
||||
|
||||
12.0.1.0.5 (2019-12-19)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Speed up export when dealing with many rows
|
||||
|
||||
12.0.1.0.4 (2019-08-28)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Fix style sum in footer
|
||||
|
||||
12.0.1.0.3 (2019-08-09)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Add report action for report_type = 'excel'
|
||||
|
||||
12.0.1.0.2 (2019-08-07)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Small fix, to ensure that system parameter 'path_temp_file' (ir.config_parameter) is readable
|
||||
|
||||
12.0.1.0.1 (2019-06-24)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Fix wizard on v12 can't download sample template file - https://github.com/OCA/server-tools/issues/1574
|
||||
|
||||
12.0.1.0.0 (2019-02-24)
|
||||
~~~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
* Start of the history
|
|
@ -1,5 +1,5 @@
|
|||
Concepts
|
||||
--------
|
||||
~~~~~~~~
|
||||
|
||||
This module contain pre-defined function and wizards to make exporting, importing and reporting easy.
|
||||
|
||||
|
@ -17,7 +17,7 @@ After install this module, go to Settings > Excel Import/Export > XLSX Templates
|
|||
As this module provide tools, it is best to explain as use cases. For example use cases, please install **excel_import_export_demo**
|
||||
|
||||
Use Cases
|
||||
---------
|
||||
~~~~~~~~~
|
||||
|
||||
**Use Case 1:** Export/Import Excel on existing document
|
||||
|
||||
|
@ -67,7 +67,7 @@ Please see example in excel_import_export_demo/report_action, which shows,
|
|||
2. Run partner list report based on search criteria.
|
||||
|
||||
Easy Reporting Option
|
||||
---------------------
|
||||
~~~~~~~~~~~~~~~~~~~~~
|
||||
|
||||
Technically, this option is the same as "Create Excel Report" use case. But instead of having to write XML / Python code like normally do,
|
||||
this option allow user to create a report based on a model or view, all by configuration only.
|
||||
|
|
|
@ -2,3 +2,6 @@
|
|||
xlsx_template_user,xlsx_template_user,model_xlsx_template,,1,1,1,1
|
||||
xlsx_template_export_user,xlsx_template_export_user,model_xlsx_template_export,,1,1,1,1
|
||||
xlsx_template_import_user,xlsx_template_import_user,model_xlsx_template_import,,1,1,1,1
|
||||
access_export_xlsx_wizard,access_export_xlsx_wizard,model_export_xlsx_wizard,base.group_user,1,1,1,1
|
||||
access_import_xlsx_wizard,access_import_xlsx_wizard,model_import_xlsx_wizard,base.group_user,1,1,1,1
|
||||
access_report_xlsx_wizard,access_report_xlsx_wizard,model_report_xlsx_wizard,base.group_user,1,1,1,1
|
||||
|
|
|
|
@ -3,6 +3,7 @@
|
|||
|
||||
from odoo import _, api, fields, models
|
||||
from odoo.exceptions import ValidationError
|
||||
from odoo.tools.safe_eval import safe_eval
|
||||
|
||||
|
||||
class ExportXLSXWizard(models.TransientModel):
|
||||
|
@ -21,7 +22,7 @@ class ExportXLSXWizard(models.TransientModel):
|
|||
ondelete="cascade",
|
||||
domain=lambda self: self._context.get("template_domain", []),
|
||||
)
|
||||
res_id = fields.Integer(string="Resource ID", readonly=True, required=True)
|
||||
res_ids = fields.Char(string="Resource IDs", readonly=True, required=True)
|
||||
res_model = fields.Char(
|
||||
string="Resource Model", readonly=True, required=True, size=500
|
||||
)
|
||||
|
@ -35,7 +36,7 @@ class ExportXLSXWizard(models.TransientModel):
|
|||
@api.model
|
||||
def default_get(self, fields):
|
||||
res_model = self._context.get("active_model", False)
|
||||
res_id = self._context.get("active_id", False)
|
||||
res_ids = self._context.get("active_ids", False)
|
||||
template_domain = self._context.get("template_domain", [])
|
||||
templates = self.env["xlsx.template"].search(template_domain)
|
||||
if not templates:
|
||||
|
@ -45,7 +46,7 @@ class ExportXLSXWizard(models.TransientModel):
|
|||
if not template.datas:
|
||||
raise ValidationError(_("No file in %s") % (template.name,))
|
||||
defaults["template_id"] = len(templates) == 1 and templates.id or False
|
||||
defaults["res_id"] = res_id
|
||||
defaults["res_ids"] = ",".join([str(x) for x in res_ids])
|
||||
defaults["res_model"] = res_model
|
||||
return defaults
|
||||
|
||||
|
@ -53,7 +54,7 @@ class ExportXLSXWizard(models.TransientModel):
|
|||
self.ensure_one()
|
||||
Export = self.env["xlsx.export"]
|
||||
out_file, out_name = Export.export_xlsx(
|
||||
self.template_id, self.res_model, self.res_id
|
||||
self.template_id, self.res_model, safe_eval(self.res_ids)
|
||||
)
|
||||
self.write({"state": "get", "data": out_file, "name": out_name})
|
||||
return {
|
||||
|
|
|
@ -16,7 +16,7 @@
|
|||
</group>
|
||||
<group>
|
||||
<field name="res_model" invisible="1" />
|
||||
<field name="res_id" invisible="1" />
|
||||
<field name="res_ids" invisible="1" />
|
||||
</group>
|
||||
</group>
|
||||
<div states="get">
|
||||
|
|
Loading…
Reference in New Issue