(04)odoo视图操作

-----------------
更新时间
19:04 2016-09-29 星期四
11:17 2016-09-18 星期日
18:13 2016-04-05 星期二
15:05 2016-03-14 星期一
11:46 2016-02-25 星期四
14:51 2016-02-23 星期二
18:07 2016-02-19 星期五
17:44 2016-02-17 星期三
-----------------
* 视图标签类型
    # <record>标签有属性 model="ir.ui.view" ,它包含本身的视图定义
    # <record>标签有属性 mo="ir.actions.act_window",它将动作和视图链接起来
    # <menuitem>标签,把菜单和动作链接起来

* 模板(Templating)
    <template id="..." name="...">
      html
    </template>
   
    --------
    <record id="..." model="ir.ui.view">
       <field name="name">...</field>
       ...
       <field name="arch" type="xml">
           html
       </field>
     </record>
    
    #渲染模板
        http.request.render(template[,values])
       
        @http.route('/hello/')
        def hello(self, **kw):
            return http.request.render(module.hello)
    -------
        <template id="hello">
            <p> hello , <t t-esc="name"/></p>
        </template>
    --------
        return http.request.render(module.hello,{'name':"World"}) # 带参数
       
    # Qweb 设置变量值
        <template id="hello">
            <t t-set="greet" t-value="name+'!!!'">
            <p> hello , <t t-esc="greet"/></p>
        </template>
    --------
        <template id="hello">
            <t t-set="greet" t-valuef="{{name}}!!!">
            <p> hello , <t t-esc="greet"/></p>
        </template>
    ---------
     原样输出
        <template id="hello">
            <t t-set="greet">
                <em>Hello</em>, <t t-esc="name"/>
            </t>
            <p> hello , <t t-raw="greet"/></p>
        </template>
       
    # Qweb 语句
      条件语句
          <template id="hello">   
            <p>
                <span t-if="name=='World'" />
                    Hello World!
                </span>
            </p>
          </template>
          -----
          <template id="hello">   
            <t t-set="condition" t-value="name=='World'">
            <p>
                <span t-if="condition" />
                    Hello World!
                </span>
            </p>
            <p>
                <span t-if=" not condition" />
                    Hello not World!
                </span>
            </p>
          </template>
     
      循环语句
           <template id="hello">
                <ul>
                    <li t-foreach="name" t-as="letter">
                        <t t-esc="letter_index"/>:<t t-esc="letter" />
                    </li>
                </ul>
           <template>
           -------
           <t t-foreach="seq" t-as="value">
           * value 项值
           * value_size 大小
           * value_all 总数
           * value_index 索引号
           * value_first:bool 第一个
           * value_last:bool 最后一个
           * value_parity:'even'|'odd' 当前是第奇数还是偶数
      
       调用其它模板
           采用t-call
           <template id="sub">
            <t t-esc="identifier" />
           </template>
           <template id="hello">
            <p>
                hello,
                <t t-call="module.sub">
                    <t t-set="identifier" t-value="name" />
                </t>
            </p>
           </template>
      
       属性设置
           t-att-{attname}="{expression}"
           t-attf-{attname}="{expression}"
           ------
           <template id="hello">
            <p t-att-class="name.lower()">Hello,world</p>
            <p t-attf-class="cl-{{name.lower()}}">Hello,world</p>
           <template>
      
       字段渲染
           @http.route('hello/<model("res.users"):user')  # 给用户的id即可
           def hello(self,user,**kw)
                return http.request.render('module.hello',{'user':user})
            -------
            <template id="hello">
                <p t-field="user.display_name" />
            </template>
       ---------
       可用字段选择修饰
           <template id="hello">
                <p t-field="user.creat_date" />
                <p t-field="user.creat_date"  t-filed-options='{"format":"long"}'/>
                <p t-field="user.creat_date"  t-filed-options='{"format":"EEE"}'/>
            </template>
            -------------
            <template id="hello">
                <p t-field="user.wealth" />
                <p t-field="user.wealth"  t-filed-options='{
                     "widget":"monetary"
                     "display_currency":"user.company_id.currency_id"
                     }'/>
            </template>
            ------------
            <template id="hello">
                <p t-field="user.create_date" t-field-options='{"widget":relative}}' />
            </template>
       
        模板继承
            <template id="hello">
                <p> Base template </p>
            </template>
            <template id="hello2" inherit_id="hello" name="Extender">
                <xpath expr="//p" position="before">
                    <h1>Extended!</h1>
                </xpath>   
            </template>
             得到的结果:
               <h1>Extended!</h1>
               <p>Base template</p>
            --------------
            <template id="hello">
                <p class="a">A</p>
                <p class="b">B</p>           
            </template>
            <template id="hello2" inherit_id="hello" name="Extender">
                <xpath expr="//p[hasclass('b')]" position="before">
                    <h1>Extended!</h1>
                </xpath>   
            </template>   
              得到的结果:
               <p class="a">A</p>
               <h1>Extended!</h1>
               <p class="b">B</p>
            ----------
            调用系统的基础模板:
              <template id="hello">
               <t t-call="website.layout">
                    <p class="a">A</p>
                    <p class="b">B</p>   
               </t>               
            </template>
            <template id="hello2" inherit_id="hello" name="Extender">
                <xpath expr="//p[hasclass('b')]" position="before">
                    <h1>Extended!</h1>
                </xpath>   
            </template>   
            -----------
                   
           
* 视图
    <record id="view_id" model="ir.ui.view">
       <field name="name">view.name</field>
       <field name="model">object_name</field>
       <field name="type">form</field>
       <field name="priority" eval="16" />
       <field name="arch" type="xml">
          <!-- view content <form>,<tree>,<graph>,.. -->
       </field>
   </record>   
   属性:
   openerp/addons/base/ir/ir_ui_view.py view类中的_columns
   model:固定为 ir.ui.view
   id 唯一,调用和继承会用到
   name 视图名称
   model 对象
   type 有多种类型 tree,form,graph,calendar,diagram,gantt,kanban,search
   priority  排序
  
* 视图继承(用于修改已有的视图):
   这里有一个小技巧:
   当你继承父视图,
   1.只添加字段时,本视图外id值可以和父视图id值 一样,这样后面再继承方便
   2.当要隐藏或删除父字段时,本视图外id不能和父视图id值一样,否则后面无法升级模块
  
    <record model="ir.ui.view" id="view_partner_form">
        <field name="name">res.partner.form.inherit</field>
        <field name="model">res.partner</field>
        <field name="inherit_id" ref="base.view_partner_form" /> <!--声明要修改那一个视图-->
        <field name="arch" type="xml">
            <notebook position="inside">
                <page string="Relations">
                    <field name="relations_ids" colspan='4' nolabel="1" />
                </page>
            <notebook>
        <field>
    </record>
    所有的操作只能在
    <field name="arch" type="xml">... </field>
   
    定位5种属性
    @ inside(默认):附加在标签内
    @ after:后添加内容标签
    @ before:前的内容标签
    @ replace: 更换标签内容
    @ attributes: 更改标签内容的属性
   
   支持定位方法:
            <notebook position="inside">
            <xpath expr="//page[@name='page_history']" position="inside">
            <field name="mobile" position="after">
            <filter name="consumable" position="after" >
            <group name="bank" position="after">
            <xpath expr="//field[@name='standard_price']" position="replace">
            <xpath expr="//button[@name='open_ui']" position="replace">
            <xpath expr="//div[@class='oe_employee_details']/h4/a" position="after">
            <xpath expr="//form/sheet/notebook/page/field[@name='line_ids']/tree/field[@name='analytic_account_id']"
                    position="replace">
            <xpath expr="/form/sheet/notebook/page/field[@name='line_ids']/form/group/field[@name='analytic_account_id']"
                    position="replace">   
    # 更换内容:
        <record model="ir.ui.view" id="view_partner_form1">
            <field name="name">res.partner.form.inherit1</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                    <page string="Extra Info" position="replace">
                        <field name="relations_ids" colspan='4' nolabel="1" />
                    </page>
            <field>
        </record>
   
    # 删除内容:
        <record model="ir.ui.view" id="view_partner_form2">
            <field name="name">res.partner.form.inherit2</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                <field name="lang" position="replace"/>
            <field>
        </record>   
       
    # 插入内容   
        <record model="ir.ui.view" id="view_partner_form3">
            <field name="name">res.partner.form.inherit3</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                <field name="lang" position="before">
                    <field name="relation_ids" />
                </field>
            <field>
        </record>  

        <record model="ir.ui.view" id="view_partner_form4">
            <field name="name">res.partner.form.inherit4</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                <field name="lang" position="after">
                    <field name="relation_ids" />
                </field>
            <field>
        </record>  
       
    # 修改属性:修改属性能实现的功能,不要使用replace
        <xpath expr="//field[@name='name']"position="attributes">
            <attribute name="required">1</attribute>
        </xpath>   
       
    # 多变化
        <record model="ir.ui.view" id="view_partner_form5">
            <field name="name">res.partner.form.inherit5</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                <field name="lang" position="replace"/>
                <field name="website" position="after">
                    <field name="lang" />
                </field>
            <field>
        </record>   
       
    # xpath的节点
        <record model="ir.ui.view" id="view_partner_form6">
            <field name="name">res.partner.form.inherit6</field>
            <field name="model">res.partner</field>
            <field name="inherit_id" ref="base.view_partner_form" />
            <field name="arch" type="xml">
                <data>
                    <xpath expr="/feild[@name='address']/form/field[@name='email']"
                         position="after">
                         <field name="age" />
                    </xpath>    
                    <xpath expr="/field[@name='address']/tree/field[@name='email']"
                         position="after">
                         <field name="age" />
                    </xpath>                   
                </data>
            <field>
        </record>    
       
* 窗口操作(视图动作)
   如下一些代码
   <?xml version="1.0" encoding="utf-8" ?>
    <openerp>
       <data>
         <act_window id="action_todo_stage"
                     name="To-Do Task Stages"
                     res_model="todo.task.stage"
                     view_mode="tree,form" />

         <act_window id="todo_app.action_todo_task"
                     name="To-Do Task "
                     res_model="todo.task"
                     view_mode="tree,form,calendar,gantt,graph"
                     target="current"
                     context="{'default_user_id':uid}"
                     domain="[]"
                     limit="80" />
         <act_window id="action_todo_task_stage"
                     name="To-Do Task Stages"
                     res_model="todo.task.stage"
                     src_model="todo.task"
                     multi="False" />
       </data>

    </openerp>
   
    可以用ir.action.act_window 重新其中一菜单对角
    <record id="action_todo_stage" model="ir.actions.act_window">
        <field name="name">To-Do Task Stage</field>
        <field name="type">ir.actions.act_window</field>
        <field name="domain">["list of 3-tuples (max 250 characters)"]</field>
        <field name="context">{"context dictionary (max 250 characters)"}</field>
        <field name="res_model">todo.task.stage</field>
        <field name="view_type">form</field>
        <field name="view_mode">tree,form</field>   这里表示有两个视图可用,且先tree,后form,  若只有tree, 则后面点列表是没反应的
        <field name="search_view_id" ref="todo_task_search"/>
        <field name="view_id" ref="view_form_todo_task_ui" />
    </record>
   
    窗口的动作是存在 ir.actions.act_window 模型中的,我们可以在xml文件中用<act_window>来快捷定义
    下面是部分属性说明,完整的可以在 openerp/addons/base/ir/ir_actions.py 中的 ir.actions.act_window
    类中可以找到
    # name 显示的标题
    # type 动作类型 ,默认ir.actions.act_window
    # view_id 中的 ref 值是对应视图对象的id 触发时用哪个视图
    # context 设置目标视图的上下文
              比如:<field name="context">{'search_default_group_by_product': True, 'search_default_group_by_location': True}</field>
              <!--默认搜索 以产品分组 和以库位分组-->             
    # domain 过滤记录按条件,如:[["customer", "=", true]]
    # res_id: 当动作打开视图是表单视图时,要指定的加载记录id,只有‘view_mode’ 值为 form时才有效
              否则就会新建一个记录 如:"res_id": a_product_id,
    # res_model 动作响应的目标模型
    # target 如果设置为new 就打开新窗口,默认是 current
    # view_mode:列出允许使用的视图模式 如 form, tree, calendar,etc 默认是 tree,form
    # view_type:设定tree时的列表是树状还是普通列表, tree-树状   form-普通列表 缺省是form
    # usage: 过滤菜单和动作
    # view_ids:
    # views:是(view_id,view_type) 元组对列表,第一组是动作默认打开的视图
            如:"views": [[False, "tree"], [False, "form"]]
    # limit 指列表视图时,一页的记录数,默认是80
    # auto_refresh: 在视图中添加一个刷新功能
    # groups_id:权限组
    # search_view_id 指定响应的搜索视图(id,name) 元组对
    # filter:列表视图显示过滤器(bool)   
    # auto_search: 加载默认视图后,自动搜索
    # src_model 指定可以启动视图的更多按钮
    # multi 设为True, 更多按钮显示在列表视图,否则显示在表单视图   
   
   
    系统的默认值:
      _defaults = {
        'type': 'ir.actions.act_window',
        'view_type': 'form',
        'view_mode': 'tree,form',
        'context': '{}',
        'limit': 80,
        'target': 'current',
        'auto_refresh': 0,
        'auto_search':True,
        'multi': False,
    }
   
* 菜单项   
    如下一些代码
   
         <menuitem id="menu_todo_task_main"
                     name="To-Do" parent="mail.mail_my_stuff"/>
         <menuitem id="todo_app.menu_todo_task"
                     name="To-Do Tasks"
                     parent="menu_todo_task_main"
                     sequence="10"
                     action="todo_app.action_todo_task"/>
         <menuitem id="menu_todo_task_stage"
                     name="To-Do Stages"
                     parent="menu_todo_task_main"
                     sequence="20"
                     action="action_todo_stage"/>

    在 设置-> 技术-> 用户界面-> 菜单  可以看到已定义的菜单
    菜单是存在 ir.ui.menu 模型中的,我们可以在xml文件中用<menuitem>来快捷定义
    openerp/addons/base/ir/ir_ui_menu.py
    # id 菜单的唯一id用于父子级别和调用
    # name 在视图中显示的名字 如果定义的 action,这个可以省略,会用对应action
            的name值
    # sequence 显示排序
    # parent 本菜单的父菜单,子菜单都要指定,只有顶级菜单不要指定
    # action 本菜单的连接动作
    # groups 指定权限组,用户组的extraID
   
    这里的action 要关联到视图动作定义中的id
    如 action="action_todo_stage  对应视图动作中的 id="action_todo_stage"
    注:在定义菜单和窗体动作时,一定要注意一下顺序,先定义菜单对应的窗体动作
    如:
    <record model="ir.actions.act_window" id="action_list_ideas">
        <field name="name">Ideas</field>
        <field name="res_model">idea.idea</field>
        <field name="view_mode">tree,form</field>
    </record>
     <menuitem id="menu_ideas" parent="menu_root" name="Ideas" sequence="10"
          action="action_list_ideas"/>
   
* 在中间“更多”按钮加动作
<record id="action_sale_order_make_invoice" model="ir.actions.act_window">
            <field name="name">Make Invoices</field>
            <field name="type">ir.actions.act_window</field>
            <field name="res_model">sale.make.invoice</field>
            <field name="view_type">form</field>
            <field name="view_mode">form</field>
            <field name="view_id" ref="view_sale_order_make_invoice"/>
            <field name="target">new</field>
            <field name="multi">True</field>
</record>

<record model="ir.values" id="sale_order_make_invoice">
            <field name="model_id" ref="sale.model_sale_order" />
            <field name="name">Make Invoices</field>
            <field name="key2">client_action_multi</field>
            <field name="value" eval="'ir.actions.act_window,' + str(ref('action_sale_order_make_invoice'))" />
            <field name="key">action</field>
            <field name="model">sale.order</field>
  </record>
重点讲第二段
@model_id  对应的值是所在那个模型中加 格式[模块名].model_模型名 用_连起来,若是本模块,可以省了[模块名].
  视图上采用ref 引用模型都是这种格式
@name 定的名称
@key2  动作多窗口
@value 引用一个已定义的窗体动作来完成
@key 产生的类型,这里是动作
@model作用在的模型

         
      
* 三个主要的视图  list或 tree(树状)视图, 表单form视图, search搜索视图    对应模型 ir.ui.view  
  在openerp/addons/base/ir/ir_ui_view.py 中可以看到有 tree, form, graph, calendar, diagram
   gantt,kanban, search,qweb 这些视图
   定义视图的总框架:
   <record model="ir.ui.view" id="view_id">
    <field name="name">view.name</field>
    <field name="model">object_name</field>
    <field name="priority" eval="16"/>
    <field name="arch" type="xml">
        <!-- view content: <form>, <tree>, <graph>, ... -->
    </field>
  </record>
  arch 字段一定要申明 type为xml 否则无法解析
  
  
* 表单视图
    # 添加在动作和菜单项后面 <act_window> <menuitem>
    代码如下:
           <record id="view_form_todo_task_ui" model="ir.ui.view">
               <field name="name">view_form_todo_task_ui</field>
               <field name="model">todo.task</field>
               <field name="arch" type="xml">
                   <form>
                       <header> <!-- Buttons and status widget --> </header>
                       <sheet> <!-- Form content --> </sheet>
                       <!-- History and communication: -->
                       <div class="oe_chatter">
                           <field name="message_follower_ids"
                                  widget="mail_followers"/>
                           <field name="message_ids"
                                  widget="mail_thread"/>
                       </div>
                   </form>
               </field>
           </record>
    #包含三个可视域:
       <header> 
       <sheet> 放内容的
       <bottom> 历史记录和社交部分
    # header 状态栏
        <header>
           <field name="stage_state" invisible="True"/>
           <button name="do_toggle_done" type="object"
                        attrs="{'invisible':
                                [('stage_state','in',['done','cancel'])]}"
                                   string="Toggle Done" class="oe_highlight"/>
                           <!-- Add stage statusbar: ... -->
        </header>   
        @ class="oe_highlight" 加高亮
        @ invisible 为可见性,还可以用到其它元素上,不仅在<botton>上
        -------------------------
        <header>
           <field name="state" widget="statusbar"
            statusbar_visible="draft,sent,progress,invoiced,done"
            statusbar_colors="{'shipping_except':'red','waiting_date':'blue'}"/>
        </header>
        @ 加入状态挂件
       
       
    # 元素功能分组排列
        group 分组的性性
         @ colspan 说明该控件占用父容器多少列 openerp form 为顶级容器,约定为4列
           colspan 缺省是 2
         @ rowspan 行数
         @ expand
         @ col  用于容器控件,如group 表不这个容器内部分几列
         @ string 组的名称
         @ openerp 控件一般都有lable 这样占两列
         @ class 加载对应的类名用于定位
        
         举一例:
            ①<group col="6" colspan='4'>
                ②<group col="2" colspan="4">
                    <separator colspan="2" string="Product Description"/>
                    <field name="name" select="1">
                    <field groups="base.group_extended" name="variants" select="2" />
                </group>
                ③<group col="2" colspan="1">
                    <separator colspan="2" string="Codes"/>
                    <field name="default_code" select="1">
                    <field groups="base.group_extended" name="ean13" select="2" />
                </group>
                ④<group col="2" colspan="1">
                    <separator colspan="2" string="Product Type"/>
                    <field name="sele_ok" select="2">
                    <field name="purchase_ok" select="2">
                    <field groups="base.group_extended" name="rental" select="2" />
                </group>
            </group>
            ① colspan="4" 占据了整行,col="6" 表示①这个组控件内分为6列
            ② col="2" 表示②这个组控件内分为2列, colspan="4" 表示占①组控件4列
            ③ col="2" 表示③这个组控件内分为2列, colspan="1" 表示占①组控件1列
            ④ col="2" 表示④这个组控件内分为2列, colspan="1" 表示占①组控件1列
           
            注:
            加了<group>...</group>, 里面的field会自动加上<label>
            <group>顶级是默认里面分为4列
            <group>子级里面默认是2列,这样<label>占一列,<field>值占一列
            <group>里面的field不要加<label>要指明,如:
            <group string="Source Term">
                <field name="source" nolabel="1" height="400"/>
            </group>
           
        group 也有对应的class,可以用于定位,如(右下角):
        <group class="oe_subtotal_footer">
            <field name="amount_untaxed"/>
            <field name="amount_tax"/>
            <field name="amount_total" class="oe_subtotal_footer_separator"/>
            <field name="residual" style="margin-top: 10px"/>
        </group>
               
    # 有时<sheet>之间的内容采用html来布局会更灵活
     如:
        <sheet>
            <div class="oe_title">
                <label for="name" class="oe_edit_only" string="Idea Name" />
                <h1><field name="name" /></h1>
            </div>
            <separator string="General" colspan="2" />
            <group colspan="2" col="2">
                <field name="description" placeholder="Idea description..." />
            </group>
        </sheet>
       
        <field name="product_image" widget="image" class="oe_avatar oe_right"/>
       
    # notebook     是标签块
        @string tab的标签名
        @accesskey
        @attrs
       
    # newline   
    # separator 分割线
    # header 头部     
       
* 视图元素
    #按钮<button>,支持的属性
     icon  可用的icon在 addons/web/static/src/img/icons
     string 按钮的显示文字
     type  值可以是 workflow, object action
     name  就是要触发的方法标识
     args  传递方法的参数
     content 上下文
     confirm  针对对话框的确认
     special="cancel" 用于向导
     states 可见的状态
     classname 加载的类名(常用 oe_highlight)
    
     例子:
     <footer>
        <button name="act_update" string="Update" type="object" class="oe_highlight"/>
        or
        <button special="cancel" string="Cancel" type="object" class="oe_link"/>
     </footer>
    
     #字段<field>,支持的属性
      name 字段技术名
      string 显示名
      help 帮助提示
      placeholder 占位文字
      widget 小物件 后面常用值(progress, many2onebutton, handle, statusbar, image, many2many_tags, selection)
      options 小物件对应的选项,
      class 加载类名
      (常用的
      oe_inline
      oe_left
      oe_right
      oe_read_only
      oe_edit_only
      oe_no_button
      oe_avatar     图片字段用
      )
      invisible="1" 标识不可见
      groups 指定权限的可见 后面的值一般为 用户组或用户名
      nolabel="1" 不显示标签,会用于<group>之内
      readonly="1" 标识只读
      required="1" 标识不能为空
      on_change 这是旧版的,新版用@api.change(self)来处理
      domain 条件过滤(仅用于关联字段)
      context 上下文(仅用于关联字段)
      required 必填
      placeholder 占位的字符
      mode 用于 One2many (有值 tree, form, kanban, graph  默认是tree)
      help 字段的提示信息
      sum,avg 小计,平均值
      password="True" 密码框
      filename 文件上传用
      attrs  后面是表达式
     
     # 标签
        <label>
        <label for="name" class="oe_edit_only"/>
         <h1><field name="name"/></h1>   

     #关联字段
      options={'no_open':True,'no_create':True}    
      常用于 context 和 domain中
     
     # 小物件
       用于文本字段
        email  url  html
       用于数字字段
        float_time   monetary   progressbar
       用于关联和选择
         many2many_tags
         selection
         radio
         kanban_state_selection
         priority

* 事件变化
   @api.onchange('field1','field2')
   举例说明:
   视图中定义如下:
   <!-- content of form view -->
    <field name="amount"/>
    <field name="unit_price"/>
    <field name="price" readonly="1"/>
   
    对应模型中的定义如下:
    # onchange handler
    @api.onchange('amount', 'unit_price')
    def _onchange_price(self): #当用api.onchange时 self为记录,不是记录集
        # set auto-changing field
        self.price = self.amount * self.unit_price
        # Can optionally return a warning and domains
        return {
            'warning': {
                'title': "Something bad happened",
                'message': "It was very bad indeed",
        }
    }
   
    这样只要 amount 或 unit_price 字段的值变化,对应的price值跟着变化
    注:只支持简单字段,像many2one, one2many 这些字段不支持
   
    若要指定在那个字段去触发,则在视图中定义
    <field name="name" on_change="0"/>
    on_change="0"
  

* 动态视图
    # group
    # states
    增加灵活性 用 refers_to
    <field name="refers_to"
          attrs="{'invisible': [('state','=','draft')]}" />
         
* 列表视图(分为两种,正常列表,和树状列表)
   <record id="todo_app.view_tree_todo_task" model="ir.ui.view">
       <field name="name">To-do Task Tree</field>
       <field name="model">todo.task</field>
       <field name="arch" type="xml">
           <tree editable="bottom"
                 colors="gray:is_done==True"
                 fonts="italic: state!='open'" delete="false">
               <field name="name"/>
               <field name="user_id"/>
           </tree>
       </field>
   </record>
  
   # <tree>
    属性:
     editable 设置字段在列表直接可以编辑
     例:<tree string="Translations" editable="top">
    
     colors 按条件设定记录的颜色
     <tree string="Idea Categories" colors="blue:state=='draft';red:state=='trashed'">
        <field name="name"/>
        <field name="state"/>
     </tree>
    
     default_order 指定列表记录的排序条件字段
     <tree default_order="sequence,name desc">
    
     toolbar 设置为True,会显示层次结构像菜单一样
     fonts  按条件设定记录的字体
     create,delete,edit  若设置为false 则为不可用
     string :视图的标签,是可以翻译的
    
     允许元素:
      field, group, separator, tree, button, filter, newline
    # 当不要创建和编辑功能时,在<tree> 加上 create="0"
   
    看一下树状列表,以产品分类为例:
            <record id="product_category_tree_view" model="ir.ui.view">
            <field name="name">product.category.tree</field>
            <field name="model">product.category</field>
            <!--父子 -->
            <field name="field_parent">child_id</field>
            <field name="arch" type="xml">
                <tree toolbar="True" string="Product Categories">
                    <field name="name"/>
                </tree>
            </field>
        </record>
        <!--产品分类动作 -->
        <record id="product_category_action" model="ir.actions.act_window">
            <field name="name">Products by Category</field>
            <field name="type">ir.actions.act_window</field>
            <field name="res_model">product.category</field>
            <field name="domain">[('parent_id','=',False)]</field>
            <!--采用树状-->
            <field name="view_type">tree</field>
            <field name="view_id" ref="product_category_tree_view"/>
            <field name="help" type="html">
              <p>
                Here is a list of all your products classified by category. You
                can click a category to get the list of all products linked to
                this category or to a child of this category.
              </p>
            </field>
        </record>
        @可以看到:<field name="view_type">tree</field>
        @视图中用到 <field name="field_parent">child_id</field>
       
         
    
* 搜索视图
   代码如下(产品模板的搜索视图):

    <record id="product_template_search_view" model="ir.ui.view">
        <field name="name">product.template.search</field>
        <field name="model">product.template</field>
        <field name="arch" type="xml">
            <search string="Product">
                <!--产品搜索-->
                <field name="name" string="Product" filter_domain="['|',('default_code','ilike',self),('name','ilike',self)]"/>
                <!--过滤-->
                <filter string="Services" icon="terp-accessories-archiver" domain="[('type','=','service')]"/>
                <filter string="Consumable" name="consumable" icon="terp-accessories-archiver" domain="[('type','=','consu')]" help="Consumable products"/>
                <separator/>
                <!--可销售-->
                <filter string="Can be Sold" name="filter_to_sell" icon="terp-accessories-archiver-minus" domain="[('sale_ok','=',1)]"/>
               <!--产品分类-->
                <field name="categ_id" filter_domain="[('categ_id', 'child_of', self)]"/>
                <field string="Product Variant" name="product_variant_ids" filter_domain="['|', ('product_variant_ids.name','ilike',self), ('product_variant_ids.attribute_value_ids.attribute_id.name','ilike',self)]"/>
                <!--公司-->
                <field name="company_id"/>
                <!--价格表-->
                <field name="pricelist_id" widget="selection" context="{'pricelist': self}" filter_domain="[]" groups="product.group_sale_pricelist"/> <!-- Keep widget=selection on this field to pass numeric `self` value, which is not the case for regular m2o widgets! -->

                <group  expand='0' string='Group by...'>
                    <!--分类-->
                   <filter string='Category' domain="[]" context="{'group_by' : 'categ_id'}"/>
                    <!--默认计量单位-->
                   <filter string='Default Unit of Measure' icon="terp-mrp" domain="[]" context="{'group_by' : 'uom_id'}"/>
                    <!--类型-->
                   <filter string='Type' icon="terp-stock_symbol-selection" domain="[]" context="{'group_by' : 'type'}"/>
                </group>
            </search>
        </field>
    </record>
  
   # <search>
      上面用于搜索字段 name  default_code
     
      field元素定义的属性
           name 字段标识名
           string 标签名
           operator 操作符
          
           filter_domain  对应字段如何对输入内容搜索
           上面的例子基于name字段搜索时,输入框内容和 name 和 default_code都比较
           有一个满足就可以
          
           context 添加上下文
           groups 指定用户权限可见
           widget 物件(selection 仅为 Many2one 字段)
           domain 过滤条件(用于Mnay2one字段)
      
      
       filter元素支持属性(过滤器)
            name 指定动作标识符 (在模型中会定义的方法名)
            string 标签名
            domain 过滤记录按条件
            content 上下文
            helper 提示信息
            groups 指定用户权限可见
            icon  小图标
           
       separator 用于分隔       
   
       group 用于分组
         它的单项类似filter,但重点用了context来分组  
                

* 日历视图
   <record id="view_calendar_todo_task" model="ir.ui.view">
       <field name="name">view_calendar_todo_task</field>
       <field name="model">todo.task</field>
       <field name="arch" type="xml">
           <calendar date_start="date_deadline" color="user_id"
                     display="[name], Stage [stage_id]">
               <!-- Fields used for the text of display attribute -->
               <field name="name"/>
               <field name="stage_id"/>
           </calendar>
       </field>
   </record>
   #<calendar>
     date_start  开始日期
     date_stop  结束日期
     date_delay  延期
     color  用颜色的实体
     event_open_popup 用弹出容器代替表单视图
     quick_add
     all_day 总天数
     display 显示文字
     attendee 出席人
    
* 甘特视图
   <record id="view_gantt_todo_task" model="ir.ui.view">
       <field name="name">view_gantt_todo_task</field>
       <field name="model">todo.task</field>
       <field name="arch" type="xml">
           <gantt date_start="date_deadline"
                  default_group_by="user_id"/>
       </field>
   </record>    
  
   #<gantt>
     date_start  开始日期
     date_stop  结束日期
     date_delay  周期
     progress  进度百分比
     default_group_by 任务分组条件
   
* 图表视图
    <record id="view_graph_todo_task" model="ir.ui.view">
       <field name="name">view_graph_todo_task</field>
       <field name="model">todo.task</field>
       <field name="arch" type="xml">
           <graph type="pivot">
               <field name="stage" type="col"/>
               <field name="user_id"/>
               <field name="date_deadline" interval="week"/>
               <field name="effort_estimate" type="measure"/>
           </graph>
       </field>
   </record>    
    #<graph>
      type:bar(默认),pie,line,pivot 图形类别
      stacked: 仅用于 bar
      orientation:horizontal,vertical 方向:水平 垂直
      字段支持的属性<field>
     name 字段标识名
     type 分组方式,默认是row 行 ,设定可以是col列,measure
     interval 时间度量 day,week,month,quarter 或 year
    
* 看板
     todo_task.py
   -----------------------------
    # -*- coding: utf-8 -*-

    from openerp import models, fields


    class TodoTask(models.Model):
         _inherit = 'todo.task'
         priority = fields.Selection(
                [('0', 'Low'), ('1', 'Normal'), ('2', 'High')],
                'Priority', default='1')
         kanban_state = fields.Selection(
                [('normal', 'In Progress'),
                 ('blocked', 'Blocked'),
                 ('done', 'Ready for next stage')],
                'Kanban State', default='normal')
    ----------------------------------
   
    todo_view.xml
    ----------------------------------
    <?xml version="1.0"?>
    <openerp>
        <data>
            <!-- Add Kanban view mode to the menu Action: -->
            <act_window id="todo_app.action_todo_task"
                        name=" To-Do Tasks"
                        res_model="todo.task"
                        view_mode="kanban,tree,form,calendar,gantt,graph"
                        context="{'search_default_filter_my_tasks': True}"/>
            <!-- Add Kanban view -->
            <record id="To-do Task Kanban" model="ir.ui.view">
                <field name="name">To-do Task Kanban</field>
                <field name="model">todo.task</field>
                <field name="arch" type="xml">
                    <!-- Empty for now, but the Kanban will go here! -->
                    <kanban>
                        <templates>
                            <t t-name="kanban-box">
                                <div class="oe_kanban_vignette">
                                    <img t-att-src="kanban_image('res.partner','image_medium', record.id.value)"
                                         class="oe_kanban_image"/>
                                    <div class="oe_kanban_details">
                                        <!-- Title and Data content -->
                                        <h4>
                                            <a type="open">
                                                <field name="name"/>
                                            </a>
                                        </h4>
                                        <field name="tags"/>
                                        <ul>
                                            <li>
                                                <field name="user_id"/>
                                            </li>
                                            <li>
                                                <field name="date_deadline"/>
                                            </li>
                                        </ul>
                                        <field name="kanban_state" widget="kanban_state_selection"/>
                                        <field name="priority" widget="priority"/>
                                    </div>
                                </div>
                            </t>
                        </templates>
                    </kanban>
                </field>
            </record>
        </data>
    </openerp>
   
    ----------------------------------
    # 可用属性
     default_group_by 默认分组
     default_order  默认排序字段
     class   类名
     quick_create
     可加载的字元素属
      field字段
        name 名称  sum,avg,min, max, count
      template 是一个qweb模板
        instance widget  record read_only_mode     
   
    # 看板视图的风格
      Vignette 就是左边有一个小插件,在系统中常用于 客户、产品、等
      Card  常用于 CRM中的机会和商机,也用在项目任务
    
    # 看板视图
      <kanban> 顶级元素
      常用两个字段 priority , kanban_state   
      priority 显示项目的紧急程度
      kanban_state    显示项目的各阶段
      <templates> 主要有一个命名 kanban-box   采用下面的元素 <t t-name="kanban-box">...</t>
      <field>这是字段,若是只是用于表达,就放在<templates>之前,字段具有聚合性如:
        <field name="effort_estimated"  sum="Total Effort" />
       <kanban> 顶级元素支持一些属性
         default_group_by  默认分组条件
         default_order 默认排序条件
         quick_create="false"    不能快速创建选项
         class  加入css类
       css类,有两个主要的 oe_kanban_vignette  和 oe_kanban_details
       <img>中的 t-att-src 可以计算图片的src存在数据库中的值
      
    # 在看板视图中的动作
       在<a>中有 type 属性 像 <button>里面的功能似的
        type的值有:
         open: 打开表单视图
         edit: 打开并编辑表单视图
         delete: 删除记录
        
    #卡片风格的看板视图
      示例代码:
        <t t-name="kanban-box">
          <div class="oe_kanban_card">
            <div class="oe_dropdown_kanban oe_dropdown_toggle">
                   <!-- Top-right drop down menu -->
            </div>
            <div class="oe_kanban_content">
                <!-- Option menu will go here! -->
                
                <h4>
                    <a type="open">
                        <field name="name"/>
                    </a>
                </h4>
                <field name="tags"/>
                <ul>
                    <li>
                        <field name="user_id"/>
                    </li>
                    <li>
                        <field name="date_deadline"/>
                    </li>
                </ul>
                <div class="oe_kanban_bottom_right">
                    <field name="kanban_state"
                           widget="kanban_state_selection"/>
                </div>
                <div class="oe_kanban_footer_left">
                    <field name="priority" widget="priority"/>
                </div>
            </div>
          </div>
        </t>   
    
* 综合来一例
    # 模型 lesson.py
    # -*- coding: utf-8 -*-

    from openerp import models, fields
    class oecn_training_lesson(models.Model):

        _name = 'oecn.training.lesson'

        _description = u'OECN 培训课程'

        _columns = {

            'name':fields.char( u'课程名',size=64,select=True),

            'date_start':fields.date(u'开始日期',select=True),

            'total_day':fields.float(u'总天数',digits=(16,1)),

            'teacher':fields.many2one('res.users',u'授课老师'),

            'students':fields.many2many('res.partner',string=u'学生'),

            'price':fields.float(u'价格',digits=(16,2)),

        }
       
        u后面再加中文,这样不会乱码,显性指明编码
       
    #视图lesson_view.xml   
    <?xml version="1.0" encoding="utf-8"?>
     <openerp>
      <data>
        <!--定义表单视图-->
        <record model="ir.ui.view" id="oecn_training_lesson_form_view">
            <field name="name">课程表单</field>
            <field name="type">form</field>
            <field name="model">oecn.training.lesson</field>
            <field name="arch" type="xml">
                <form>
                  <group>
                    <field name="name"/>
                    <field name="date_start"/>
                    <field name="total_day"/>
                    <field name="price"/>
                    <field name="teacher"/>
                    <field name="students" colspan="4"/>
                 </group>
                </form>
            </field>
        </record>
       
        <!--定义列表视图-->
        <record model="ir.ui.view" id="oecn_training_lesson_tree_view">
            <field name="name">课程列表</field>
            <field name="type">tree</field>
            <field name="model">oecn.training.lesson</field>
            <field name="arch" type="xml">
                <tree>
                    <field name="name"/>
                    <field name="date_start"/>
                    <field name="teacher"/>
                    <field name="price" sum="合计"/>
                </tree>
            </field>
        </record>
       
       <!--定义视图动作-->
        <record model="ir.actions.act_window"  id="action_oecn_training_lesson">
            <field name="name">课程</field>
            <field name="res_model">oecn.training.lesson</field>
            <field name="view_type">form</field>
            <field name="view_mode">form,tree</field>
            <field name="view_id" ref="oecn_training_lesson_tree_view"/>
        </record>
       
       <!--定义菜单-->
        <menuitem id="oecn_menu" name="OECN"/>
        <menuitem id="oecn_training_menu" name="OECN Training" parent="oecn_menu"/>
        <menuitem id="oecn_training_lesson_menu" name="OECN Training Lesson" parent="oecn_training_menu"
            action="action_oecn_training_lesson"/> 
    
    </data>
   </openerp>
  
   这样从菜单到响应完成,打开展开一个表单视图
   视图动作到了8.0可以用采用 <act_window>标签来简写
   ====================================
   接着继承插件写法,给课程分配教室
   # 模型 lesson.py
    # -*- coding: utf-8 -*-

    from openerp import models, fields
    class oecn_training_lesson(models.Model):

        _inherit = 'oecn.training.lesson'

        _columns = {

            'classroom_id':fields.many2one('oecn.training.classroom','教室'),#添加一个教室属性,为多对一对象。

        }
       
     教室模型 classroom.py
    # -*- coding: utf-8 -*-

    from openerp import models, fields
    class oecn_training_classroom(models.Model):

        _name = 'oecn.training.classroom'
        _description = u'OECN 教室'
        _columns = {
            'number':fields.char(u'编号', size=64, select=True),
            'capacity':fields.integer(u'容纳人数', select=True),
            'location':fields.char(u'地点', size=125, select=True),
        }    
  
    #视图lesson_view.xml   
    <?xml version="1.0" encoding="utf-8"?>
        <openerp>
            <data>
                <!--OECN 教室-->
                    <record model="ir.ui.view" id="oecn_training_lesson_from_inherit_classroom_view">
                    <field name="name">课程教室继承视图</field>
                    <field name="type">form</field>
                    <field name="model">oecn.training.lesson</field>
                    <field name="inherit_id" ref="oecn_training.oecn_training_lesson_form_view"/>
                    <field name="arch" type="xml">
                        <field name="name" position="after">
                            <field name="classroom_id"/>
                        </field>
                    </field>
                </record>
               
            </data>
        </openerp>
       
     <field name="inherit_id" ref="oecn_training.oecn_training_lesson_form_view"/>
     这句就是继承,ref指名要继承的视图id
     后面就找位置
     <field name="name" position="after">
            <field name="classroom_id"/>
     </field>
     意思是在要继承的视图的name字段后(position="after"),添加一个教室字段。
    
     # 教室的视图
     <?xml version="1.0" encoding="utf-8"?>
        <openerp>
            <data>
                <!--OECN 教室-->
                    <record model="ir.ui.view" id="oecn_training_classroom_from_view">
                    <field name="name">教室</field>
                    <field name="type">form</field>
                    <field name="model">oecn.training.classroom</field>
                    <field name="arch" type="xml">
                        <field name="number"/>
                        <field name="capacity"/>
                        <field name="location" />
                    </field>
                </record>
                 <!--定义列表视图-->
                <record model="ir.ui.view" id="oecn_training_classroom_tree_view">
                    <field name="name">教室列表</field>
                    <field name="type">tree</field>
                    <field name="model">oecn.training.classroom</field>
                    <field name="arch" type="xml">
                        <field name="number"/>
                        <field name="capacity"/>
                        <field name="location" />
                    </field>
                </record>
                <!--定义视图动作-->
                <record model="ir.actions.act_window"  id="action_oecn_training_classroom">
                    <field name="name">教室</field>
                    <field name="res_model">oecn.training.classroom</field>
                    <field name="view_type">form</field>
                    <field name="view_mode">form,tree</field>
                    <field name="view_id" ref="oecn_training_classroom_tree_view"/>
                </record>
                <menuitem id="oecn_training_classroom_menu" name="OECN Training Classroom"
                    parent="oecn_training.oecn_training_menu" action="action_oecn_training_classroom"/>  
            </data>
        </openerp>

* 代码分析时有用点
    <!--仅在编辑时显示标签-->
    <label for="name" class="oe_edit_only"/>
   
    <!--多公司时显示公司名-->
     <field name="company_id" groups="base.group_multi_company" widget="selection"/>
   
   
* css的运用
    oe_grey:灰色
   
    <p class="oe_grey">Select the places where this route can be selected</p>

* 设置context的值
   <filter name="show_inactive" string="Show inactive" context="{'active_test': False}" />
   这样表示,不测试 active这个字段 显示列表数据,系统默认所有在列表时,若有active这个字段
   显示是都是 active 为t时才能显示出来
   <field name="name" context="{default_name:'hello'}" /> 设置默认值

posted @ 2016-02-02 11:33  toby2chen  阅读(4361)  评论(0编辑  收藏  举报