aboutsummaryrefslogtreecommitdiff
path: root/lib/stitched_article.ml
blob: 41efa756732894564fa8e25e69b04c7e7bc11eac (plain)
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
module Grep = Grep
module Common = Common
open Notty
module Input_screen = Input_screen

type state =
  { pos : int * int
  ; scroll : int
  ; content : (string * int * string) array
  ; content_pretty : Grep.display_type array
  }

let render_line size_x y scroll (el : Grep.display_type) i =
  match el with
  | Bold el ->
    if i == y - scroll
    then (
      let fill = String.make (max (size_x - String.length el) 0) ' ' in
      I.strf "%s%s" ~attr:A.(st bold ++ st reverse) el fill |> I.pad ~l:0 ~t:i)
    else I.strf "%s" ~attr:A.(st bold) el |> I.pad ~l:0 ~t:i
  | Normal el ->
    if i == y - scroll
    then (
      let fill = String.make (max (size_x - String.length el) 0) ' ' in
      I.strf "%s%s" ~attr:A.(st reverse) el fill |> I.pad ~l:0 ~t:i)
    else I.strf "%s" el |> I.pad ~l:0 ~t:i


let rec render t ({ pos; scroll; content_pretty; _ } as state) =
  let size_x, size_y = Common.Term.size t in
  let x, y = pos in
  let img =
    let elements =
      Array.mapi
        (fun i el -> render_line size_x y scroll el i)
        (* TODO: Fix this ugly slow conversion *)
        (Array.to_seq content_pretty |> Seq.drop scroll |> Array.of_seq)
    in
    let open I in
    Array.fold_left (fun sum el -> el </> sum) I.empty elements
  in
  Common.Term.image t img;
  let content_length = Array.length content_pretty in
  let scroll_up () =
    let scroll = if y - scroll = 0 then max (scroll - 1) 0 else scroll in
    render t @@ { state with pos = x, max (y - 1) 0; scroll }
  in
  let scroll_down () =
    let scroll = if y - scroll >= size_y - 1 then scroll + 1 else scroll in
    render t @@ { state with pos = x, min (y + 1) (content_length - 1); scroll }
  in
  match Common.Term.event t with
  | `End | `Key (`Escape, []) | `Key (`ASCII 'q', []) | `Key (`ASCII 'C', [ `Ctrl ]) -> ()
  | `Mouse (`Press (`Scroll s), _, _) ->
    (match s with
     | `Down -> scroll_down ()
     | `Up -> scroll_up ())
  | `Resize _ -> render t state
  | `Mouse ((`Press _ | `Drag), (_, y), _) ->
    render t { state with pos = 0, min y content_length }
  | `Key (`ASCII 'j', []) | `Key (`ASCII 'N', [ `Ctrl ]) -> scroll_down ()
  | `Key (`ASCII 'k', []) | `Key (`ASCII 'P', [ `Ctrl ]) -> scroll_up ()
  | `Key (`Arrow d, _) ->
    (match d with
     | `Up -> scroll_up ()
     | `Down -> scroll_down ()
     | _ -> render t state)
  | _ -> render t state