« get me outta code hell

Form.js « form « ui - tui-lib - Pure Node.js library for making visual command-line programs (ala vim, ncdu)
about summary refs log tree commit diff
path: root/ui/form/Form.js
blob: e5d756adba719a1ce1a9b53d257a284447b16c13 (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
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
const telc = require('../../util/telchars')

const FocusElement = require('./FocusElement')

module.exports = class Form extends FocusElement {
  constructor() {
    super()

    this.inputs = []
    this.curIndex = 0
    this.captureTab = true
  }

  addInput(input, asChild = true) {
    // Adds the given input as a child element and pushes it to the input
    // list. If the second optional, asChild, is false, it won't add the
    // input element as a child of the form.

    this.inputs.push(input)

    if (asChild) {
      this.addChild(input)
    }
  }

  removeInput(input, asChild = true) {
    // Removes the given input from the form's input list. If the optional
    // argument asChild is false, it won't try to removeChild the input.

    if (this.inputs.includes(input)) {
      this.inputs.splice(this.inputs.indexOf(input), 1)

      if (asChild) {
        this.removeChild(input)
      }
    }
  }

  keyPressed(keyBuf) {
    // Don't do anything if captureTab is set to false. This is handy for
    // nested forms.
    if (!this.captureTab) {
      return
    }

    if (telc.isTab(keyBuf) || telc.isBackTab(keyBuf)) {
      // No inputs to tab through, so do nothing.
      if (this.inputs.length < 2) {
        return
      }

      if (telc.isTab(keyBuf)) {
        this.nextInput()
      } else {
        this.previousInput()
      }

      return false
    }
  }

  get selectable() {
    return this.inputs.some(inp => inp.selectable)
  }

  updateSelectedElement() {
    if (this.root.select) {
      this.root.select(this.inputs[this.curIndex])
    }
  }

  previousInput() {
    // TODO: Forms currently assume there is at least one selectable input,
    // but this isn't necessarily always the case.
    do {
      this.curIndex = (this.curIndex - 1)
      if (this.curIndex < 0) {
        this.curIndex = (this.inputs.length - 1)
      }
    } while (!this.inputs[this.curIndex].selectable)

    this.updateSelectedElement()
  }

  nextInput() {
    // TODO: See previousInput
    do {
      this.curIndex = (this.curIndex + 1) % this.inputs.length
    } while (!this.inputs[this.curIndex].selectable)

    this.updateSelectedElement()
  }

  firstInput(selectForm = true) {
    this.curIndex = 0

    // TODO: See previousInput
    if (!this.inputs[this.curIndex].selectable) {
      this.nextInput()
    }

    if (selectForm || (
      this.root.isChildOrSelfSelected && this.root.isChildOrSelfSelected(this)
    )) {
      this.updateSelectedElement()
    }
  }

  focused() {
    this.updateSelectedElement()
  }
}