Odoo14使用hiPrint实现打印功能

使用hiPrint代替odoo原生的打印功能

可以实现快速自定义修改打印模板,无需每次都调整打印模板

无论是表单分页还是各种需求,都能满足

目录

1 使用命令创建新的模块,无用的demo文件可以删除掉

2 新建“打印模板”,用于保存打印模板

1)编写打印模板模型

2)增加打印模板权限

3)增加打印模板视图和菜单

3 首次运行先安装模块

4 增加按钮,定制并保存模板

1)增加自定义按钮“定制模板”xml

2)增加自定义按钮事件js,点击按钮实现打开新的页面

3)增加引入自定义按钮事件js的xml

4)列表视图增加自定义按钮

5)引入以上2个文件

6)引入js,bootstrap,hiprint并引用

7)编写接口,用于打开模板编辑界面,并实现修改保存打印模板

8)编写打开的打印模板页面,在打印模板页面调用接口,实现修改保存打印模板

5 编辑并保存打印模板

1)勾选打印模板记录,点击定制模板

2)在模板设计页面设计打印模板,并保存

6 新建“打印数据”,用于模拟测试打印功能(类似新建“打印模板”,此处省略说明)

1)增加模型,增加权限,增加视图,增加菜单,增加数据

7 在需要进行打印的列表处增加打印按钮

1)增加自定义按钮“打印”xml

2)增加自定义按钮事件js,点击按钮实现打印

3)增加引入自定义按钮事件js的xml

4)列表视图增加自定义按钮

5)引入以上2个文件

8 勾选数据记录,并点击打印按钮即可

9 插件下载


1 使用命令创建新的模块,无用的demo文件可以删除掉

 python odoo-bin scaffold hiprint

2 新建“打印模板”,用于保存打印模板

1)编写打印模板模型

2)增加打印模板权限

3)增加打印模板视图和菜单

class PrintTemp(models.Model):
    _name = 'print.temp'
    _order = 'name DESC'
    _description = '打印模板'

    _sql_constraints = [
        ('unique_code_print_temp', 'unique(code)', '编码需要唯一.'),
    ]

    name = fields.Char(u'名称', copy=False, index=True, required=True)
    code = fields.Char(u'编码', copy=False, index=True, default='/')
    type = fields.Char('类型')
    template = fields.Text('模板')
    remark = fields.Char(string='备注')

    def copy(self, default=None):
        default = dict(default or {})

        copied_count = self.search_count(
            [('name', '=like', u"副本 {}%".format(self.name))])
        if not copied_count:
            new_name = u"副本 {}".format(self.name)
        else:
            new_name = u"副本 {} ({})".format(self.name, copied_count)

        default['name'] = new_name
        return super(PrintTemp, self).copy(default)
id,name,model_id:id,group_id:id,perm_read,perm_write,perm_create,perm_unlink
print_temp,print_temp,model_print_temp,,1,1,1,1
<?xml version="1.0" encoding="utf-8"?>
<odoo>
    <record model="ir.ui.view" id="print_temp_tree_view">
        <field name="name">print.temp.tree</field>
        <field name="model">print.temp</field>
        <field name="arch" type="xml">
            <tree>
                <field name="name"/>
                <field name="code"/>
                <field name="type"/>
                <field name="remark"/>
            </tree>
        </field>
    </record>


    <record model="ir.ui.view" id="print_temp_form_view">
        <field name="name">print.temp.form</field>
        <field name="model">print.temp</field>
        <field name="arch" type="xml">
            <form delete="0">
                <sheet>
                    <group>
                        <group>
                            <field name="code"/>
                            <field name="name" />
                        </group>
                        <group>
                            <field name="template"/>
                            <field name="type"/>
                            <field name="remark"/>
                        </group>
                    </group>
                </sheet>
            </form>
        </field>
    </record>

    <record model="ir.actions.act_window" id="action_print_temp">
        <field name="name">打印模板</field>
        <field name="res_model">print.temp</field>
        <field name="view_mode">tree,form</field>
    </record>
</odoo>
<?xml version="1.0" encoding="utf-8"?>
<odoo>
    <data>
        <menuitem id="hiprint_menu" name="hiprint" web_icon="hiprint,static/description/icon.png" sequence="10"/>

        <menuitem id="hiprint_print_temp" name="打印模板" parent="hiprint_menu" sequence="10" action="action_print_temp"/>
    </data>
</odoo>

3 首次运行先安装模块

python odoo-bin -c odoo.conf -i hiprint

4 增加按钮,定制并保存模板

1)增加自定义按钮“定制模板”xml

2)增加自定义按钮事件js,点击按钮实现打开新的页面

3)增加引入自定义按钮事件js的xml

4)列表视图增加自定义按钮

5)引入以上2个文件

6)引入js,bootstrap,hiprint并引用

7)编写接口,用于打开模板编辑界面,并实现修改保存打印模板

8)编写打开的打印模板页面,在打印模板页面调用接口,实现修改保存打印模板

<?xml version="1.0" encoding="UTF-8"?>

<templates id="print_temp_button_template" xml:space="preserve">

    <t t-extend="ListView.buttons" t-name="hiprint_print_temp_tree_view.buttons">
        <t t-jquery="button.o_list_export_xlsx" t-operation="before">
            <t t-if="widget.modelName=='print.temp'">
                <button type="button" class="btn btn-primary o_list_button_custom">
                    定制模板
                </button>
            </t>
        </t>
    </t>

</templates>

odoo.define('add_print_temp_button', function (require) {
    "use strict";
    var ListController = require('web.ListController');
    var ListView = require('web.ListView');
    var viewRegistry = require('web.view_registry');

    function renderGenerateButton() {
        if (this.$buttons) {
            var self = this;
            this.$buttons.on('click', '.o_list_button_custom', function () {
                console.info('click');
                var actived_ids = []
                var state = self.model.get(self.handle, {raw: true});
                for (var i = 0; i < $('tbody .o_list_record_selector input').length; i++) {
                    if ($('tbody .o_list_record_selector input')[i].checked === true) {
                        actived_ids.push(state.res_ids[i]);
                    }
                }
                var ctx = state.context;
                if(actived_ids.length > 0){
                    window.open("pt_custom/" + actived_ids[0], '_blank');
                }
            });
        }
    }

    var ModelListController = ListController.extend({
        willStart: function () {
            var self = this;
            //设置管理员权限组才看到按钮,可以按需修改
            var ready = this.getSession().user_has_group('base.group_no_one')
                .then(function () {
                    if (true) {
                        self.buttons_template = 'hiprint_print_temp_tree_view.buttons';
                    }
                });
            return Promise.all([this._super.apply(this, arguments), ready]);
        },
        renderButtons: function () {
            this._super.apply(this, arguments);
            renderGenerateButton.apply(this, arguments);
        }
    });

    var ModelListView = ListView.extend({
        config: _.extend({}, ListView.prototype.config, {
            Controller: ModelListController,
        }),
    });

    //将tree视图的按钮注册到视图
    viewRegistry.add('print_temp_tree', ModelListView);
});
<odoo>
    <data>
        <template id="button_js" inherit_id="web.assets_backend">
            <xpath expr="." position="inside">
                <script type="text/javascript" src="/hiprint/static/src/js/print_temp_button.js"/>
            </xpath>
        </template>
    </data>
</odoo>
    <!--增加js_class="print_temp_tree"-->
    <record model="ir.ui.view" id="print_temp_tree_view">
        <field name="name">print.temp.tree</field>
        <field name="model">print.temp</field>
        <field name="arch" type="xml">
            <tree js_class="print_temp_tree">
                <field name="name"/>
                <field name="code"/>
                <field name="type"/>
                <field name="remark"/>
            </tree>
        </field>
    </record>
    'data': [
        'security/ir.model.access.csv',

        'views/print_temp_view.xml',
        'views/menus_view.xml',

        'static/src/xml/include_print_temp_button_js.xml',
    ],
    'qweb': [
        'static/src/xml/print_temp_button_template.xml',
    ]

<?xml version="1.0" encoding="utf-8"?>
<odoo>
    <template id="assets_backend" inherit_id="web.assets_backend">
        <xpath expr="." position="inside">
            <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/hinnn.css"/>
            <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/hiprint.css"/>
            <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/print-lock.css"/>

            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/polyfill.min.js"></script>
            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/jquery.minicolors.min.js"></script>
            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/JsBarcode.all.min.js"></script>
            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/qrcode.js"></script>
            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/hiprint.bundle.js"></script>
            <script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/jquery.hiwprint.js"></script>
            <!--<script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/socket.io.js"></script>-->
        </xpath>
    </template>
</odoo>

# -*- coding: utf-8 -*-


import logging

from odoo import http
from odoo.http import request
from odoo.tools import exception_to_unicode
import jinja2, sys, os

_logger = logging.getLogger(__name__)


class PrintCtrl(http.Controller):

    _pt_custom = '/pt_custom/<int:id>'
    _pt_find = '/pt_find/<int:id>'
    _pt_save = '/pt_save/<int:id>'
    _pd_list = '/pd_list'

    @http.route(_pt_custom, type='http', auth='none', method=['GET', 'POST'], csrf=False)
    def _pt_custom(self, **kw):
        try:
            if hasattr(sys, 'frozen'):
                # When running on compiled windows binary, we don't have access to package loader.
                path = os.path.realpath(os.path.join(os.path.dirname(__file__), '..', 'views'))
                loader = jinja2.FileSystemLoader(path)
            else:
                loader = jinja2.PackageLoader('odoo.addons.hiprint', "views")
            env = jinja2.Environment(loader=loader, autoescape=True)
            return env.get_template("customtemp.html").render({'id': kw.get("id")})
        except Exception as e:
            return exception_to_unicode(e)

    @http.route(_pt_find, type='json', auth='none', methods=['GET', "POST"], csrf=False)
    def pt_find(self, **kw):
        result = {
            'method': self._pt_find,
            'code': 200,
            'message': 'ok',
            'data': {},
        }
        try:
            id = kw.get('id', False)
            code = kw.get('code', False)
            if code:
                print_temp = request.env['print.temp'].sudo().search([("code", "=", code)])
                if print_temp:
                    result['data'] = {
                        'template': print_temp.template,
                    }
                    return result
            if id:
                print_temp = request.env['print.temp'].sudo().search([("id", "=", id)])
                if print_temp:
                    result['data'] = {
                        'template': print_temp.template,
                    }
                    return result
        except Exception as e:
            request.env.cr.rollback()
            result['code'] = 500
            result['message'] = exception_to_unicode(e)
            return result
        return result

    @http.route(_pt_save, type='json', auth='none', methods=["POST"], csrf=False)
    def pt_save(self, **kw):
        result = {
            'method': self._pt_save,
            'code': 200,
            'message': 'ok',
            'data': {},
        }
        try:
            id = kw.get('id', False)
            template = kw.get('template', False)
            if id:
                print_temp = request.env['print.temp'].sudo().search([("id", "=", id)])
                print_temp.template = template
            else:
                result['code'] = 500
                result['message'] = '缺少必要的参数'
                return result
        except Exception as e:
            request.env.cr.rollback()
            result['code'] = 500
            result['message'] = exception_to_unicode(e)
            return result
        return result

    @http.route(_pd_list, type='json', auth='user', methods=["POST"], csrf=False)
    def pd_list(self, **kw):
        result = {
            'method': self._pd_list,
            'code': 200,
            'message': 'ok',
            'data': {},
        }
        try:
            result_data = []
            ids = kw.get('ids', False)
            if ids:
                for id in ids:
                    # 改成逐行查询,一次查询会导致顺序不正确
                    record = request.env['print.data'].sudo().search([("id", "=", id)])
                    result_data.append({
                        'id': record.id,
                        'name': record.name,
                        'code': record.code,
                    })
                result['data'] = {
                    'print_data': result_data,
                    'total': len(result_data),
                }
            else:
                result['code'] = 500
                result['message'] = '缺少必要的参数'
                return result
        except Exception as e:
            request.env.cr.rollback()
            result['code'] = 500
            result['message'] = exception_to_unicode(e)
            return result
        return result
<!DOCTYPE html>
<html>
<head>
    <meta charset="utf-8"/>
    <title></title>
    <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/hinnn.css"/>
    <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/hiprint.css"/>
    <link rel="stylesheet" href="/hiprint/static/src/js/hiprint/css/print-lock.css"/>

    <link rel="stylesheet" href="/hiprint/static/src/js/bootstrap-3.4.1-dist/css/bootstrap.min.css"/>
    <script type="text/javascript" src="/hiprint/static/src/js/jquery/jquery-1.12.4.min.js"></script>

</head>
<body>
<layout class="layout hinnn-layout hinnn-layout-has-sider height-100-per" style="background:#fff;">
    <content class="hinnn-layout-content" style="border-left:1px solid #e8e8e8;">
        <div class="container-fluid height-100-per print-content">

            <div class="row">
                <div class="col-sm-12">
                    <div class="row">
                        <div class="col-sm-3 col-md-2" style="padding-right:0px;">

                            <div class="rect-printElement-types hiprintEpContainer">
                                <ul class="hiprint-printElement-type">

                                    <li>
                                        <span class="title"><code>拖拽列表</code></span>
                                        <ul>
                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.text" style="">

                                                    <span class="glyphicon glyphicon-text-width" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">文本</span>
                                                </a>
                                            </li>

                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.image" style="">
                                                    <span class="glyphicon glyphicon-picture" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">图片</span>
                                                </a>
                                            </li>
                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.longText">
                                                    <span class="glyphicon glyphicon-subscript" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">长文</span>


                                                </a>
                                            </li>

                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.tableCustom" style="">
                                                    <span class="glyphicon glyphicon-th" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">表格</span>
                                                </a>
                                            </li>

                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.html">
                                                    <span class="glyphicon glyphicon-header" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">html</span>
                                                </a>
                                            </li>

                                        </ul>
                                    </li>
                                    <li>
                                        <span class="title">辅助</span>
                                        <ul>
                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.hline" style="">

                                                    <span class="glyphicon glyphicon-resize-horizontal" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">横线</span>
                                                </a>
                                            </li>

                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.vline" style="">
                                                    <span class="glyphicon glyphicon-resize-vertical" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">竖线</span>
                                                </a>
                                            </li>
                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.rect">
                                                    <span class="glyphicon glyphicon-unchecked" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">矩形</span>
                                                </a>
                                            </li>
                                            <li>
                                                <a class="ep-draggable-item" tid="testModule.oval">
                                                    <span class="glyphicon glyphicon-record" aria-hidden="true"></span>
                                                    <span class="glyphicon-class">椭圆</span>
                                                </a>
                                            </li>
                                        </ul>
                                    </li>
                                </ul>
                            </div>

                        </div>
                        <div class="col-sm-9 col-md-10">
                            <div class="hiprint-toolbar" style="margin-top:15px;">
                                <ul>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('A3')">A3</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('A4')">A4</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('A5')">A5</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('B3')">B3</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('B4')">B4</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="setPaper('B5')">B5</a></li>

                                    <li><a class="hiprint-toolbar-item"><input type="text" id="customWidth" style="width: 50px;height: 19px;border: 0px;" placeholder="宽/mm"/></a>
                                    </li>
                                    <li><a class="hiprint-toolbar-item"><input type="text" id="customHeight" style="width: 50px;height: 19px;border: 0px;" placeholder="高/mm"/></a>
                                    </li>

                                    <li><a class="hiprint-toolbar-item" onclick="setPaper($('#customWidth').val(),$('#customHeight').val())">自定义</a></li>
                                    <li><a class="hiprint-toolbar-item" onclick="rotatePaper()">旋转</a></li>
                                    <li>
                                        <a id="save" class="btn hiprint-toolbar-item " style="color: #fff;background-color: #d9534f;border-color: #d43f3a;">保存模板</a>
                                    </li>
                                </ul>
                                <div style="clear:both;"></div>
                            </div>
                            <div id="hiprint-printTemplate" class="hiprint-printTemplate" style="margin-top:20px;">

                            </div>
                        </div>
                    </div>
                </div>
            </div>
        </div>
    </content>
    <sider class="hinnn-layout-sider" style="">
        <div class="container height-100-per" style="width:250px;">
            <div class="row">
                <div class="col-sm-12">
                    <div id="PrintElementOptionSetting" style="margin-top:10px;"></div>
                </div>
            </div>
        </div>
    </sider>
</layout>

<div class="modal fade" id="alertModal" tabindex="-1" role="dialog" aria-labelledby="alertModalLabel">
    <div class="modal-dialog modal-lg" role="document" style="width: 825px;">
        <div class="modal-content">
            <div class="modal-header">
                <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">&times;</span></button>
            </div>
            <div class="modal-body">
                <div class="prevViewDiv"></div>
            </div>
            <div class="modal-footer">
                <button type="button" class="btn btn-default" data-dismiss="modal">关闭</button>

            </div>
        </div>
    </div>
</div>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/custom_test/custom-etype-provider.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/polyfill.min.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/jquery.minicolors.min.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/JsBarcode.all.min.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/qrcode.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/hiprint.bundle.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/hiprint/plugins/jquery.hiwprint.js"></script>
<script type="text/javascript" src="/hiprint/static/src/js/bootstrap-3.4.1-dist/js/bootstrap.min.js"></script>
<script>
        var hiprintTemplate;
        var printTemplateJson;
        var printData = {

        };
        $(function() {
            var settings = {
               "url": "/pt_find/{{id}}",
               "method": "POST",
               "timeout": 0,
               "headers": {
                  "Content-Type": "application/json",
               },
               "data": JSON.stringify({
                  "params": {}
               }),
            };

            $.ajax(settings).done(function (response) {
               console.log(response.result.data.template);
               printTemplateJson = $.parseJSON($.parseJSON(response.result.data.template))

                //初始化打印插件
                hiprint.init({
                    providers: [new customElementTypeProvider()]
                });

                //hiprint.PrintElementTypeManager.build('.hiprintEpContainer', 'testModule');
                //设置左侧拖拽事件
                hiprint.PrintElementTypeManager.buildByHtml($('.ep-draggable-item'));
                //console.info(printTemplateJson);
                hiprintTemplate = new hiprint.PrintTemplate({
                    template: printTemplateJson,
                    settingContainer: '#PrintElementOptionSetting',
                    paginationContainer: '.hiprint-printPagination'
                });
                //打印设计
                hiprintTemplate.design('#hiprint-printTemplate');
            });


			$('#save').click(function () {
                var settings = {
                   "url": "/pt_save/{{id}}",
                   "method": "POST",
                   "timeout": 0,
                   "headers": {
                      "Content-Type": "application/json",
                   },
                   "data": JSON.stringify({
                      "params": {
                        "template": JSON.stringify(JSON.stringify(hiprintTemplate.getJson()))
                      }
                   }),
                };

                $.ajax(settings).done(function (response) {
                    if(response.result.code=='200'){
	    				$('#alertModal .modal-body .prevViewDiv').html("保存成功")
	                    $('#alertModal').modal('show')
                    }
                    else{

                    }
                });
            })
        });

        var setPaper = function (paperTypeOrWidth, height) {
        	if(paperTypeOrWidth!="" && height!="")
            	hiprintTemplate.setPaper(paperTypeOrWidth, height);
        }
        var rotatePaper = function () {
            hiprintTemplate.rotatePaper();
        }

</script>
</body>
</html>

5 编辑并保存打印模板

1)勾选打印模板记录,点击定制模板

2)在模板设计页面设计打印模板,并保存

6 新建“打印数据”,用于模拟测试打印功能(类似新建“打印模板”,此处省略说明)

1)增加模型,增加权限,增加视图,增加菜单,增加数据

7 在需要进行打印的列表处增加打印按钮

1)增加自定义按钮“打印”xml

2)增加自定义按钮事件js,点击按钮实现打印

3)增加引入自定义按钮事件js的xml

4)列表视图增加自定义按钮

5)引入以上2个文件

<?xml version="1.0" encoding="UTF-8"?>

<templates id="print_data_button_template" xml:space="preserve">

    <t t-extend="ListView.buttons" t-name="hiprint_print_data_tree_view.buttons">
        <t t-jquery="button.o_list_export_xlsx" t-operation="before">
            <t t-if="widget.modelName=='print.data'">
                <button type="button" class="btn btn-warning o_list_button_print">
                    打印
                </button>
            </t>
        </t>
    </t>

</templates>

odoo.define('add_print_data_button', function (require) {
    "use strict";
    var ListController = require('web.ListController');
    var ListView = require('web.ListView');
    var viewRegistry = require('web.view_registry');
    var Dialog = require('web.Dialog');
    var core = require('web.core');
    var _t = core._t;

    function renderGenerateButton() {
        if (this.$buttons) {
            var self = this;
            this.$buttons.on('click', '.o_list_button_print', function () {
                var actived_ids = []
                var state = self.model.get(self.handle, {raw: true});
                for (var i = 0; i < $('tbody .o_list_record_selector input').length; i++) {
                    if ($('tbody .o_list_record_selector input')[i].checked === true) {
                        actived_ids.push(state.res_ids[i]);
                    }
                }

                $.ajax({
                   "url": "/pt_find/0",
                   "method": "POST",
                   "timeout": 0,
                   "headers": {
                      "Content-Type": "application/json",
                   },
                   "data": JSON.stringify({
                      "params": {
                        "code": "打印模板1"
                      }
                   }),
                }).done(function (resp) {
                    $.ajax({
                       "url": "/pd_list",
                       "method": "POST",
                       "timeout": 0,
                       "headers": {
                          "Content-Type": "application/json",
                       },
                       "data": JSON.stringify({
                          "params": {
                            "ids": actived_ids
                          }
                       }),
                    }).done(function (response) {
                        if(JSON.stringify(response.result.data) != "{}"){
                            var printTemplateJson = $.parseJSON($.parseJSON(resp.result.data.template));
                            hiprint.init({});
                            var hiprintTemplate = new hiprint.PrintTemplate({
                                template: printTemplateJson,
                                settingContainer: '#PrintElementOptionSetting',
                                paginationContainer: '.hiprint-printPagination'
                            });
                            hiprintTemplate.print(JSON.parse(JSON.stringify(response.result.data.device_device)));
                        }
                        else{
                            console.info(response.result);
                        }
                    });
                });
            });
        }
    }

    var ModelListController = ListController.extend({
        willStart: function () {
            var self = this;
            //设置管理员权限组才看到按钮,可以按需修改
            var ready = this.getSession().user_has_group('base.group_no_one')
                .then(function () {
                    if (true) {
                        self.buttons_template = 'hiprint_print_data_tree_view.buttons';
                    }
                });
            return Promise.all([this._super.apply(this, arguments), ready]);
        },
        renderButtons: function () {
            this._super.apply(this, arguments);
            renderGenerateButton.apply(this, arguments);
        }
    });

    var ModelListView = ListView.extend({
        config: _.extend({}, ListView.prototype.config, {
            Controller: ModelListController,
        }),
    });

    //将tree视图的按钮注册到视图
    viewRegistry.add('print_data_tree', ModelListView);
});
<odoo>
    <data>
        <template id="include_print_data_button_js" inherit_id="web.assets_backend">
            <xpath expr="." position="inside">
                <script type="text/javascript" src="/hiprint/static/src/js/print_data_button.js"/>
            </xpath>
        </template>
    </data>
</odoo>

8 勾选数据记录,并点击打印按钮即可

9 插件下载

https://download.csdn.net/download/u010520912/89520006

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/777788.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

MySQL表的增删改查(CRUD)

MySQL表的增删改查&#xff08;CRUD&#xff09; 文章目录 MySQL表的增删改查&#xff08;CRUD&#xff09;1. Create1.1 单行数据 全列插入1.2 多行数据 指定列插入1.3 插入否则更新1.4 替换 2. Retrieve2.1 SELECT 列2.1.1 全列查询2.1.2 指定列查询2.1.3 查询字段为表达式…

嵌入式系统中状态机实现详解

嵌入式开发中系统经常处于某种状态,如何处理呢?接下来分析一下,状态机的实现无非就是 3 个要素:状态、事件、响应。转换成具体的行为就 3 句话。 发生了什么事? 现在系统处在什么状态? 在这样的状态下发生了这样的事,系统要干什么? 用 C 语言实现状态机主要有 3 种方法…

JVM专题之垃圾收集算法

标记清除算法 第一步:标记 (找出内存中需要回收的对象,并且把它们标记出来) 第二步:清除 (清除掉被标记需要回收的对象,释放出对应的内存空间) 缺点: 标记清除之后会产生大量不连续的内存碎片,空间碎片太多可能会导致以后在程序运行过程中需 要分配较大对象时,无法找到…

spark on k8s两种方式的原理与对比

spark on k8s两种方式的原理与对比 1、spark on k8s 方式 spark-submit可以直接用来向 Kubernetes 集群提交 Spark 应用&#xff0c;提交机制如下&#xff1a; 1、Spark 创建一个在Kubernetes pod中运行的 Spark 驱动程序。 2、驱动程序创建在 Kubernetes Pod 中运行的执行器…

阿里云服务器配置、搭建(针对Spring boot和MySQL项目)

这是一篇极其详细且痛苦的文章&#xff0c;还是在两位后端的大佬手把手教导下、以及我找遍全网所有资料、问了N遍AI、甚至直接申请阿里云工单一对一询问客服一整天、连续清空再上传反复30多次整个项目jar包......总结出来的终极要人命踩坑的问题总结 一、首先购买服务器 其实不…

25.【C语言】循环结构之for 上

1.基本使用 类比while 在while循环中&#xff0c;有三个不可或缺的部分&#xff1a;初始化&#xff0c;判断部分&#xff0c;调整部分 int i 0;//初始化 while (i < 10)//判断部分 {……i;//调整部分 }三个部分太分散&#xff0c;用for循环可集为一体&#xff0c;简洁 …

dotnet ef工具使用

设置工具安装目录 dotnet tool install dotnetsay --tool-path G:\dotnet-tools安装 dotnet tool install --global dotnet-ef更新 dotnet tool update --global dotnet-ef查看版本 dotnet ef --version创建迁移文件 # 只有一个dbcontext dotnet ef migrations add init #…

Android Camera2 集成人脸识别算法

这可能是全网唯一一篇介绍Android Camera2接口集成人脸算法的文章了~ 写在前面&#xff1a; 说起人脸识别&#xff0c;相信大家都不会感到陌生&#xff0c;在我们平时的工作生活中&#xff0c;人脸打卡、刷脸支付等等已经是应用的非常广泛了&#xff0c;人脸识别也给我们的生活…

基于若依的文件上传、下载

基于若依实现文件上传、下载 文章目录 基于若依实现文件上传、下载1、前端实现-文件上传1.1 通用上传分析1.2 修改实现上传接口 2、后端实现-文件上传3、后端实现-文件下载4、前端实现-文件下载 官网其实也写了&#xff0c;但是我是自己改造封装了一下&#xff0c;再次迈向全栈…

kafka系列之消费后不提交offset情况的分析总结

概述 每当我们调用Kafka的poll()方法或者使用KafkaListener(其实底层也是poll()方法)时&#xff0c;它都会返回之前被写入Kafka的记录&#xff0c;即我们组中的消费者还没有读过的记录。 这意味着我们有一种方法可以跟踪该组消费者读取过的记录。 如前所述&#xff0c;Kafka的一…

一种一维时间序列信号变化/事件/异常检测方法(MATLAB)

随着工业物联网、大数据和人工智能的发展&#xff0c;传统工业正在向数字化和智能化升级&#xff0c;从而创造了大量的时间序列数据。通过分析这些数据&#xff0c;可以提供准确可靠的信息服务和决策依据&#xff0c;促进制造业的转型升级。工业物联网在传统工业向“工业 4.0”…

CASS如何输入距离和坐标绘制图形

1、软件版本 这里使用CASS7.0软件进行操作。如下&#xff1a; 2、 操作 输入&#xff1a;500,45【表示距离500米&#xff0c;方向45度】。 点击回车&#xff0c;完成绘制。

V3.76 ViVaCut高级版本!已经开启永久订阅!一款专业视频剪辑软件

在短视频和社交媒体盛行的今天&#xff0c;视频内容的创作和分享已经成为人们日常生活的一部分。为了满足广大视频创作者的需求&#xff0c;一款专业视频剪辑软件应运而生&#xff0c;以其强大的功能和用户友好的界面&#xff0c;为用户提供了全新的视频编辑体验。他提供了众多…

JWT(Json Web Token)在.NET Core中的使用

登录成功时生成JWT字符串目录 JWT是什么&#xff1f; JWT的优点&#xff1a; JWT在.NET Core 中的使用 JWT是什么&#xff1f; JWT把登录信息&#xff08;也称作令牌&#xff09;保存在客户端为了防止客户端的数据造假&#xff0c;保存在客户端的令牌经过了签名处理&#xf…

TikTok马来西亚直播网络怎么配置?

TikTok是一款全球流行的社交媒体应用&#xff0c;在东南亚地区拥有大量用户。在马来西亚这个多元化的国家&#xff0c;配置高效稳定的直播网络对TikTok的运营至关重要。 配置马来西亚直播网络的必要性 广泛的地理覆盖&#xff1a;马来西亚包括大片陆地和众多岛屿&#xff0c;网…

加入运动健康数据开放平台,共赢鸿蒙未来

HarmonyOS SDK运动健康服务&#xff08;Health Service Kit&#xff09;是为华为生态应用打造的基于华为帐号和用户授权的运动健康数据开放平台。在获取用户授权后&#xff0c;开发者可以使用运动健康服务提供的开放能力获取运动健康数据&#xff0c;基于多种类型数据构建运动健…

大数据Spark 面经

1: Spark 整体架构 Spark 是新一代的大数据处理引擎&#xff0c;支持批处理和流处理&#xff0c;也还支持各种机器学习和图计算&#xff0c;它就是一个Master-worker 架构&#xff0c;所以整个的架构就如下所示&#xff1a; 2: Spark 任务提交命令 一般我们使用shell 命令提…

理解MySQL核心技术:存储过程与函数的强大功能

在大型应用程序和复杂的数据库操作中&#xff0c;存储过程与函数扮演着至关重要的角色。它们不仅可以提高代码的可维护性&#xff0c;还能加强数据库的安全性和性能。本篇文章将深入探讨MySQL存储过程与函数的基础知识、创建、管理及其在实际应用中的优势。 什么是存储过程和函…

set的应用(C++)

set的使用 【基本用法】 大家可以敲一下这段代码体会一下set的基本初始化和使用 #include <iostream> #include <set> #include <vector> using namespace std;int main() {set<int> st1; // 空的set// 使用迭代器构造string str("abcdef"…

uniapp实现一个键盘功能

前言 因为公司需要&#xff0c;所以我.... 演示 代码 键盘组件代码 <template><view class"keyboard_container"><view class"li" v-for"(item, index) in arr" :key"index" click"changArr(item)" :sty…