可编辑ALV的内表同步更新

【博主SAP/ABAP交流群53144545

本文所用ALV FUNCTION是REUSE_ALV_GRID_DISPLAY_LVC。该FUNCTION有一个USER_COMMAND参数,可以实现触发事件功能码判断,比如鼠标双击事件,点击保存按钮事件等。如果填此参数,会调用同名的FORM。

FORM USER_COMMAND USING R_UCOMM LIKE SY-UCOMM
                        RS_SELFIELD TYPE SLIS_SELFIELD .
  CASE  R_UCOMM.
    WHEN 'EXPORT'.
    WHEN 'PNT'.
    WHEN 'SAVE'.
    WHEN 'BACK'.
      LEAVE TO SCREEN 0.
    WHEN '&IC1'.
    WHEN 'CANCEL' OR 'EXIT'.
      LEAVE PROGRAM.
  ENDCASE.
ENDFORM .                    "user_command

 

在ALV输出表上改写某些单元的值后,鼠标双击或者点击保存按钮,改变后的数据会自动保存到输出的内表。如果此时想继续更新到数据库的话,可以在ALV输出前把内表(假设为内表A)里的内容复制到另外一个内表(假设为内表B)里去,当内表A数据变化了后,LOOP内表A和内表B,比较两者的工作区,在工作区不同时作UPDATE或者MODIFY处理。

 

注意:REUSE_ALV_GRID_DISPLAY_LVC有个参数IT_EVENTS里也有USER_COMMAND事件,作用和参数USER_COMMAND相同,代码如下:

FORM SET_ALV_EVENT .
  DATA L_EVENTS TYPE SLIS_ALV_EVENT.

  CALL FUNCTION 'REUSE_ALV_EVENTS_GET'
    EXPORTING
      I_LIST_TYPE     = 0
    IMPORTING
      ET_EVENTS       = IT_EVENTS
    EXCEPTIONS
      LIST_TYPE_WRONG = 1
      OTHERS          = 2.
  IF SY-SUBRC <> 0.
  ENDIF.
  CLEAR L_EVENTS.
  READ TABLE IT_EVENTS WITH KEY NAME = SLIS_EV_USER_COMMAND INTO L_EVENTS.
  IF SY-SUBRC = 0.
    MOVE 'USER_COMMAND' TO L_EVENTS-FORM.
    MODIFY IT_EVENTS INDEX SY-TABIX FROM L_EVENTS.
  ENDIF.
ENDFORM.                    " SET_ALV_EVENT

以上代码看似有些复杂,实际上主要就是把'USER_COMMAND'这个字符串赋值到IT_EVENTS这张内表里。然后在REUSE_ALV_GRID_DISPLAY_LVC的参数IT_EVENTS调用。

 

下面是自定义的菜单栏的场合:

PF_STATUS_SET是ALV输出界面上的菜单栏,如果不填此参数,那么会调用系统自带的菜单栏。如果填此参数,则会调用同名的FORM,在FORM里设置菜单栏。

FORM PF_STATUS_SET.
  SET PF-STATUS 'ZALVYM_STATUS'.
ENDFORM .

这里值得注意的就是保存按钮的功能码设置,是&DATA_SAVE,如果仅仅是设置成SAVE,那么之后ALV输出列表上变更数据并点击保存按钮时,改变后的数据将不被传输到输出内表里。实际上在系统自带的菜单栏里,保存按钮的功能码也是&DATA_SAVE。



后记:USE COMMAND似乎不能捕捉界面上的所有功能码,而且如果随便点击按钮的话,保存数据到内表的功能有时会失灵。

其实还有CLASS的方法可以实现此功能,面向对象的ALV也比传统ALV功能更为强大。


<think>我们正在讨论的是在SAPOOALV(面向对象的ALV)中如何设置整列可编辑。根据引用[3]中的提示,在传统的函数式ALV中,我们通过在字段目录(FieldCatalog)中设置某个字段的EDIT属性为'X'来使整列可编辑。那么在OOALV中,我们同样可以在字段目录中设置该属性来实现整列可编辑。步骤:1.创建OOALV控件(如CL_GUI_ALV_GRID)并设置数据。2.准备字段目录(FieldCatalog)。我们可以通过结构复制、直接构建等方式生成字段目录。3.在字段目录中,对需要编辑的列,将EDIT属性设置为'X'(即ABAP_TRUE)。4.在显示ALV之前,将字段目录传递给ALV控件。示例代码:假设我们有一个内GT_TABLE,并且我们想要将其中字段名为'FIELD1'的列设置为可编辑。首先,我们声明必要的变量:DATA:go_containerTYPEREFTOcl_gui_custom_container,go_alvTYPEREFTOcl_gui_alv_grid.创建容器和ALV控件:CREATEOBJECTgo_containerEXPORTINGcontainer_name='MY_CONTAINER'."假设在屏幕上有自定义控件名为MY_CONTAINERCREATEOBJECTgo_alvEXPORTINGi_parent=go_container.准备字段目录。我们可以使用函数'LVC_FIELDCATALOG_MERGE'来从内结构生成字段目录,然后修改我们需要编辑的列:DATA:lt_fcatTYPElvc_t_fcat.CALLFUNCTION'LVC_FIELDCATALOG_MERGE'EXPORTINGi_structure_name='ZMY_STRUCTURE'"内对应的结构名CHANGINGct_fieldcat=lt_fcat.然后,我们修改字段目录,将'FIELD1'的编辑属性设置为允许编辑:LOOPATlt_fcatASSIGNINGFIELD-SYMBOL(<fs_fcat>)WHEREfieldname='FIELD1'.<fs_fcat>-edit='X'."设置可编辑ENDLOOP.显示ALV:go_alv->set_table_for_first_display(EXPORTINGis_layout=lw_layout"可以设置一些布局参数CHANGINGit_outtab=gt_tableit_fieldcatalog=lt_fcat).注意:为了让编辑生效,我们还需要处理数据更改事件(如DATA_CHANGED事件)。当用户编辑单元格并离开时,ALV会触发该事件。我们需要在该事件中处理数据校验等逻辑。另外,根据引用[2]所述,我们也可以使用方法set_ready_for_input来设置整个ALV格进入编辑状态。但请注意,这个方法设置的是整个格的编辑状态,而不仅仅是某一列。如果我们要让整个格都可以编辑,那么可以调用:go_alv->set_ready_for_input(1)."1可编辑,0示不可编辑但是,即使格设置为可编辑状态,我们仍然需要在字段目录中指定哪些列是可编辑的,否则所有列默认不可编辑(除非字段目录中设置了EDIT属性为'X')。总结:在OOALV中设置整列可编辑的方法是在字段目录(FieldCatalog)中将该字段的EDIT属性设置为'X',同时确保整个ALV格处于可编辑状态(可以通过set_ready_for_input设置)。注意:如果我们既没有在字段目录中设置EDIT属性,也没有调用set_ready_for_input,那么格将是只读的。如果调用了set_ready_for_input(1),但字段目录中没有设置EDIT='X'的列,那么这些列仍然是只读的。相关问题:1.如何动态改变OOALV中某列的编辑状态?2.OOALV中如何处理单元格编辑后的数据校验?3.在OOALV中如何实现部分单元格可编辑而同一列其他单元格不可编辑?</think> ### SAP OO ALV 设置整列可编辑的实现方法 在SAP OO ALV(面向对象的ALV)中设置整列可编辑需要操作字段目录(Field Catalog)并控制编辑状态。以下是具体实现步骤: --- #### 步骤1:设置字段目录编辑属性 在创建ALV字段目录时,将需要可编辑的列属性设置为`EDIT = 'X'`: ```abap DATA: lt_fieldcat TYPE lvc_t_fcat. " 构建字段目录 CALL METHOD cl_alv_table_create=>create_field_catalog EXPORTING i_structure_name = 'ZMY_STRUCTURE' CHANGING ct_fieldcat = lt_fieldcat. " 设置特定列可编辑 LOOP AT lt_fieldcat ASSIGNING FIELD-SYMBOL(<fs_fieldcat>). CASE <fs_fieldcat>-fieldname. WHEN 'AUFNR'. " 订单号列 <fs_fieldcat>-edit = abap_true. " 设置整列可编辑 WHEN 'MATNR'. " 物料号列 <fs_fieldcat>-edit = abap_true. " 设置整列可编辑 ENDCASE. ENDLOOP. ``` **关键参数说明**: - `fieldname`:数据库字段名 - `edit = abap_true`:等同于传统ALV的`EDIT = 'X'`[^3] - 字段目录需传递给ALV的`set_table_for_first_display`方法 --- #### 步骤2:激活ALV编辑状态 必须调用`set_ready_for_input`方法激活整体编辑模式: ```abap " 创建ALV GRID实例 DATA(lo_alv) = NEW cl_gui_alv_grid( i_parent = lo_container ). " 显示数据 lo_alv->set_table_for_first_display( CHANGING it_fieldcatalog = lt_fieldcat it_outtab = lt_data ). " 激活编辑模式 lo_alv->set_ready_for_input( i_ready_for_input = 1 ). " 1=可编辑状态[^2] ``` --- #### 步骤3:处理单元格级例外(可选) 若需在可编辑列中实现**特定单元格只读**: ```abap LOOP AT lt_data ASSIGNING FIELD-SYMBOL(<fs_row>). IF <fs_row>-status = 'LOCKED'. " 状态为锁定 <fs_row>-style_aufnr = cl_gui_alv_grid=>mc_style_disabled. " 禁用订单号单元格 ENDIF. ENDLOOP. ``` **技术说明**: - `mc_style_disabled`:内置常量,使单元格灰显不可编辑 - 单元格样式会覆盖列级设置[^2] --- #### ⚠️ 重要注意事项 1. **数据校验必需**: ```abap " 注册数据变更事件 SET HANDLER lcl_handler=>on_data_changed FOR lo_alv. ``` 在`on_data_changed`方法中实现数据合法性校验 2. **与SALV的区别**: - SALV (Simple ALV) 原生不支持编辑模式[^1] - 需强制转换类型实现编辑: ```abap CAST cl_salv_model_grid( lo_salv )->set_editable( abap_true ) ``` 3. **保存数据**: 在编辑完成后调用`check_changed_data()`同步修改到内: ```abap lo_alv->check_changed_data( IMPORTING e_valid = lv_valid ). ``` --- ### 完整示例流程 1. 初始化容器 (`cl_gui_custom_container`) 2. 创建ALV Grid实例 (`cl_gui_alv_grid`) 3. 准备含`edit = abap_true`的字段目录 4. 调用`set_table_for_first_display` 5. 激活编辑状态 (`set_ready_for_input(1)`) 6. 实现事件处理器 (`on_data_changed`) > **最佳实践**:对于关键业务字段(如订单号),建议在字段目录设置`edit`的同时,在事件处理器中加入二次校验逻辑。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值