abapGit/src/objects/zcl_abapgit_object_sfpf.clas.abap
2023-12-18 17:16:00 -05:00

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.