[MODULE] Vertical scope annotation

Description

This module helps to figure out which scope is enclosing the current code line by displaying the line preceding the start of the scope vertically in the margin.

vertical scope annotation screenshot

Version

This was tested with 4coder 4.1.7.

Setup

You need to #include this file in your custom layer (4coder_default_bindings.cpp by default) below the line:

1
#include "4coder_default_include.cpp"

You need to call the function vertical_scope_annotation_draw from the render caller hook (HookID_RenderCaller). By default this is the default_render_buffer function in 4coder_default_hooks.cpp.

If you're using the default hook, you will need to insert the following function prototype above the declaration of default_render_buffer.

1
function void vertical_scope_annotation_draw( Application_Links *app, View_ID view, Buffer_ID buffer, Text_Layout_ID text_layout_id, u32 flags );

You'll also need to either move the declaration of the vertical_scope_annotation_flag_t enum before the declaration of default_render_buffer or hardcoding the values instead of using the enum names. The values are:

1
2
3
4
5
6
vertical_scope_annotation_flag_top_to_bottom = 0;
vertical_scope_annotation_flag_bottom_to_top = 1;
vertical_scope_annotation_flag_highlight = 2;
vertical_scope_annotation_flag_highlight_case_label = 4;
vertical_scope_annotation_flag_no_comment = 8;
vertical_scope_annotation_flag_only_last_case = 16;

If you're using a custom render caller hook, make sure you include the file before the definition of your render caller hook.

The call should look like this:

1
vertical_scope_annotation_draw( app, view_id, buffer, text_layout_id, flags );

Where flags is a combination of any of the vertical_scope_annotation_flag_* values.

  • flag_top_top_bottom: display the text from the top towards the bottom (the baseline is on the left);
  • flag_bottom_to_top: display the text from the bottom towards the top (the baseline is on the right);
  • flag_highlight: display part of the text (keyword, identifier, label... depending on the line) in a different color;
  • flag_highlight_case_label: highlight case labels instead of the case keyword;
  • flag_no_comment: don't use comment in the text;
  • flag_only_last_case: if the text contains several case labels (fallthrough), display only the last one.

Colors can be configured by adding the following lines in your theme file.

1
2
vertical_scope_annotation_text = defcolor_comment;
vertical_scope_annotation_highlight = defcolor_text_default;

Issues

If the text is purple, make sure the theme file is loaded.

The full theme file name needs to be set in 'config.4coder', which by default isn't the case. Use :

1
default_theme_name = "theme-4coder";

instead of

1
default_theme_name = "4coder";

Details

Prefix

This module uses vertical_scope_annotation as a prefix for its types and functions.

Ids

This module declares the following managed ids:

In the colors group:

  • vertical_scope_annotation_text;
  • vertical_scope_annotation_highlight.

Modified files

This module needs modifications in the following files to work:

  • 4coder_default_bindings.h or your custom layer file;
  • 4coder_default_hooks.h.

Global variables

This module doesn't use any global variables.

Types

This module defines the following types:

  • vertical_scope_annotation_flag_t;
  • vertical_scope_annotation_start_t.

Functions

This module declare the following functions:

  • vertical_scope_annotation_draw;
  • vertical_scope_annotation_get_text;
  • vertical_scope_annotation_get_start.

Scope attachment

This module doesn't declare any scope attachment.

vertical scope_annotation.cpp

  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
/*
# Description

This module helps to figure out which scope is enclosing the current code line by
displaying the line preceding the start of the scope vertically in the margin.

# Version

This was tested with 4coder 4.1.7.

# Setup

You need to #include this file in your custom layer ('4coder_default_bindings.cpp' by
default) below the line:

'''c
#include "4coder_default_include.cpp"
'''

You need to call the function 'vertical_scope_annotation_draw' from the render caller hook
('HookID_RenderCaller'). By default this is the 'default_render_buffer' function in 
'4coder_default_hooks.cpp'.

If you're using the default hook, you will need to insert the following function prototype
above the declaration of 'default_render_buffer'.

'''c
function void vertical_scope_annotation_draw( Application_Links *app, View_ID view, Buffer_ID buffer, Text_Layout_ID text_layout_id, u32 flags );
'''

You'll also need to either move the declaration of the 'vertical_scope_annotation_flag_t'
enum before the declaration of 'default_render_buffer' or hardcoding the values instead of
using the enum names. The values are:

'''c
vertical_scope_annotation_flag_top_to_bottom = 0;
vertical_scope_annotation_flag_bottom_to_top = 1;
vertical_scope_annotation_flag_highlight = 2;
vertical_scope_annotation_flag_highlight_case_label = 4;
vertical_scope_annotation_flag_no_comment = 8;
vertical_scope_annotation_flag_only_last_case = 16;
'''

If you're using a custom render caller hook, make sure you include the file before the
definition of your render caller hook.

The call should look like this:

'''c
vertical_scope_annotation_draw( app, view_id, buffer, text_layout_id, flags );
'''

Where flags is a combination of any of the 'vertical_scope_annotation_flag_*' values.

- 'flag_top_top_bottom': display the text from the top towards the bottom (the baseline is
on the left);
- 'flag_bottom_to_top': display the text from the bottom towards the top (the baseline is
on the right);
- 'flag_highlight': display part of the text (keyword, identifier, label... depending on
the line) in a different color;
- 'flag_highlight_case_label': highlight case labels instead of the 'case' keyword;
- 'flag_no_comment': don't use comment in the text;
- 'flag_only_last_case': if the text contains several case labels (fallthrough), display
only the last one.

Colors can be configured by adding the following lines in your theme file.

'''c
vertical_scope_annotation_text = defcolor_comment;
vertical_scope_annotation_highlight = defcolor_text_default;
'''

# Issues

If the text is purple, make sure the theme file is loaded.

The full theme file name needs to be set in 'config.4coder', which by default isn't the
case. Use :

'''c
default_theme_name = "theme-4coder";
'''

instead of

'''c
default_theme_name = "4coder";
'''

# Details

## Prefix

This module uses 'vertical_scope_annotation' as a prefix for its types and functions.

## Ids

This module declares the following managed ids:

In the 'colors' group:
- 'vertical_scope_annotation_text';
- 'vertical_scope_annotation_highlight'.

## Modified files

This module needs modifications in the following files to work:
- '4coder_default_bindings.h' or your custom layer file;
- '4coder_default_hooks.h'.

## Global variables

This module doesn't use any global variables.

## Types

This module defines the following types:
- 'vertical_scope_annotation_flag_t';
- 'vertical_scope_annotation_start_t'.

## Functions

This module declare the following functions:
- 'vertical_scope_annotation_draw';
- 'vertical_scope_annotation_get_text';
- 'vertical_scope_annotation_get_start'.

## Scope attachment

This module doesn't declare any scope attachment.
*/

CUSTOM_ID( colors, vertical_scope_annotation_text );
CUSTOM_ID( colors, vertical_scope_annotation_highlight );

typedef enum vertical_scope_annotation_flag_t {
    vertical_scope_annotation_flag_top_to_bottom = 0,
    vertical_scope_annotation_flag_bottom_to_top = 1,
    vertical_scope_annotation_flag_highlight = 1 << 1,
    vertical_scope_annotation_flag_highlight_case_label = 1 << 2,
    vertical_scope_annotation_flag_no_comment = 1 << 3,
    vertical_scope_annotation_flag_only_last_case = 1 << 4,

    /* NOTE: Do not use this. */
    vertical_scope_annotation_orientation_mask = vertical_scope_annotation_flag_bottom_to_top,

} vertical_scope_annotation_flag_t;

typedef struct vertical_scope_annotation_start_t {

    Token_Iterator_Array it;
    b32 contains_if;
    b32 contains_else;
    b32 is_valid;

} vertical_scope_annotation_start_t;

function vertical_scope_annotation_start_t vertical_scope_annotation_get_start( Token_Array* tokens, i64 position, u32 flags ) {

    /* NOTE: The annotation_start_t struct iterator will point to the token preceding the one to use for the string. */
    vertical_scope_annotation_start_t annotation = { 0 };
    annotation.it = token_iterator_pos( 0, tokens, position );
    i64 parenthesis = 0;

    b32 contains_keyword = false;
    b32 contains_identifier = false;
    b32 contains_comment = false;
    b32 stop = false;

    while ( !stop && token_it_dec_non_whitespace( &annotation.it ) ) {

        Token *token = token_it_read( &annotation.it );

        if ( token->kind == TokenBaseKind_ParentheticalClose ) {
            parenthesis++;
        } else if ( token->kind == TokenBaseKind_ParentheticalOpen ) {
            parenthesis--;
        } else if ( parenthesis == 0 ) {

            switch ( token->kind ) {

                case TokenBaseKind_Identifier: {

                    if ( token->flags == TokenBaseFlag_PreprocessorBody ) {
                        stop = true;
                    } else {
                        contains_identifier = true;
                    }
                } break;

                case TokenBaseKind_Preprocessor:
                case TokenBaseKind_LiteralInteger:
                case TokenBaseKind_LiteralFloat:
                case TokenBaseKind_LiteralString:
                case TokenBaseKind_ScopeOpen:
                case TokenBaseKind_ScopeClose: {
                    stop = true;
                } break;

                case TokenBaseKind_StatementClose: {
                    if ( token->sub_kind != TokenCppKind_Colon ) {
                        stop = true;
                    }
                } break;

                case TokenBaseKind_Comment: {

                    if ( !( flags & vertical_scope_annotation_flag_no_comment ) ) {

                        if ( !contains_keyword ) {
                            token_it_dec_non_whitespace( &annotation.it );
                            contains_comment = true;
                        };
                        stop = true;
                    }
                } break;

                case TokenBaseKind_Keyword: {

                    if ( token->sub_kind == TokenCppKind_Else ) {
                        annotation.contains_else = true;
                    } else if ( token->sub_kind == TokenCppKind_If ) {
                        annotation.contains_if = true;
                    } else if ( token->sub_kind == TokenCppKind_Case && ( flags & vertical_scope_annotation_flag_only_last_case ) ) {
                        token_it_dec_non_whitespace( &annotation.it );
                        stop = true;
                    }

                    contains_keyword = true;

                } break;
            }
        }
    }

    annotation.is_valid = contains_keyword || contains_identifier || contains_comment;

    return annotation;
}

function i64 vertical_scope_annotation_get_text( Application_Links* app, Buffer_ID buffer, Token_Array* tokens, i64 position, u32 flags, String_u8* out, Range_i64* highlight_range ) {

    ProfileScope( app, "vertical scope annotation get_text" );

    Scratch_Block scratch( app );

    i64 first_character_position = 0;
    highlight_range->min = highlight_range->max = -1;

    vertical_scope_annotation_start_t annotation = vertical_scope_annotation_get_start( tokens, position, flags );
    b32 stop_after_new_line = false;
    b32 contains_else_of = false;

    if ( annotation.is_valid && annotation.contains_else && !annotation.contains_if ) {

        /* NOTE: if we are on "else", not "else if" we need to seek the if to get the condition. */
        annotation.is_valid = false;
        string_append( out, string_u8_litexpr( "else of " ) );
        contains_else_of = true;

        Token* token = token_it_read( &annotation.it );

        if ( token->kind == TokenBaseKind_ScopeClose ) {

            i64 if_scope_open = -1;
            b32 found = find_nest_side( app, buffer, token->pos - 1, FindNest_Balanced | FindNest_Scope, Scan_Backward, NestDelim_Open, &if_scope_open );

            if ( found ) {
                annotation = vertical_scope_annotation_get_start( tokens, if_scope_open, flags );
            }

        } else {

            /* NOTE: "if" without scope bracket, search backward for the first "if". */
            stop_after_new_line = true;

            while ( token_it_dec_non_whitespace( &annotation.it ) ) {
                token = token_it_read( &annotation.it );
                if ( token->kind == TokenBaseKind_Keyword && token->sub_kind == TokenCppKind_If ) {
                    /* NOTE: There could be a potential "else if" here so we need to call get_annotation_start again. */
                    annotation = vertical_scope_annotation_get_start( tokens, token->pos + token->size, flags );
                    break;
                }
            }
        }
    }

    if ( annotation.is_valid ) {

        Assert( !annotation.contains_else || annotation.contains_if );

        b32 has_token = token_it_inc_non_whitespace( &annotation.it );
        b32 stop = false;
        i64 parenthesis = 0;
        first_character_position = annotation.it.ptr->pos;
        b32 find_highlight = true;

        while ( !stop && has_token ) {

            Token* token = token_it_read( &annotation.it );

            if ( token->kind == TokenBaseKind_ParentheticalOpen ) {
                parenthesis++;
            } else if ( token->kind == TokenBaseKind_ParentheticalClose ) {
                parenthesis--;
            } else if ( parenthesis == 0 ) {

                switch ( token->kind ) {

                    case TokenBaseKind_ScopeOpen: {
                        stop = true;
                    } break;

                    case TokenBaseKind_Whitespace: {

                        /* NOTE: "if" without scope bracket. */
                        if ( stop_after_new_line ) {
                            u8 c = buffer_get_char( app, buffer, token->pos );

                            if ( c == '\n' || c == '\r' ) {
                                stop = true;
                            }
                        }

                    } break;
                }
            }

            if ( !stop ) {

                /* NOTE: a single whitespace token groups concecutive whitespaces and we want to only print one white space and no new lines.*/
                if ( token->kind == TokenBaseKind_Whitespace ) {
                    string_append( out, string_u8_litexpr( " " ) );
                } else {

                    if ( token->kind != TokenBaseKind_Comment || !( flags & vertical_scope_annotation_flag_no_comment ) ) {

                        /* NOTE: Should I highlight "else" in "else if" ? */
                        if ( find_highlight && ( token->kind == TokenBaseKind_Keyword || token->kind == TokenBaseKind_Identifier ) ) {
                            highlight_range->min = out->size;
                            highlight_range->max = out->size + token->size;
                        }

                        b32 case_condition = false;

                        if ( flags & vertical_scope_annotation_flag_highlight_case_label ) {
                            case_condition = ( token->kind == TokenBaseKind_StatementClose && token->sub_kind == TokenCppKind_Colon );
                        } else {
                            case_condition = ( token->kind == TokenBaseKind_Keyword && token->sub_kind == TokenCppKind_Case );
                        }

                        if ( token->kind == TokenBaseKind_ParentheticalOpen || case_condition ) {
                            find_highlight = false;
                        }

                        String_Const_u8 temp = push_token_lexeme( app, scratch, buffer, token );
                        string_append( out, temp );
                    }
                }

                has_token = token_it_inc_all( &annotation.it );
            }
        }

        if ( find_highlight ) {
            highlight_range->min = highlight_range->max = -1;
        } else if ( contains_else_of ) {
            highlight_range->min =  0;
        }
    }

    return first_character_position;
}

function void vertical_scope_annotation_draw( Application_Links *app, View_ID view, Buffer_ID buffer, Text_Layout_ID text_layout_id, u32 flags ) {

    ProfileScope( app, "vertical scope annotation" );

    Range_i64 visible_range = text_layout_get_visible_range( app, text_layout_id );
    Token_Array tokens = get_token_array_from_buffer( app, buffer );

    Face_ID face_id = get_face_id( app, buffer );
    Face_Metrics metrics = get_face_metrics( app, face_id );
    Rect_f32 region = text_layout_region( app, text_layout_id );

    Buffer_Scroll scroll = view_get_buffer_scroll( app, view );
    float x_start = region.x0 - scroll.position.pixel_shift.x;

    i64 indent_space_count = def_get_config_u64( app, vars_save_string_lit( "indent_width" ) );

    if ( def_get_config_b32( vars_save_string_lit( "enable_virtual_whitespace" ) ) ) {
        indent_space_count = def_get_config_u64( app, vars_save_string_lit( "virtual_whitespace_regular_indent" ) );
    }

    Scratch_Block scratch( app );

    /* NOTE: Get the end of the textual line. */
    i64 new_pos = get_line_side_pos_from_pos( app, buffer, visible_range.min, Side_Max );
    Buffer_Cursor cursor = view_compute_cursor( app, view, seek_pos( new_pos ) );
    Vec2_f32 p = view_relative_xy_of_pos( app, view, cursor.line, cursor.pos );
    i64 end_of_line = view_pos_at_relative_xy( app, view, cursor.line, p );

    ProfileBlockNamed( app, "vertical scope annotation get_enclosure_ranges", get_enclosure_ranges_block );
    Range_i64_Array ranges = get_enclosure_ranges( app, scratch, buffer, end_of_line, RangeHighlightKind_CharacterHighlight );
    ProfileCloseNow( get_enclosure_ranges_block );

    ProfileBlockNamed( app, "vertical scope annotation ranges loop", ranges_loop );

    for ( i32 i = ranges.count - 1; i >= 0; i-- ) {

        Range_i64 range = ranges.ranges[ i ];

        u8 string_buffer[ 1024 ] = { 0 };
        String_u8 annotation = Su8( string_buffer, 0, ArrayCount( string_buffer ) );
        Range_i64 highlight = { -1, -1 };
        i64 first_character_position = vertical_scope_annotation_get_text( app, buffer, &tokens, range.min, flags, &annotation, &highlight );

        b32 draw = false;
        f32 region_top_offset = 0;

        if ( first_character_position >= visible_range.min ) {
            /* NOTE: Is the text only partially visible vertically ? */
            Rect_f32 rect = text_layout_character_on_screen( app, text_layout_id, first_character_position );

            if ( rect.y1 < region.y0 + metrics.text_height ) {
                rect = text_layout_character_on_screen( app, text_layout_id, range.min );
                region_top_offset = rect.y1 - region.y0;
                draw = true;
            }
        } else {

            /* NOTE: If an "if else" has the "else" in view but the "if" isn't, we need to adjust the region to avoid overlapping annotation with code. */
            if ( range.min >= visible_range.min ) {
                Rect_f32 rect = text_layout_character_on_screen( app, text_layout_id, range.min );
                region_top_offset = rect.y1 - region.y0;
            }

            draw = true;
        }

        if ( draw ) {

            ProfileScope( app, "vertical scope annotation draw" );

            b32 override_next_range = false;
            Range_i64 override_range = { 0 };

            if ( highlight.min > -1 ) {

                ProfileScope( app, "vertical scope annotation find else range of if" );

                /* NOTE simon: When there is an "if else" and the if is outside the view we want show the annotation for the else even if it's in view. */
                u64 keyword_size = annotation.size - highlight.min;
                b32 is_if = false;

                if ( keyword_size >= 2 ) {
                    is_if = ( annotation.str[ highlight.min ] == 'i' && annotation.str[ highlight.min + 1 ] == 'f' );

                    if ( keyword_size > 2 ) {
                        is_if = is_if && ( annotation.str[ highlight.min + 2 ] == ' ' || annotation.str[ highlight.min + 2 ] == '(' );
                    }
                }

                if ( is_if ) {

                    Token_Iterator_Array it = token_iterator_pos( 0, &tokens, range.max );
                    token_it_inc_non_whitespace( &it );
                    Token* token = token_it_read( &it );

                    if ( token->kind == TokenBaseKind_Keyword && token->sub_kind == TokenCppKind_Else ) {

                        token_it_inc_non_whitespace( &it );
                        token = token_it_read( &it );

                        if ( token->kind == TokenBaseKind_ScopeOpen ) {
                            override_range.start = token->pos;
                            override_next_range = find_nest_side( app, buffer, token->pos + 1, FindNest_Scope | FindNest_Balanced | FindNest_EndOfToken, Scan_Forward, NestDelim_Close, &override_range.end );
                            if ( override_range.start >= visible_range.end || override_range.end <= visible_range.start ) {
                                override_next_range = false;
                            }
                        }
                    }
                }
            }

            float y0 = region.y0 + region_top_offset;
            float y1 = region.y1;

            if ( range.max - 1 < visible_range.max ) {
                ProfileScope( app, "vertical scope annotation text_layout_character_on_screen" );
                Rect_f32 rect = text_layout_character_on_screen( app, text_layout_id, range.end );
                y1 = rect.y0;
            }

            f32 max_size = y1 - y0;

            if ( max_size < 0 ) {
                max_size = 0;
            }

            ProfileBlockNamed( app, "vertical scope annotation get_string_advance", get_string_advance_block );

            f32 size = get_string_advance( app, face_id, SCu8( annotation.str, annotation.size ) );
#if 0
            while ( size > max_size && annotation.size > 1 ) {
                annotation.size -= 1;
                size = get_string_advance( app, face_id, SCu8( annotation.str, annotation.size ) );
            }
#else
            if ( size > max_size ) {

                f32 percentage = max_size / size;
                u64 string_size = ( u64 ) ( ( f32 ) annotation.size * percentage );

                size = get_string_advance( app, face_id, SCu8( annotation.str, string_size ) );

                if ( size > max_size ) {
                    while ( size > max_size && string_size > 1 ) {
                        string_size--;
                        size = get_string_advance( app, face_id, SCu8( annotation.str, string_size ) );
                    }
                } else {
                    f32 previous_size = size;
                    while ( size < max_size && string_size < annotation.size ) {
                        string_size++;
                        previous_size = size;
                        size = get_string_advance( app, face_id, SCu8( annotation.str, string_size ) );
                    }

                    size = previous_size;
                    string_size--;
                }

                annotation.size = string_size;
            }
#endif
            ProfileCloseNow( get_string_advance_block );

            if ( size <= max_size ) {

                ProfileBlockNamed( app, "vertical scope annotation draw preparation", draw_prep );

                if ( highlight.min > ( i64 ) annotation.size - 1 ) {
                    highlight.min = -1;
                }

                if ( highlight.max > ( i64 ) annotation.size ) {
                    highlight.max = annotation.size;
                }

                Vec2_f32 position = { 0, 0 };
                Vec2_f32 delta = { 0, 0 };

                u32 orientation = flags & vertical_scope_annotation_orientation_mask;

                if ( orientation == vertical_scope_annotation_flag_top_to_bottom ) {

                    float y = y0 + ( max_size - size ) * 0.5f;
                    float x_centering = ( ( indent_space_count * metrics.space_advance ) - metrics.line_height ) * 0.5f;
                    i64 space_count = indent_space_count * ( ranges.count - 1 - i );
                    float x = x_start + space_count * metrics.space_advance + metrics.line_height + x_centering;
                    position = { x, y };
                    delta = V2f32( 0.0f, 1.0f );

                } else {

                    float y = y1 - ( max_size - size ) * 0.5f;
                    float x_centering = ( ( indent_space_count * metrics.space_advance ) - metrics.line_height ) * 0.5f;
                    i64 space_count = indent_space_count * ( ranges.count - 1 - i );
                    float x = x_start + space_count * metrics.space_advance + x_centering;
                    position = { x, y };
                    delta = V2f32( 0.0f, -1.0f );
                }

                String_Const_u8 str_1, str_2, str_3;

                if ( highlight.min >= 0 && ( flags & vertical_scope_annotation_flag_highlight ) ) {
                    str_1 = SCu8( annotation.str, highlight.min );
                    str_2 = SCu8( str_1.str + str_1.size, highlight.max - highlight.min );
                    str_3 = SCu8( str_2.str + str_2.size, annotation.size - ( str_1.size + str_2.size ) );
                } else {
                    str_1 = SCu8( annotation.str, annotation.size );
                    str_2 = str_3 = SCu8( ( u8* ) 0, ( u64 ) 0 );
                }

                ProfileCloseNow( draw_prep );

                ProfileBlockNamed( app, "vertical scope annotation draw strings", draw_strings );

                draw_string_oriented( app, face_id, finalize_color( vertical_scope_annotation_text, 0 ), str_1, position, 0, delta );

                if ( str_2.size ) {
                    f32 str_size = get_string_advance( app, face_id, str_1 );
                    position.y += delta.y * str_size;
                    draw_string_oriented( app, face_id, finalize_color( vertical_scope_annotation_highlight, 0 ), str_2, position, 0, delta );
                }

                if ( str_3.size ) {
                    f32 str_size = get_string_advance( app, face_id, str_2 );
                    position.y += delta.y * str_size;
                    draw_string_oriented( app, face_id, finalize_color( vertical_scope_annotation_text, 0 ), str_3, position, 0, delta );
                }

                ProfileCloseNow( draw_strings );
            }

            if ( override_next_range ) {
                ranges.ranges[ i ] = override_range;
                i++;
            }
        }
    }

    ProfileCloseNow( ranges_loop );
}

Edited by Simon Anciaux on Reason: Update to 4.1.7