mirror of
https://github.com/abapGit/abapGit.git
synced 2025-04-30 20:03:20 +08:00
382 lines
11 KiB
ABAP
382 lines
11 KiB
ABAP
CLASS zcl_abapgit_object_sfpf DEFINITION
|
|
PUBLIC
|
|
INHERITING FROM zcl_abapgit_objects_super
|
|
FINAL
|
|
CREATE PUBLIC .
|
|
|
|
PUBLIC SECTION.
|
|
|
|
INTERFACES zif_abapgit_object .
|
|
|
|
CLASS-METHODS fix_oref
|
|
IMPORTING
|
|
!ii_document TYPE REF TO if_ixml_document
|
|
RAISING
|
|
zcx_abapgit_exception .
|
|
PROTECTED SECTION.
|
|
PRIVATE SECTION.
|
|
|
|
CONSTANTS c_layout_file_ext TYPE string VALUE 'xdp'.
|
|
|
|
METHODS:
|
|
load
|
|
RETURNING VALUE(ri_wb_form) TYPE REF TO if_fp_wb_form
|
|
RAISING zcx_abapgit_exception,
|
|
form_to_xstring
|
|
RETURNING VALUE(rv_xstr) TYPE xstring
|
|
RAISING zcx_abapgit_exception.
|
|
|
|
ENDCLASS.
|
|
|
|
|
|
|
|
CLASS zcl_abapgit_object_sfpf IMPLEMENTATION.
|
|
|
|
|
|
METHOD fix_oref.
|
|
|
|
* During serialization of a SFPF / SFPI object the interface hierarchy
|
|
* is represented by attributes "id" and "href", where the id looks
|
|
* like "o<number>" and href like "#o<number>". Every run of
|
|
* serialization generates a new <number> in these attributes, that
|
|
* leads to differences even by comparing of untouched forms.
|
|
* The purpose of this method is to renumber the id's consequentially
|
|
* and therefore to avoid fictive differences.
|
|
|
|
* NB: As the method iterator->get_next() works quite slowly,
|
|
* it is better to collect all attributes in a cache table
|
|
* instead of implementing of a nested loop using get_next().
|
|
|
|
DATA:
|
|
li_iterator TYPE REF TO if_ixml_node_iterator,
|
|
li_elem TYPE REF TO if_ixml_element,
|
|
lv_new TYPE string,
|
|
lv_old TYPE string,
|
|
lv_count TYPE i,
|
|
BEGIN OF ls_attr_href,
|
|
val TYPE string,
|
|
attr TYPE REF TO if_ixml_attribute,
|
|
END OF ls_attr_href,
|
|
lt_attr_href LIKE SORTED TABLE OF ls_attr_href WITH NON-UNIQUE KEY val.
|
|
|
|
FIELD-SYMBOLS <ls_attr_href> LIKE LINE OF lt_attr_href.
|
|
|
|
* Collect all attributes href='#o...' in the cache table
|
|
li_iterator = ii_document->create_iterator_filtered(
|
|
ii_document->create_filter_and(
|
|
filter1 = ii_document->create_filter_node_type( if_ixml_node=>co_node_element )
|
|
filter2 = ii_document->create_filter_attribute( 'href' ) ) ).
|
|
li_elem ?= li_iterator->get_next( ).
|
|
WHILE li_elem IS NOT INITIAL.
|
|
ls_attr_href-attr = li_elem->get_attribute_node( 'href' ).
|
|
ls_attr_href-val = ls_attr_href-attr->get_value( ).
|
|
IF ls_attr_href-val CP '##o*'.
|
|
INSERT ls_attr_href INTO TABLE lt_attr_href.
|
|
ENDIF.
|
|
li_elem ?= li_iterator->get_next( ).
|
|
ENDWHILE.
|
|
|
|
* Renumber id='o...' attributes
|
|
li_iterator = ii_document->create_iterator_filtered(
|
|
ii_document->create_filter_and(
|
|
filter1 = ii_document->create_filter_node_type( if_ixml_node=>co_node_element )
|
|
filter2 = ii_document->create_filter_attribute( 'id' ) ) ).
|
|
li_elem ?= li_iterator->get_next( ).
|
|
WHILE li_elem IS NOT INITIAL.
|
|
lv_old = li_elem->get_attribute( 'id' ).
|
|
IF lv_old CP 'o*'.
|
|
lv_count = lv_count + 1.
|
|
lv_new = |o{ lv_count }|.
|
|
* Rewrite id
|
|
IF li_elem->set_attribute( name = 'id'
|
|
value = lv_new ) IS NOT INITIAL.
|
|
zcx_abapgit_exception=>raise( 'SFPF error, FIX_OREF' ).
|
|
ENDIF.
|
|
* Update references
|
|
LOOP AT lt_attr_href ASSIGNING <ls_attr_href> WHERE val = '#' && lv_old.
|
|
IF <ls_attr_href>-attr->set_value( '#' && lv_new ) IS NOT INITIAL.
|
|
zcx_abapgit_exception=>raise( 'SFPF error, FIX_OREF' ).
|
|
ENDIF.
|
|
ENDLOOP.
|
|
ENDIF.
|
|
li_elem ?= li_iterator->get_next( ).
|
|
ENDWHILE.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD form_to_xstring.
|
|
|
|
CONSTANTS: lc_empty_data TYPE xstring VALUE ''.
|
|
|
|
DATA: li_fp_form TYPE REF TO if_fp_form,
|
|
li_wb_form TYPE REF TO if_fp_wb_form,
|
|
li_fp_layout TYPE REF TO if_fp_layout,
|
|
lx_fp_err TYPE REF TO cx_fp_api,
|
|
lx_fp_conv_err TYPE REF TO cx_fp_api,
|
|
lv_layout_data TYPE xstring.
|
|
|
|
li_wb_form = load( ).
|
|
li_fp_form ?= li_wb_form->get_object( ).
|
|
li_fp_layout = li_fp_form->get_layout( ).
|
|
lv_layout_data = li_fp_layout->get_layout_data( ).
|
|
|
|
mo_files->add_raw(
|
|
iv_ext = c_layout_file_ext
|
|
iv_data = lv_layout_data ).
|
|
|
|
TRY.
|
|
li_fp_layout->set_layout_data( i_layout_data = lc_empty_data
|
|
i_set_xliff_ids = abap_false ).
|
|
CATCH cx_fp_api INTO lx_fp_err.
|
|
zcx_abapgit_exception=>raise( |SFPF remove layout: { lx_fp_err->get_text( ) }| ).
|
|
ENDTRY.
|
|
|
|
TRY.
|
|
rv_xstr = cl_fp_helper=>convert_form_to_xstring( li_fp_form ).
|
|
CATCH cx_fp_api INTO lx_fp_conv_err.
|
|
" Pass - the exception is handled below!
|
|
ENDTRY.
|
|
|
|
TRY.
|
|
li_fp_layout->set_layout_data( i_layout_data = lv_layout_data
|
|
i_set_xliff_ids = abap_false ).
|
|
CATCH cx_fp_api INTO lx_fp_err.
|
|
" Be aware that there might be another exception
|
|
" raised by cl_fp_helper=>convert_form_to_xstring( )
|
|
zcx_abapgit_exception=>raise( |SFPF recover layout: { lx_fp_err->get_text( ) }| ).
|
|
ENDTRY.
|
|
|
|
IF lx_fp_conv_err IS BOUND.
|
|
" This statement handles the exception raised from cl_fp_helper=>convert_form_to_xstring( )
|
|
zcx_abapgit_exception=>raise( |SFPF convert_form_to_xstring: { lx_fp_conv_err->get_text( ) }| ).
|
|
ENDIF.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD load.
|
|
|
|
DATA: lv_name TYPE fpname.
|
|
|
|
|
|
lv_name = ms_item-obj_name.
|
|
|
|
TRY.
|
|
ri_wb_form = cl_fp_wb_form=>load( lv_name ).
|
|
CATCH cx_fp_api.
|
|
zcx_abapgit_exception=>raise( 'SFPF error, load' ).
|
|
ENDTRY.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~changed_by.
|
|
|
|
SELECT SINGLE lastuser FROM fplayout
|
|
INTO rv_user
|
|
WHERE name = ms_item-obj_name
|
|
AND state = 'A'.
|
|
IF rv_user IS INITIAL.
|
|
SELECT SINGLE firstuser FROM fplayout
|
|
INTO rv_user
|
|
WHERE name = ms_item-obj_name
|
|
AND state = 'A'.
|
|
ENDIF.
|
|
IF rv_user IS INITIAL.
|
|
rv_user = c_user_unknown.
|
|
ENDIF.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~delete.
|
|
|
|
DATA: lv_name TYPE fpname.
|
|
|
|
lv_name = ms_item-obj_name.
|
|
|
|
TRY.
|
|
TRY.
|
|
CALL METHOD cl_fp_wb_form=>('DELETE')
|
|
EXPORTING
|
|
i_name = lv_name
|
|
i_ordernum = iv_transport
|
|
i_dark = abap_true. " > 740
|
|
CATCH cx_sy_dyn_call_error.
|
|
cl_fp_wb_form=>delete(
|
|
i_name = lv_name
|
|
i_ordernum = iv_transport ).
|
|
ENDTRY.
|
|
CATCH cx_fp_api.
|
|
zcx_abapgit_exception=>raise( 'SFPI error, delete' ).
|
|
ENDTRY.
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~deserialize.
|
|
|
|
DATA: lv_xstr TYPE xstring,
|
|
lv_layout TYPE xstring,
|
|
lv_name TYPE fpname,
|
|
li_wb_object TYPE REF TO if_fp_wb_form,
|
|
li_form TYPE REF TO if_fp_form,
|
|
lx_fp_err TYPE REF TO cx_fp_api.
|
|
|
|
|
|
lv_name = ms_item-obj_name.
|
|
lv_xstr = cl_ixml_80_20=>render_to_xstring( io_xml->get_raw( ) ).
|
|
|
|
TRY.
|
|
li_form = cl_fp_helper=>convert_xstring_to_form( lv_xstr ).
|
|
|
|
IF mo_files->contains_file( c_layout_file_ext ) = abap_true.
|
|
lv_layout = mo_files->read_raw( c_layout_file_ext ).
|
|
li_form->get_layout( )->set_layout_data( lv_layout ).
|
|
ENDIF.
|
|
|
|
IF zif_abapgit_object~exists( ) = abap_true.
|
|
TRY.
|
|
CALL METHOD cl_fp_wb_form=>('DELETE')
|
|
EXPORTING
|
|
i_name = lv_name
|
|
i_ordernum = iv_transport
|
|
i_dark = abap_true. " > 740
|
|
CATCH cx_sy_dyn_call_error.
|
|
cl_fp_wb_form=>delete(
|
|
i_name = lv_name
|
|
i_ordernum = iv_transport ).
|
|
ENDTRY.
|
|
ENDIF.
|
|
|
|
tadir_insert( iv_package ).
|
|
|
|
TRY.
|
|
CALL METHOD cl_fp_wb_form=>('CREATE')
|
|
EXPORTING
|
|
i_name = lv_name
|
|
i_form = li_form
|
|
i_ordernum = iv_transport
|
|
i_dark = abap_true " > 740
|
|
RECEIVING
|
|
r_wb_form = li_wb_object.
|
|
CATCH cx_sy_dyn_call_error.
|
|
li_wb_object = cl_fp_wb_form=>create(
|
|
i_name = lv_name
|
|
i_form = li_form
|
|
i_ordernum = iv_transport ).
|
|
ENDTRY.
|
|
|
|
li_wb_object->save( ).
|
|
li_wb_object->free( ).
|
|
CATCH cx_fp_api INTO lx_fp_err.
|
|
zcx_abapgit_exception=>raise( |SFPF deserialization error: { lx_fp_err->get_text( ) }| ).
|
|
ENDTRY.
|
|
|
|
zcl_abapgit_objects_activation=>add_item( ms_item ).
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~exists.
|
|
|
|
DATA: lv_name TYPE fpname.
|
|
|
|
SELECT SINGLE name FROM fplayout
|
|
INTO lv_name
|
|
WHERE name = ms_item-obj_name
|
|
AND state = 'A'.
|
|
rv_bool = boolc( sy-subrc = 0 ).
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~get_comparator.
|
|
RETURN.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~get_deserialize_order.
|
|
RETURN.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~get_deserialize_steps.
|
|
APPEND zif_abapgit_object=>gc_step_id-abap TO rt_steps.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~get_metadata.
|
|
rs_metadata = get_metadata( ).
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~is_active.
|
|
rv_active = is_active( ).
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~is_locked.
|
|
|
|
DATA: lv_object TYPE seqg3-garg.
|
|
|
|
lv_object = |{ ms_item-obj_name }|.
|
|
OVERLAY lv_object WITH ' '.
|
|
lv_object = lv_object && '*'.
|
|
|
|
rv_is_locked = exists_a_lock_entry_for( iv_lock_object = 'EFPFORM'
|
|
iv_argument = lv_object ).
|
|
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~jump.
|
|
" Covered by ZCL_ABAPGIT_OBJECTS=>JUMP
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~map_filename_to_object.
|
|
RETURN.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~map_object_to_filename.
|
|
RETURN.
|
|
ENDMETHOD.
|
|
|
|
|
|
METHOD zif_abapgit_object~serialize.
|
|
|
|
DATA: lv_xstr TYPE xstring,
|
|
li_document TYPE REF TO if_ixml_document,
|
|
li_node_collection TYPE REF TO if_ixml_node_collection,
|
|
li_node_iter TYPE REF TO if_ixml_node_iterator,
|
|
li_node TYPE REF TO if_ixml_node,
|
|
li_node_new TYPE REF TO if_ixml_node,
|
|
li_node_parent TYPE REF TO if_ixml_node.
|
|
|
|
lv_xstr = form_to_xstring( ).
|
|
li_document = cl_ixml_80_20=>parse_to_document( stream_xstring = lv_xstr ).
|
|
|
|
* Clear CACHE_INFO
|
|
li_node_collection = li_document->get_elements_by_tag_name_ns( 'CACHE_INFO' ).
|
|
IF li_node_collection IS NOT INITIAL.
|
|
li_node_iter = li_node_collection->create_iterator( ).
|
|
DO.
|
|
li_node = li_node_iter->get_next( ).
|
|
IF li_node IS INITIAL.
|
|
EXIT.
|
|
ENDIF.
|
|
li_node_new = li_document->create_element_ns( 'CACHE_INFO' ).
|
|
li_node_parent = li_node->get_parent( ).
|
|
li_node_parent->replace_child( new_child = li_node_new
|
|
old_child = li_node ).
|
|
ENDDO.
|
|
ENDIF.
|
|
|
|
fix_oref( li_document ).
|
|
io_xml->set_raw( li_document->get_root_element( ) ).
|
|
|
|
ENDMETHOD.
|
|
ENDCLASS.
|