现在的位置: 首页 > 综合 > 正文

三个月精通ABAP(4)

2013年06月04日 ⁄ 综合 ⁄ 共 6648字 ⁄ 字号 评论关闭

用户增强: 所有的Enhancement在表MODSAP,用户增强大概有三类

MODSAP是Enhancement表, 而TFDIR是看是否此enhancement被激活,就看字段MAND是否是"C"而已

1. E Enhancement exits :就是常说的写User_exit

2. C GUI codes 没用过

3. 3. S Subscreens 屏幕增强

OK,其实上面不用coding, 通过search MODSAP发现EXIT_SAPMM06E_013 在Enhancement MM06E005中.

使用T-code :SMOD F8 test运行然后可激活EXIT_SAPMM06E_013 ,如果你确实找不到enhancement name和exit函数对应关系,也没关系. 使用SMOD自己建立一个Enhancement 比如叫

ZPOSO ,然后你将function 加入,它也会有错误提示告诉你它属于的enhancement,还有就是使用上面的code直接update .

你自己可写个code将所有的user_exit找出来,很容易的哟.如你完全明白这篇文章,我想你就理解了user_exit 和BAPI的使用. 建立BAPI并使用其它语言call同样很简单.

EXIT_SAPMM06E_013 include程序 ZXM06U44

在coding前使用SE37 test BAPI :BAPI_SALESORDER_CREATEFROMDATA

***注意各企业因为设置的fields status不同可能有所不同.

对ORDER_HEADER_IN只需下面几个参数

ORDER_ITEMS_IN只需要MATERIAL,REQ_QTY-(Order qty in sales units - 00009001 corresponds to 9.001,就是说call BAP REQ_QTY测试时= actual qty * 1000)

, COND_TYPE, COND_VALUE,(即对应到price很奇怪必须/10, 不知其他系统是否这样,此是注意点)

PO_METH_S(Ship-to party purchase order type,如果设置了必须输入的话),

ORDER_PARTNERS如果SP和SH相同的话,只要一条SP然后输入customer No就可,如果ship-to-party不同多家一条SH+customer No.吧.

还有一样要注意. AG-SP, RE-BP

-SH (SAP经常有这样的case,就是save在DB中的数据和display出来的数据有所不同, 为什么要这样, 不得而知,难怪SAP不推荐直接使用DB table 做报表,或者这是SAP AG的一个策略)

测试OK.开始coding on ZXM06U44, Source code 如下,log写在ZPOSO表中.

测试使用BAPI_SALESORDER_CHANGE

就是说它实际是call BAPI_SALESDOCUMENT_CHANGE的

你可知道ORDER_HEADER_INX-UPDATEFLAG为U时是update,为D时是Delete sales order,不选还不行.

好了下面是Source Code(仅供参考,大体框架OK,可能根据你的SAP实际情况做些调整) .

*----------------------------------------------------------------------*

* INCLUDE ZXM06U44

* 在company 1000建立PO时同时建立SO in 1000和PO in 2000 *

* 注意此程序会反复call .

*----------------------------------------------------------------------*

data:

ZORDER_HEADER_IN like BAPISDHEAD ,

ZORDER_ITEMS_IN like BAPIITEMIN occurs 0 with header line ,

ZORDER_PARTNERS like BAPIPARTNR occurs 0 with header line ,

ZSALESDOCUMENT like BAPIVBELN-VBELN ,

ZRETURN like BAPIRETURN .

Data:

ZPOHEADER like BAPIEKKOC ,

ZPOITEMS like BAPIEKPOC occurs 0 with header line,

ZPO_ITEM_SCHEDULES like BAPIEKET occurs 0 with header line,

ZPURCHASEORDER like BAPIEKKOC-PO_NUMBER .

tables :ZPOSO .

data zposolog like zposo occurs 0 with header line.

data : Zrepcall(1) type C. "防止反复调用BAPI_PO_CREATE.

Refresh : ZORDER_ITEMS_IN .

*** 实际应用根据I_EKKO-EBELN(其实从EBELN-PO No.可判断出plant)

*** 1.I_EKKO-EBELN前面5位用case语句判断

*** 2.根据I_EKKO-BUKRS(comp. code)和XEKPO-WERKS(plant)判断

*** company code 2000是专门用来采购的HK 公司.

*if I_EKKO-BUKRS <> '1000'.

* exit.

*endif.

check I_EKKO-BUKRS eq '1000'. "只对comp 1000适用.

***判断表ZPOSO有无PO no.无call BAPI_SALESORDER_CREATEFROMDATA create SO

*** 如果存在call BAPi BAPI_SALESORDER_CHANGE change SO.

***注意在ME22N change PO时间XEKPO内表中只保留了被改变line item的数据.

select single * from ZPOSO where EBELN eq I_EKKO-EBELN.

if sy-subrc eq 0 .

* Change Sales Order,Purchase Order留给你自己写不告诉你

*CALL FUNCTION 'BAPI_SALESORDER_CHANGE'

*CALL FUNCTION 'BAPI_PO_CHANGE'

else . "建立Purchase Order 和Sales Order

*** 从ZFLAG表中读取Zrepcall看是否是T, 不管使用什么方法总之不要反复

**在此处设置断点在第二次调用时将Zrepcall设成T退出.

**好象无法设置到那种真正的全局变量只好使用一个表罗

**你就会发现当在comp 1000建立PO同时也建立了SO并且comp 2000也有PO

*select single Zrepcall into Zrepcall from zflag .

if Zrepcall = 'T' .

exit .

endif.

***注意使用PO另一个user_exit 001将Zflag的zrepcall设置成F

***接下来的语句立即update zflag将zrepcall设置成T.

***----------------------------------------------***

*** 开始建立Purchase Order in company 2000 ***

***----------------------------------------------***

***----------------------------------------------***

*** 开始建立Sales Order in same company 1000 ***

***----------------------------------------------***

***根据公司实际更改ZOR是标准Sales order

***----Sales Order Header --- ***

ZORDER_HEADER_IN-DOC_TYPE = 'ZOR'.

ZORDER_HEADER_IN-SALES_ORG = '1100'.

ZORDER_HEADER_IN-DISTR_CHAN = '11'.

ZORDER_HEADER_IN-DIVISION = '11'.

ZORDER_HEADER_IN-CURRENCY = I_EKKO-WAERS. "Order currency

ZORDER_HEADER_IN-PURCH_NO = I_EKKO-EBELN .

***---Purchase Order Header ---***

ZPOHEADER-PURCH_ORG = I_EKKO-EKORG. "正式使用换2000的采购组织

ZPOHEADER-PUR_GROUP = I_EKKO-EKGRP . "正式使用换comp2000的采购组

ZPOHEADER-CO_CODE = '2000' .

ZPOHEADER-VENDOR = I_EKKO-LIFNR .

ZPOHEADER-DOC_TYPE = 'NB' . "I_EKKO-BSART.

***根据PO item写入SO item和另家公司的PO items

loop at XEKPO .

***----写SO items ---***

ZORDER_ITEMS_IN-ITM_NUMBER = XEKPO-EBELP .

* 使PO item NO和SO item No完全相对应,PO,SO item可manual input.

ZORDER_ITEMS_IN-MATERIAL = XEKPO-MATNR. "'08K2555'.

ZORDER_ITEMS_IN-REQ_QTY = XEKPO-MENGE .

ZORDER_ITEMS_IN-REQ_QTY = ZORDER_ITEMS_IN-REQ_QTY * 1000 .

* Call SO BAPI时Qty 测试时间必须乘上1000.ABAP中则不用

ZORDER_ITEMS_IN-COND_TYPE = 'ZPR1'. "ZPR1是EST定义的price cond.

ZORDER_ITEMS_IN-COND_VALUE = XEKPO-NETPR.

ZORDER_ITEMS_IN-COND_VALUE = ZORDER_ITEMS_IN-COND_VALUE / 10 .

*Call SO BAPI test时Price必须/10

ZORDER_ITEMS_IN-PO_METH_S = '010' . "表示是维修期,EST设置成必输字段.

Append ZORDER_ITEMS_IN .

***---写另公司的PO items,如果有必要让俩公司PO外部给号PONo可一致---***

ZPOITEMS-PUR_MAT = XEKPO-MATNR.

ZPOITEMS-PLANT = XEKPO-WERKS .

ZPOITEMS-NET_PRICE = XEKPO-NETPR.

***仅仅是test,注意XEKET,shedules delivery date 没item不同哟.自己去加强吧

ZPO_ITEM_SCHEDULES-DELIV_DATE = SY-DATUM . " XEKET-EINDT

ZPO_ITEM_SCHEDULES-QUANTITY = XEKPO-MENGE .

append ZPOITEMS .

append ZPO_ITEM_SCHEDULES .

endloop.

***为了简单就将SP-Sold_to_Party和SH-Ship_to_Party设置成相同.

***AG-SP:sold_to WE:SH Shipping-to,BAPI测试只要SP,call时则要求俩者

***下面假设PO vendor no就是SO 的customer no,如非就建立对应关系.

ZORDER_PARTNERS-PARTN_ROLE = 'AG'.

ZORDER_PARTNERS-PARTN_NUMB = I_EKKO-LIFNR.

***实际应用时HK开PO时产生的SO customer当然只有一个就是SZ公司.

Append ZORDER_PARTNERS .

ZORDER_PARTNERS-PARTN_ROLE = 'WE'.

ZORDER_PARTNERS-PARTN_NUMB = I_EKKO-LIFNR.

Append ZORDER_PARTNERS .

CALL FUNCTION 'BAPI_SALESORDER_CREATEFROMDATA'

EXPORTING

ORDER_HEADER_IN = ZORDER_HEADER_IN

IMPORTING

SALESDOCUMENT = ZSALESDOCUMENT

* SOLD_TO_PARTY =

* SHIP_TO_PARTY =

* BILLING_PARTY =

RETURN = ZRETURN

TABLES

ORDER_ITEMS_IN = ZORDER_ITEMS_IN

ORDER_PARTNERS = ZORDER_PARTNERS

* ORDER_ITEMS_OUT =

* ORDER_CFGS_REF =

* ORDER_CFGS_INST =

* ORDER_CFGS_PART_OF =

* ORDER_CFGS_VALUE =

* ORDER_CFGS_BLOB =

* ORDER_CCARD =

.

if ZSALESDOCUMENT <> '' .

loop at XEKPO .

zposolog-ebeln = I_EKKO-EBELN. "PO No.

zposolog-ebelp = XEKPO-EBELP. "PO item No.

zposolog-vbeln = ZSALESDOCUMENT."SO No.

zposolog-posnr = XEKPO-EBELP . "SO Item NO.

zposolog-waers = I_EKKO-WAERS . "Currency

zposolog-matnr = XEKPO-MATNR . "Material

zposolog-kwmeng = XEKPO-MENGE. "PO/SO qty

zposolog-netpr = XEKPO-netpr. "PO/SO price

zposolog-bapimsg = 'Creation OK' . "ZRETURN-MESSAGE .

zposolog-flag = 'T'. "Failure

append zposolog .

insert ZPOSO from zposolog.

endloop.

else.

zposolog-ebeln = I_EKKO-EBELN.

* zposolog-bapimsg = ZRETURN-MESSAGE .

***为了好SE91查找message合并message type, messge code,和message text

concatenate ZRETURN-TYPE '--' ZRETURN-CODE '--:' ZRETURN-MESSAGE into

zposolog-bapimsg .

zposolog-flag = 'F'. "Failure

append zposolog .

insert ZPOSO from zposolog.

endif.

***----------------------PO可能会反复调用吗---------------------

*** 不管如何做到call一次就退出.

*** 建立一表ZFLAG, 可只有一个字段, Zrepcall,default是F,

CALL FUNCTION 'BAPI_PO_CREATE'

EXPORTING

PO_HEADER = ZPOHEADER

* PO_HEADER_ADD_DATA =

* HEADER_ADD_DATA_RELEVANT =

* PO_ADDRESS =

* SKIP_ITEMS_WITH_ERROR = 'X'

* ITEM_ADD_DATA_RELEVANT =

IMPORTING

PURCHASEORDER = ZPURCHASEORDER "好瞧瞧

TABLES

PO_ITEMS = ZPOITEMS

* PO_ITEM_ADD_DATA =

PO_ITEM_SCHEDULES = ZPO_ITEM_SCHEDULES

* PO_ITEM_ACCOUNT_ASSIGNMENT =

* PO_ITEM_TEXT =

* RETURN =

* PO_LIMITS =

* PO_CONTRACT_LIMITS =

* PO_SERVICES =

* PO_SRV_ACCASS_VALUES =

* PO_SERVICES_TEXT =

* PO_BUSINESS_PARTNER =

* EXTENSIONIN =

* POADDRDELIVERY =

.

Endif. "结束建立Purchase Order 和Sales Order

抱歉!评论已关闭.