[14.0][MIG] excel_import_export, excel_import_export_demo

pull/2505/head
Kitti U 2021-02-13 14:13:07 +07:00 committed by Aungkokolin1997
parent 354890e57a
commit 0421635d7a
11 changed files with 66 additions and 88 deletions

View File

@ -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",

View File

@ -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 = [
(

View File

@ -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

View 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):

View File

@ -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)

View File

@ -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,

View File

@ -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

View File

@ -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.

View File

@ -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

1 id name model_id:id group_id:id perm_read perm_write perm_create perm_unlink
2 xlsx_template_user xlsx_template_user model_xlsx_template 1 1 1 1
3 xlsx_template_export_user xlsx_template_export_user model_xlsx_template_export 1 1 1 1
4 xlsx_template_import_user xlsx_template_import_user model_xlsx_template_import 1 1 1 1
5 access_export_xlsx_wizard access_export_xlsx_wizard model_export_xlsx_wizard base.group_user 1 1 1 1
6 access_import_xlsx_wizard access_import_xlsx_wizard model_import_xlsx_wizard base.group_user 1 1 1 1
7 access_report_xlsx_wizard access_report_xlsx_wizard model_report_xlsx_wizard base.group_user 1 1 1 1

View File

@ -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 {

View File

@ -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">