-
Notifications
You must be signed in to change notification settings - Fork 289
/
zcl_demo_abap_date_time.clas.abap
1601 lines (1335 loc) · 64.2 KB
/
zcl_demo_abap_date_time.clas.abap
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
"! <p class="shorttext"><strong>Date, Time, and Time Stamp</strong><br/>ABAP cheat sheet example class</p>
"!
"! <p>The example class demonstrates the handling and procesing dates, times, and time stamps in ABAP.<br/>
"! Choose F9 in ADT to run the class.</p>
"!
"! <h2>Note</h2>
"! <p>Find information on <strong>getting started with the example class</strong> and the
"! <strong>disclaimer</strong> in the ABAP Doc comment of class {@link zcl_demo_abap_aux}.</p>
CLASS zcl_demo_abap_date_time DEFINITION
PUBLIC
FINAL
CREATE PUBLIC.
PUBLIC SECTION.
INTERFACES if_oo_adt_classrun.
PROTECTED SECTION.
PRIVATE SECTION.
ENDCLASS.
CLASS zcl_demo_abap_date_time IMPLEMENTATION.
METHOD if_oo_adt_classrun~main.
out->write( |ABAP Cheat Sheet Example: Date, Time, and Time Stamp\n\n| ).
**********************************************************************
out->write( |1) Retrieving the Time Zone\n\n| ).
"Retrieving the time zone of a given user
TRY.
DATA(tz_user) = cl_abap_context_info=>get_user_time_zone( ).
CATCH cx_abap_context_info_error.
ENDTRY.
"Using XCO
"The reference created without '->value' is used in some XCO method calls,
"as shown in some of the following code snippets.
"User's time zone
DATA(tz_user_xco) = xco_cp_time=>time_zone->user->value.
"UTC
DATA(tz_utc_xco) = xco_cp_time=>time_zone->utc->value.
out->write( data = tz_user name = `tz_user` ).
out->write( |\n| ).
out->write( data = tz_user_xco name = `tz_user_xco` ).
out->write( |\n| ).
out->write( data = tz_utc_xco name = `tz_utc_xco` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `2) Retrieving and Creating Dates` ) ).
"--------------------- Retrieving the current date --------------------
"Retrieving the current date with respect to UTC.
"The result's base type is the DDIC type dats that is mapped to the
"ABAP type d.
DATA(utc_date) = cl_abap_context_info=>get_system_date( ).
"Using XCO
"Notes:
"- The result of the following chained statement is of type string.
"- With the 'as' method, a given format (available via xco_cp_time=>format)
" is applied to the time.
"- The 'date' method has a parameter for specifying the time zone.
" By default, the user's time zone is used. For the specification, you
" can use xco_cp_time=>time_zone.
"- The examples explore multiple formatting options that are specified in the
" 'date' and 'as' methods.
"Various formatting options
DATA(date_xco_iso_basic) = xco_cp=>sy->date( )->as( xco_cp_time=>format->iso_8601_basic )->value.
DATA(date_xco_iso_ext) = xco_cp=>sy->date( )->as( xco_cp_time=>format->iso_8601_extended )->value.
DATA(date_xco_abap) = xco_cp=>sy->date( )->as( xco_cp_time=>format->abap )->value.
"Specifying the user time zone explicitly (it is the default value)
DATA(date_xco_expl_user_tz) = xco_cp=>sy->date( xco_cp_time=>time_zone->user
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Specifying UTC explicitly
DATA(date_xco_expl_utc) = xco_cp=>sy->date( xco_cp_time=>time_zone->utc
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"--------------------- Retrieving current date values (XCO) --------------------
DATA(this_day) = xco_cp=>sy->date( )->day.
DATA(this_month) = xco_cp=>sy->date( )->month.
DATA(this_year) = xco_cp=>sy->date( )->year.
"--------------------- Creating dates --------------------
DATA date_cr1 TYPE d.
date_cr1 = '20240101'.
DATA date_cr2 TYPE d VALUE '20240202'.
DATA(date_cr3) = CONV d( '20240303' ).
"Using XCO
"The result contains a reference. It can be used for further processing
"with the XCO library.
DATA(date_w_xco) = xco_cp_time=>date( iv_year = 2024
iv_month = 3
iv_day = 3 ).
"Examples
DATA(date_cr4) = date_w_xco->as( xco_cp_time=>format->iso_8601_extended )->value.
DATA(day_from_date) = date_w_xco->day.
DATA(month_from_date) = date_w_xco->month.
out->write( data = utc_date name = `utc_date` ).
out->write( |\n| ).
out->write( data = date_xco_iso_basic name = `date_xco_iso_basic` ).
out->write( |\n| ).
out->write( data = date_xco_iso_ext name = `date_xco_iso_ext` ).
out->write( |\n| ).
out->write( data = date_xco_abap name = `date_xco_abap` ).
out->write( |\n| ).
out->write( data = date_xco_expl_user_tz name = `date_xco_expl_user_tz` ).
out->write( |\n| ).
out->write( data = date_xco_expl_utc name = `date_xco_expl_utc` ).
out->write( |\n| ).
out->write( data = this_day name = `this_day` ).
out->write( |\n| ).
out->write( data = this_month name = `this_month` ).
out->write( |\n| ).
out->write( data = this_year name = `this_year` ).
out->write( |\n| ).
out->write( data = date_cr4 name = `date_cr4` ).
out->write( |\n| ).
out->write( data = day_from_date name = `day_from_date` ).
out->write( |\n| ).
out->write( data = month_from_date name = `month_from_date` ).
out->write( |\n| ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `3) Validity of Date Fields` ) ).
"Before accessing date (or time) fields, ensure that the content of these
"fields is valid to avoid unexpected results, such as incorrect calculations.
"The ABAP runtime framework checks the validity of these fields in various
"contexts, including lossless assignments and assignments to numeric fields.
"Lossless assignements using the EXACT operator
"Due to the invalid dates, an exception is raised.
TRY.
DATA(inv_date1) = EXACT d( '20240231' ).
CATCH cx_sy_conversion_no_date INTO DATA(error1).
out->write( data = error1->get_text( ) name = `error1->get_text( )` ).
out->write( |\n| ).
ENDTRY.
TRY.
DATA(inv_date2) = EXACT d( '2024XY31' ).
CATCH cx_sy_conversion_no_date INTO DATA(error2).
out->write( data = error2->get_text( ) name = `error2->get_text( )` ).
out->write( |\n| ).
ENDTRY.
"Assignment of an invalid date of type d to type i; the initial value
"is produced
"In newer ABAP releases, the following statement shows a
"syntax warning that the date (intentionally specified as
"invalid date here) does not match the type. Therefore,
"the example is provided differently to circumvent the
"syntax warning.
"DATA(inv_date3) = CONV i( CONV d( '20240231' ) ).
TYPES c8 TYPE c LENGTH 8.
DATA false_date TYPE c8 VALUE '20240231'.
DATA(inv_date3) = CONV i( CONV d( false_date ) ).
IF inv_date3 = 0.
out->write( `inv_date3 = 0` ).
ENDIF.
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `4) Character-Like Access to Date Fields` ) ).
"Since the content is character-like, you can use string processing functionalities
"to access date values. This also applies to time fields of type t.
"The following examples show a selection.
"Extracting date values using the string function substring.
DATA some_date TYPE d VALUE '20240102'.
DATA(year_sub) = substring( val = some_date off = 0 len = 4 ).
DATA(month_sub) = substring( val = some_date off = 4 len = 2 ).
DATA(day_sub) = substring( val = some_date off = 6 len = 2 ).
"Replacing using the string function replace
DATA(year_repl) = replace( val = some_date off = 0 len = 4 with = `2025` ).
"Offset and length specifications
DATA(off_len_spec_day) = some_date+6(2).
some_date+4(2) = '10'.
out->write( data = year_sub name = `year_sub` ).
out->write( |\n| ).
out->write( data = month_sub name = `month_sub` ).
out->write( |\n| ).
out->write( data = day_sub name = `day_sub` ).
out->write( |\n| ).
out->write( data = year_repl name = `year_repl` ).
out->write( |\n| ).
out->write( data = off_len_spec_day name = `off_len_spec_day` ).
out->write( |\n| ).
out->write( data = some_date name = `some_date` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `5) Numeric Access and Calculations` ) ).
"When converting date fields to numeric values, the type d produces an integer
"representing the number of days since 01.01.000
"This is especially important when using date fields in calculations and converting
"the values to numeric types.
"The XCO library offers methods for performing calculations. In the code snippet,
"examples use the value attribute. The return value of the method calls is of type
"string.
DATA date_calc_1 TYPE d VALUE '20240101'.
DATA date_calc_2 TYPE d VALUE '20231227'.
DATA date_calc_3 TYPE d VALUE '20231230'.
DATA date_calc_4 TYPE d VALUE '20240220'.
DATA(days_diff_1) = date_calc_1 - date_calc_2.
DATA(days_diff_2) = date_calc_1 - date_calc_3.
DATA(days_since_01_01_0001) = CONV i( date_calc_1 ).
"Getting the weekday (1 = Monday, 2 = Tuesday, ...)
DATA(weekday1) = ( 5 + date_calc_1 MOD 7 ) MOD 7 + 1.
DATA(weekday2) = ( 5 + date_calc_3 MOD 7 ) MOD 7 + 1.
DATA(date_w_first_day_of_month) = CONV d( replace( val = `202403020` off = 6 len = 2 with = `01` ) ).
DATA(date_w_last_day_of_prev_month) = CONV d( date_w_first_day_of_month - 1 ).
"------------ Performing date additions and subtractions using the XCO library ------------
"Adding days to the current date using the 'add' method
DATA(xco_date_add_5days) = xco_cp=>sy->date( )->add( iv_day = 5
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"The 'add' method has various parameters, adding 1 day/month/year
DATA(xco_date_add_1_mult) = xco_cp=>sy->date( )->add( iv_day = 1 iv_month = 1 iv_year = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Addition with a created date
DATA(xco_date_add_1day_custom) = xco_cp_time=>date( iv_year = 2024 iv_month = 02 iv_day = 28
)->add( iv_day = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Subtracting using the 'subtract' method
DATA(xco_date_subtr_1_mult) = xco_cp=>sy->date( )->subtract( iv_day = 1 iv_month = 1 iv_year = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Optional parameter io_calculation
"io_calculation parameter: xco_cp_time=>date_calculation->preserving,
"i.e. the date is calculated mathmatically and preserved. It is the
"default.
"In case of an invalid resulting date, an exception is raised.
TRY.
DATA(inv_date_a) = xco_cp_time=>date( iv_year = '2024'
iv_month = '08'
iv_day = '31'
)->add( iv_month = 1
io_calculation = xco_cp_time=>date_calculation->preserving
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
CATCH cx_root INTO DATA(err_pre).
ENDTRY.
"io_calculation parameter: xco_cp_time=>date_calculation->ultimo,
"Here, the actual last day of the month is considered. If the calculated
"date is invalid, the ultimo is automatically used. In the example, one month is added to
"the date. However, September does not have 31 days, so the result is adjusted to the actual
"last day of September, which is 30 in this case.
TRY.
DATA(inv_date_b) = xco_cp_time=>date( iv_year = '2024'
iv_month = '08'
iv_day = '31'
)->add( iv_month = 1
io_calculation = xco_cp_time=>date_calculation->ultimo
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
CATCH cx_root INTO DATA(err_ult).
ENDTRY.
out->write( data = days_diff_1 name = `days_diff_1` ).
out->write( |\n| ).
out->write( data = days_diff_2 name = `days_diff_2` ).
out->write( |\n| ).
out->write( data = days_since_01_01_0001 name = `days_since_01_01_0001` ).
out->write( |\n| ).
out->write( data = weekday1 name = `weekday1` ).
out->write( |\n| ).
out->write( data = weekday2 name = `weekday2` ).
out->write( |\n| ).
out->write( data = date_w_first_day_of_month name = `date_w_first_day_of_month` ).
out->write( |\n| ).
out->write( data = date_w_last_day_of_prev_month name = `date_w_last_day_of_prev_month` ).
out->write( |\n| ).
out->write( data = xco_date_add_5days name = `xco_date_add_5days` ).
out->write( |\n| ).
out->write( data = xco_date_add_1_mult name = `xco_date_add_1_mult` ).
out->write( |\n| ).
out->write( data = xco_date_add_1day_custom name = `xco_date_add_1day_custom` ).
out->write( |\n| ).
out->write( data = xco_date_subtr_1_mult name = `xco_date_subtr_1_mult` ).
out->write( |\n| ).
IF err_pre IS INITIAL.
out->write( data = inv_date_a name = `inv_date_a` ).
ELSE.
out->write( data = err_pre->get_text( ) name = `err_pre->get_text( )` ).
ENDIF.
out->write( |\n| ).
IF err_ult IS INITIAL.
out->write( data = inv_date_b name = `inv_date_b` ).
ELSE.
out->write( data = err_ult->get_text( ) name = `err_ult->get_text( )` ).
ENDIF.
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `6) CL_ABAP_DATFM: Date Conversions` ) ).
"Using the CL_ABAP_DATFM class, you can perform conversions with external
"and internal representations of a time according to the date format, e.g.
"the conversion of a date in a data object of type string to type d and vice
"versa using a specific date format. Multiple methods are available, of which
"two are covered in the example. For more information, refer to the class
"documentation.
"For the values of the date format for the conversion, see the fixed values
"by exploring the F2 help for cl_abap_datfm=>ty_datfm.
"Data type and obect declarations for the example. The example, explores
"multiple date formats. For this purpose, each loop pass in a DO loop
"uses a different date format. First, the internal time format is converted
"to external time formats. In another loop, the conversions are performed
"the other way round. The results are stored in an internal table.
DATA(date4conversion) = CONV d( '20240202' ).
DATA conv_date_str TYPE string.
DATA conv_date_d TYPE d.
DATA date_format TYPE cl_abap_datfm=>ty_datfm.
TYPES: BEGIN OF date_ty_s,
ext_date TYPE string,
format TYPE c LENGTH 1,
int_date TYPE d,
END OF date_ty_s.
DATA date_tab TYPE TABLE OF date_ty_s WITH EMPTY KEY.
"Conversion of d (internal) to string (external time format)
DO 7 TIMES.
TRY.
cl_abap_datfm=>conv_date_int_to_ext(
EXPORTING im_datint = date4conversion
im_datfmdes = SWITCH #( sy-index
WHEN 1 THEN cl_abap_datfm=>get_datfm( )
WHEN 2 THEN '1'
WHEN 3 THEN '2'
WHEN 4 THEN '3'
WHEN 5 THEN '6'
WHEN 6 THEN 'A'
WHEN 7 THEN 'C' )
IMPORTING ex_datext = conv_date_str
ex_datfmused = date_format ).
date_tab = VALUE #( BASE date_tab ( ext_date = conv_date_str format = date_format ) ).
CATCH cx_abap_datfm_format_unknown.
ENDTRY.
ENDDO.
"Conversion of string (external) to d (internal time format)
LOOP AT date_tab REFERENCE INTO DATA(ref_date).
TRY.
cl_abap_datfm=>conv_date_ext_to_int(
EXPORTING im_datext = ref_date->ext_date
im_datfmdes = ref_date->format
IMPORTING ex_datint = conv_date_d
ex_datfmused = date_format ).
ref_date->int_date = conv_date_d.
CATCH cx_abap_datfm_no_date cx_abap_datfm_invalid_date
cx_abap_datfm_format_unknown cx_abap_datfm_ambiguous.
ENDTRY.
ENDLOOP.
"Note: When outputting the data object with the classrun,
"20240202, which is of type d, is displayed as 2014-02-02 in
"the console for better readability.
out->write( data = date_tab name = `date_tab` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `7) Examples for Time Processing` ) ).
"The code snippet below provides examples of time processing, such
"as retrieving the current time, accessing time values, creating time
"values, and performing time calculations. You can also utilize the
"XCO library in different scenarios.
"--------------------- Retrieving the current time --------------------
"Retrieving the current time in UTC.
DATA(utc_time) = cl_abap_context_info=>get_system_time( ).
"Using XCO
"Note the optional time zone specification.
DATA(time_w_xco) = xco_cp=>sy->time( xco_cp_time=>time_zone->user
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"--------------------- Accessing time values --------------------
"Note: As mentioned in a previous section on dates, the access to time fields
"works similar as date fields. As an example, seconds, minutes, and hours are
"extracted from a time field.
DATA some_time TYPE t VALUE '123456'.
DATA(hour_extr) = substring( val = some_time off = 0 len = 2 ).
DATA(minute_extr) = substring( val = some_time off = 2 len = 2 ).
DATA(second_extr) = substring( val = some_time off = 4 len = 2 ).
"Retrieving the current seconds, minutes, hours using XCO
DATA(sec_w_xco) = xco_cp=>sy->time( xco_cp_time=>time_zone->user )->second.
DATA(min_w_xco) = xco_cp=>sy->time( xco_cp_time=>time_zone->user )->minute.
DATA(hour_w_xco) = xco_cp=>sy->time( xco_cp_time=>time_zone->user )->hour.
"--------------------- Creating times --------------------
DATA time_cr1 TYPE t.
time_cr1 = '095812'.
DATA time_cr2 TYPE t VALUE '112400'.
DATA(time_cr3) = CONV t( '120000' ).
"Using XCO
"The result contains a reference. It can be used for further processing
"with the XCO library.
DATA(xco_time) = xco_cp_time=>time( iv_hour = '08'
iv_minute = '34'
iv_second = '05' ).
"Examples
DATA(time_cr4) = xco_time->as( xco_cp_time=>format->iso_8601_extended )->value.
DATA(hours_from_xco_time) = xco_time->hour.
DATA(minutes_from_xco_time) = xco_time->minute.
DATA(seconds_from_xco_time) = xco_time->second.
"------------ Performing time calculations ------------
"Retrieving seconds, minutes, and hours from a time value in a data object
"of type t
DATA time4calc TYPE t VALUE '115708'.
DATA(seconds_total) = CONV i( time4calc ).
DATA(calc_hours) = seconds_total DIV 3600.
DATA(min_calc) = ( seconds_total - calc_hours * 3600 ) DIV 60.
DATA(sec_calc) = ( seconds_total - calc_hours * 3600 ) - min_calc * 60.
"Using the XCO library
"See the snippet above in the date section as well as the class documentation.
"Adding
DATA(time_xco_add) = xco_cp=>sy->time( xco_cp_time=>time_zone->user
)->add( iv_hour = 1 iv_minute = 1 iv_second = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Subtracting
DATA(time_xco_subtr) = xco_cp=>sy->time( xco_cp_time=>time_zone->user
)->subtract( iv_hour = 1 iv_minute = 1 iv_second = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"------------ Conversions with the CL_ABAP_TIMEFM class ------------
"Using the CL_ABAP_TIMEFM class, you can perform conversions with external
"and internal representations of a time, e.g. conversion of a time in a data
"object of type string to type t and vice versa. Multiple methods are available,
"of which two are covered in the example. For more information, refer to the
"class documentation.
DATA(time4conversion) = CONV t( '123456' ).
DATA conv_time_str TYPE string.
DATA conv_time_t TYPE t.
"Conversion of t (internal) to string (external time format)
TRY.
cl_abap_timefm=>conv_time_int_to_ext(
EXPORTING time_int = time4conversion
without_seconds = abap_false
format_according_to = cl_abap_timefm=>iso "hh:mm:ss
IMPORTING time_ext = conv_time_str ).
CATCH cx_parameter_invalid_range.
ENDTRY.
"Conversion of string (external) to t (internal time format)
TRY.
cl_abap_timefm=>conv_time_ext_to_int(
EXPORTING time_ext = conv_time_str
IMPORTING time_int = conv_time_t ).
CATCH cx_abap_timefm_invalid.
ENDTRY.
out->write( data = utc_time name = `utc_time` ).
out->write( |\n| ).
out->write( data = time_w_xco name = `time_w_xco` ).
out->write( |\n| ).
out->write( data = hour_extr name = `hour_extr` ).
out->write( |\n| ).
out->write( data = minute_extr name = `minute_extr` ).
out->write( |\n| ).
out->write( data = second_extr name = `second_extr` ).
out->write( |\n| ).
out->write( data = sec_w_xco name = `sec_w_xco` ).
out->write( |\n| ).
out->write( data = min_w_xco name = `min_w_xco` ).
out->write( |\n| ).
out->write( data = hour_w_xco name = `hour_w_xco` ).
out->write( |\n| ).
out->write( data = time_cr4 name = `time_cr4` ).
out->write( |\n| ).
out->write( data = hours_from_xco_time name = `hours_from_xco_time` ).
out->write( |\n| ).
out->write( data = minutes_from_xco_time name = `minutes_from_xco_time` ).
out->write( |\n| ).
out->write( data = seconds_from_xco_time name = `seconds_from_xco_time` ).
out->write( |\n| ).
out->write( data = seconds_total name = `seconds_total` ).
out->write( |\n| ).
out->write( data = calc_hours name = `calc_hours` ).
out->write( |\n| ).
out->write( data = min_calc name = `min_calc` ).
out->write( |\n| ).
out->write( data = sec_calc name = `sec_calc` ).
out->write( |\n| ).
out->write( data = time_xco_add name = `time_xco_add` ).
out->write( |\n| ).
out->write( data = time_xco_subtr name = `time_xco_subtr` ).
out->write( |\n| ).
out->write( data = conv_time_str name = `conv_time_str` ).
out->write( |\n| ).
"Note: When outputting the data object of type t with the classrun,
"it is displayed in the format hh:mm:ss in the console for better
"readability.
out->write( data = conv_time_t name = `conv_time_t` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `8) Time Stamps of Type utclong` ) ).
"Retrieving an UTC time stamp using the built-in function utclong_current
"The return value has the type utclong.
DATA(ts1) = utclong_current( ).
"Using XCO
"In the case of XCO, the return value is of type string.
"Retrieving a time stamp in the user's time zone (which is the default)
DATA(ts2) = xco_cp=>sy->moment( xco_cp_time=>time_zone->user
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Current time stamp in UTC
DATA(ts3) = xco_cp=>sy->moment( xco_cp_time=>time_zone->utc
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Different formatting options
DATA(ts_xco) = xco_cp=>sy->moment( xco_cp_time=>time_zone->utc ).
DATA(ts4) = ts_xco->as( xco_cp_time=>format->abap )->value.
DATA(ts5) = ts_xco->as( xco_cp_time=>format->iso_8601_basic )->value.
DATA(ts6) = ts_xco->as( xco_cp_time=>format->iso_8601_extended )->value.
out->write( data = ts1 name = `ts1` ).
out->write( |\n| ).
out->write( data = ts2 name = `ts2` ).
out->write( |\n| ).
out->write( data = ts3 name = `ts3` ).
out->write( |\n| ).
out->write( data = ts4 name = `ts4` ).
out->write( |\n| ).
out->write( data = ts5 name = `ts5` ).
out->write( |\n| ).
out->write( data = ts6 name = `ts6` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `9) Creating/Modifying a Time Stamp` ) ).
"--------------------- Creating time stamps --------------------
DATA ts7 TYPE utclong.
ts7 = utclong_current( ).
DATA ts8 TYPE utclong VALUE '2024-11-05 15:30:00'.
DATA(ts9) = CONV utclong( '2024-11-03 05:30:00' ).
"Using XCO
"As above, the return value is of type string. Various methods, formatting options,
"etc. are available.
DATA(ts10) = xco_cp_time=>moment( iv_year = '2024'
iv_month = '01'
iv_day = '01'
iv_hour = '12'
iv_minute = '34'
iv_second = '55'
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"--------------------- Modifying time stamps (XCO) --------------------
"As covered for date and time types, you can modify time stamps using string
"processing functionalities. They are not covered here. XCO provides, for
"example, the 'overwrite' method. Optional parameters are available.
DATA(ts11) = xco_cp_time=>moment( iv_year = '2024'
iv_month = '03'
iv_day = '05'
iv_hour = '02'
iv_minute = '54'
iv_second = '12' ).
DATA(ts12) = ts11->overwrite( iv_year = '2025'
iv_month = '07'
iv_day = '15'
iv_hour = '01'
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
out->write( data = ts7 name = `ts7` ).
out->write( |\n| ).
out->write( data = ts8 name = `ts8` ).
out->write( |\n| ).
out->write( data = ts9 name = `ts9` ).
out->write( |\n| ).
out->write( data = ts10 name = `ts10` ).
out->write( |\n| ).
out->write( data = ts12 name = `ts12` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `10) Time Stamp Calculations with the Built-In Function utclong_add` ) ).
"With the built-in function utclong_add, at least one parameter must be specified
"besides 'val'.
"Note that there are no parameters for years and months.
DATA(utc4calc) = CONV utclong( '2024-01-01 15:55:14.1173220' ).
"Adding one hour
DATA(ts13) = utclong_add( val = utc4calc
hours = 1 ).
"Subtracting one hour by passing a negative integer value (no separate substract
"function available)
DATA(ts14) = utclong_add( val = utc4calc
hours = -1 ).
"Using all parameters
DATA(ts15) = utclong_add( val = utc4calc
days = 1
hours = 2
minutes = CONV int8( '13' )
seconds = CONV decfloat34( '53.12' ) ).
out->write( data = ts13 name = `ts13` ).
out->write( |\n| ).
out->write( data = ts14 name = `ts14` ).
out->write( |\n| ).
out->write( data = ts15 name = `ts15` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `11) Time Stamp Calculations with XCO` ) ).
"Creating two time stamps with XCO
DATA(ts_ref1) = xco_cp_time=>moment( iv_year = '2024'
iv_month = '01'
iv_day = '01'
iv_hour = '11'
iv_minute = '37'
iv_second = '54' ).
DATA(ts_ref2) = xco_cp_time=>moment( iv_year = '2024'
iv_month = '02'
iv_day = '02'
iv_hour = '10'
iv_minute = '28'
iv_second = '59' ).
"Additions; various optional parameters are available
DATA(ts18) = ts_ref1->add( iv_day = 1 iv_month = 2 iv_year = 3
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
DATA(ts19) = ts_ref2->add( iv_hour = 1 iv_minute = 2 iv_second = 4
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Adding an hour to the current time stamp
DATA(ts20) = xco_cp=>sy->moment( )->add( iv_hour = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Substractions
DATA(ts21) = ts_ref1->subtract( iv_day = 1 iv_month = 2 iv_year = 3
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
DATA(ts22) = ts_ref2->subtract( iv_hour = 1 iv_minute = 2 iv_second = 4
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"Substracting 1 year/month/day/hour/minute/second from the curernt time stamp
DATA(ts23) = xco_cp=>sy->moment( )->subtract( iv_day = 1 iv_month = 1 iv_year = 1
iv_hour = 1 iv_minute = 1 iv_second = 1
)->as( xco_cp_time=>format->iso_8601_extended
)->value.
"---- Excursion: Defining a time interval and checking if ----
"---- a time stamp is within that time interval --------------
"In the following example, a time interval is defined from the current time stamp
"retrieved with XCO to a specified one. Using the 'contains' method and providing
"another time stamp reference, it is checked whether that time stamp is contained
"in the time interval. The return value is of type abap_bool. Using the lower_bound
"and upper_bound attributes, the low and high values are retrieved.
DATA(ts_ref3) = xco_cp_time=>moment( iv_year = '2030'
iv_month = '01'
iv_day = '01'
iv_hour = '10'
iv_minute = '00'
iv_second = '00' ).
DATA(ts_ref4) = xco_cp_time=>moment( iv_year = '2028'
iv_month = '01'
iv_day = '01'
iv_hour = '11'
iv_minute = '00'
iv_second = '00' ).
DATA(ts_interval) = xco_cp=>sy->moment( )->interval_to( ts_ref3 ).
DATA(ts_in) = ts_interval->contains( ts_ref4 ).
DATA(ts_interval_low) = ts_interval->lower_bound->as( xco_cp_time=>format->iso_8601_extended )->value.
DATA(ts_interval_high) = ts_interval->upper_bound->as( xco_cp_time=>format->iso_8601_extended )->value.
out->write( data = ts18 name = `ts18` ).
out->write( |\n| ).
out->write( data = ts19 name = `ts19` ).
out->write( |\n| ).
out->write( data = ts20 name = `ts20` ).
out->write( |\n| ).
out->write( data = ts21 name = `ts21` ).
out->write( |\n| ).
out->write( data = ts22 name = `ts22` ).
out->write( |\n| ).
out->write( data = ts23 name = `ts23` ).
out->write( |\n| ).
out->write( data = ts_in name = `ts_in` ).
out->write( |\n| ).
out->write( data = ts_interval_low name = `ts_interval_low` ).
out->write( |\n| ).
out->write( data = ts_interval_high name = `ts_interval_high` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `12) Calculating Time Stamp Differences Using the Built-In Function utclong_diff` ) ).
DATA(ts16) = CONV utclong( '2024-01-01 05:30:00' ).
DATA(ts17) = CONV utclong( '2024-01-01 06:30:00' ).
"The return value has the type decfloat34. It contains the exact
"difference in seconds.
DATA(ts_diff1) = utclong_diff( high = ts17
low = ts16 ).
DATA(ts_diff2) = utclong_diff( high = ts16
low = ts17 ).
out->write( data = ts_diff1 name = `ts_diff1` ).
out->write( |\n| ).
out->write( data = ts_diff2 name = `ts_diff2` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `13) CONVERT UTCLONG: Time Stamp (utclong) -> Local Date/Time` ) ).
DATA ts_utc TYPE utclong VALUE '2024-11-03 05:30:00'.
CONVERT UTCLONG ts_utc
INTO DATE DATA(date_est)
TIME DATA(time_est)
TIME ZONE 'EST'.
out->write( data = date_est name = `date_est` ).
out->write( |\n| ).
out->write( data = time_est name = `time_est` ).
out->write( |\n| ).
"More optional additions:
"- FRACTIONAL SECONDS: Getting the fractions of seconds
"- DAYLIGHT SAVING TIME: Determining if the time stamp is
" in a daylight saving time (the example uses time stamp
" in August)
CONVERT UTCLONG CONV utclong( '2024-08-08 09:23:11.7681270' )
INTO DATE date_est
TIME time_est
FRACTIONAL SECONDS DATA(sec_est)
DAYLIGHT SAVING TIME DATA(dsl_est)
TIME ZONE 'EST'.
out->write( data = date_est name = `date_est` ).
out->write( |\n| ).
out->write( data = time_est name = `time_est` ).
out->write( |\n| ).
out->write( data = sec_est name = `sec_est` ).
out->write( |\n| ).
out->write( data = dsl_est name = `dsl_est` ).
out->write( |\n| ).
"If the time zone specified is initial, there is no conversion.
CONVERT UTCLONG CONV utclong( '2024-08-08 09:23:11.7681270' )
INTO DATE DATA(date_init)
TIME DATA(time_init)
TIME ZONE VALUE #( ).
out->write( data = date_init name = `date_init` ).
out->write( |\n| ).
out->write( data = time_init name = `time_init` ).
out->write( |\n| ).
"Specifying a non-existent time zone raises a catchable exception.
TRY.
CONVERT UTCLONG CONV utclong( '2024-08-08 09:23:11.7681270' )
INTO DATE DATA(date_nope)
TIME DATA(time_nope)
TIME ZONE 'NOPE'.
CATCH cx_sy_conversion_no_date_time INTO DATA(err_conv).
ENDTRY.
IF err_conv IS INITIAL.
out->write( data = date_nope name = `date_nope` ).
out->write( |\n| ).
out->write( data = time_nope name = `time_nope` ).
ELSE.
out->write( data = err_conv->get_text( ) name = `err_conv->get_text( )` ).
ENDIF.
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `14) CONVERT INTO UTCLONG: Local Date/Time -> Time Stamp (utclong)` ) ).
DATA date2utcl TYPE d VALUE '20240101'.
DATA time2utcl TYPE t VALUE '112458'.
DATA utcl TYPE utclong.
CONVERT DATE date2utcl
TIME time2utcl
TIME ZONE 'EST'
INTO UTCLONG utcl.
out->write( data = utcl name = `utcl` ).
out->write( |\n| ).
"Using optional additions
"Check the details in the ABAP Keyword Documentation.
"Specifying 'X' for DAYLIGHT SAVING TIME in the following
"example raises a catchable exception.
CONVERT DATE date2utcl
TIME time2utcl
FRACTIONAL SECONDS CONV decfloat34( '0.768127' )
DAYLIGHT SAVING TIME ''
TIME ZONE 'EST'
INTO UTCLONG DATA(utcl_inl1).
out->write( data = utcl_inl1 name = `utcl_inl1` ).
out->write( |\n| ).
"There is no time shift in case of an initial time zone specification.
CONVERT DATE date2utcl
TIME time2utcl
TIME ZONE VALUE #( )
INTO UTCLONG DATA(utcl_inl2).
out->write( data = utcl_inl2 name = `utcl_inl2` ).
out->write( |\n| ).
"Ensure that valid values are passed
"The following example explores ABAP statements with invalid values
"that are passed. The valid time stamp, the exceptions raised and
"the error messages are added to an internal table.
DATA error_checks TYPE string_table.
DATA date_test TYPE d.
DATA time_test TYPE t.
DATA frac_sec_test TYPE decfloat34.
DATA dls_test TYPE abap_bool.
DATA tz_test TYPE string.
DO 6 TIMES.
date_test = '20240101'.
time_test = '112458'.
frac_sec_test = '0.768127'.
dls_test = abap_false.
tz_test = `EST`.
CASE sy-index.
WHEN 1.
"No data object change. The statement below should return a valid time stamp.
WHEN 2.
"Invalid date
"In newer ABAP releases, the following statement shows a
"syntax warning that the date (intentionally specified as
"invalid date here) does not match the type. Therefore,
"the example is provided differently to circumvent the
"syntax warning.
"date_test = '20249999'.
TYPES c_l8 TYPE c LENGTH 8.
DATA falsedate TYPE c_l8 VALUE '20240231'.
date_test = falsedate.
WHEN 3.
"Invalid time
"The following statement is commented out for the reasons
"mentioned above.
"time_test = '992458'.
TYPES c_l6 TYPE c LENGTH 6.
DATA falsetime TYPE c_l6 VALUE '992458'.
time_test = falsetime.
WHEN 4.
"Invalid fractions of seconds
frac_sec_test = '1'.
WHEN 5.
"Invalid specification for this particular example.
dls_test = 'X'.
WHEN 6.
"Invalid time zone
"The following statement is commented out for the reasons
"mentioned above.
"dls_test = `NOPE`.
TYPES c_l4 TYPE c LENGTH 4.
DATA falsetimezone TYPE c_l4 VALUE 'NOPE'.
dls_test = falsetimezone.
ENDCASE.
TRY.
CONVERT DATE date_test
TIME time_test
FRACTIONAL SECONDS frac_sec_test
DAYLIGHT SAVING TIME dls_test
TIME ZONE tz_test
INTO UTCLONG DATA(utcl_inl3).
error_checks = VALUE #( BASE error_checks ( |({ sy-index }) Valid time stamp: { utcl_inl3 }| ) ).
CATCH cx_root INTO DATA(err).
error_checks = VALUE #( BASE error_checks
( |({ sy-index }) Exception | &&
|{ replace( val = cl_abap_typedescr=>describe_by_object_ref( err )->absolute_name
sub = `\CLASS=`
with = `` ) } was raised: | &&
|{ err->get_text( ) }| ) ).
ENDTRY.
ENDDO.
out->write( data = error_checks name = `error_checks` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `15) CL_ABAP_UTCLONG: Utilities for Time Stamps (utclong)` ) ).
"Check the class documentation. More methods are available.
DATA(low_timestamp) = CONV utclong( '2024-01-01 05:30:00' ).
DATA(high_timestamp) = CONV utclong( '2024-01-03 10:35:12' ).
"'diff' method: Calculating time differences
cl_abap_utclong=>diff( EXPORTING high = high_timestamp
low = low_timestamp
IMPORTING days = DATA(diff_days)
hours = DATA(diff_hours)
minutes = DATA(diff_minutes)
seconds = DATA(diff_seconds) ).
"Converting a time stamp from a character-like format to utclong
"Creating a time stamp in a string
DATA(ts_string) = |{ utclong_current( ) TIMESTAMP = ENVIRONMENT TIMEZONE = 'UTC' }|.
TRY.
cl_abap_utclong=>read( EXPORTING string = ts_string
timezone = 'UTC'
IMPORTING value = DATA(utc_ts) ).
CATCH cx_abap_utclong_invalid.
ENDTRY.
out->write( data = diff_days name = `diff_days` ).
out->write( |\n| ).
out->write( data = diff_hours name = `diff_hours` ).
out->write( |\n| ).
out->write( data = diff_minutes name = `diff_minutes` ).
out->write( |\n| ).
out->write( data = diff_seconds name = `diff_seconds` ).
out->write( |\n| ).
out->write( data = utc_ts name = `utc_ts` ).
**********************************************************************
out->write( zcl_demo_abap_aux=>heading( `16) GET TIME STAMP: Retrieving the Current Time Stamp` ) ).
"Short form
DATA ts_short TYPE timestamp.
GET TIME STAMP FIELD ts_short.
"Long form