GoodERP
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

158 lines
5.6KB

  1. # Copyright 2016 上海开阖软件有限公司 (http://www.osbzr.com)
  2. # Copyright 2019 信莱德软件
  3. # License AGPL-3.0 or later (http://www.gnu.org/licenses/agpl)
  4. import base64
  5. import json
  6. from werkzeug import exceptions
  7. from werkzeug.urls import url_decode,url_unquote_plus,url_quote
  8. from odoo.tools.safe_eval import safe_eval, time
  9. from odoo.http import route, request, serialize_exception, content_disposition
  10. from odoo.tools import html_escape
  11. from odoo.addons.web.controllers import report
  12. import logging
  13. _logger = logging.getLogger(__name__)
  14. html_template='''
  15. <!DOCTYPE html>
  16. <html>
  17. <head>
  18. <title>%s</title>
  19. <style>
  20. .pdf-container {
  21. width: 100%%;
  22. height: 100vh; /* 使用视口高度作为高度 */
  23. }
  24. </style>
  25. </head>
  26. <body>
  27. <embed class="pdf-container" src="data:application/pdf;base64,%s" type="application/pdf" title="%s">
  28. </body>
  29. </html>
  30. '''
  31. class ReportController(report.ReportController):
  32. TYPES_MAPPING = {
  33. 'doc': 'application/vnd.ms-word',
  34. 'html': 'text/html',
  35. 'odt': 'application/vnd.oasis.opendocument.text',
  36. 'pdf': 'application/pdf',
  37. 'sxw': 'application/vnd.sun.xml.writer',
  38. 'xls': 'application/vnd.ms-excel',
  39. 'xlsx': 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet',
  40. 'docx': 'application/vnd.openxmlformats-officedocument.wordprocessingml.document',
  41. }
  42. @route( [
  43. "/report/<converter>/<reportname>",
  44. "/report/<converter>/<reportname>/<docids>",
  45. "/report/<converter>/<reportname>/<docids>/<filename>",
  46. ],
  47. type="http",
  48. auth="user",
  49. website=True,)
  50. def report_routes(
  51. self, reportname, docids=None, converter=None, options=None, **data
  52. ):
  53. report = request.env['ir.actions.report']
  54. context = dict(request.env.context)
  55. if docids:
  56. docids = [int(i) for i in docids.split(',') if i.isdigit()]
  57. if data.get('options'):
  58. data.update(json.loads(data.pop('options')))
  59. if data.get('context'):
  60. data['context'] = json.loads(data['context'])
  61. context.update(data['context'])
  62. if converter == "docx":
  63. action_docx_report = report.get_from_report_name(
  64. reportname, "docx").with_context(context)
  65. if not action_docx_report:
  66. raise exceptions.HTTPException(
  67. description='Docx action report not found for report_name '
  68. '%s' % reportname)
  69. pdf, filetype = action_docx_report.render_docx(docids, data)
  70. filename = action_docx_report.gen_report_download_filename(
  71. docids, data)
  72. if not filename.endswith(filetype):
  73. filename = "{}.{}".format(filename, filetype)
  74. if filetype == 'docx':
  75. headers = [
  76. ("Content-Disposition", "%s" % content_disposition(filename))
  77. ]
  78. else:
  79. headers = [
  80. ("Content-Type", "application/pdf"),
  81. ("Content-Length", len(pdf)),
  82. ("Content-Disposition", 'inline; filename="%s"' % url_quote(filename))
  83. ]
  84. return request.make_response(
  85. pdf,
  86. headers
  87. )
  88. elif converter =='qweb-pdf':
  89. pdf = report.with_context(context)._render_qweb_pdf(reportname, docids, data=data)[0]
  90. pdfhttpheaders = [('Content-Type', 'application/pdf'),
  91. ('Content-Length', len(pdf)),
  92. ("Content-Disposition", 'inline; filename="%s"' % url_quote('filename.pdf'))]
  93. return request.make_response(pdf, headers=pdfhttpheaders)
  94. else:
  95. return super().report_routes(
  96. reportname,
  97. docids=docids,
  98. converter=converter,
  99. options=options,
  100. **data,
  101. )
  102. @route()
  103. def report_download(self, data, context=None, token=None):
  104. """This function is used by 'qwebactionmanager.js' in order to trigger
  105. the download of a docx/controller report.
  106. :param data: a javascript array JSON.stringified containg report
  107. internal url ([0]) and type [1]
  108. :returns: Response with a filetoken cookie and an attachment header
  109. """
  110. requestcontent = json.loads(data)
  111. url, report_type = requestcontent[0], requestcontent[1]
  112. if 'docx' not in report_type:
  113. return super().report_download(data, context, token)
  114. try:
  115. reportname = url.split('/report/docx/')[1].split('?')[0]
  116. docids = None
  117. if '/' in reportname:
  118. reportname, docids = reportname.split('/')
  119. if docids:
  120. # Generic report:
  121. response = self.report_routes(
  122. reportname, docids=docids, converter='docx')
  123. else:
  124. # Particular report:
  125. # decoding the args represented in JSON
  126. data = list(url_decode(url.split('?')[1]).items())
  127. response = self.report_routes(
  128. reportname, converter='docx', **dict(data))
  129. response.set_cookie('fileToken', token)
  130. return response
  131. except Exception as e:
  132. se = serialize_exception(e)
  133. error = {
  134. 'code': 200,
  135. 'message': "Odoo Server Error",
  136. 'data': se
  137. }
  138. return request.make_response(html_escape(json.dumps(error)))
上海开阖软件有限公司 沪ICP备12045867号-1