ScrollBar.js 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122
  1. import * as ansi from 'tui-lib/util/ansi'
  2. import unic from 'tui-lib/util/unichars'
  3. import {DisplayElement} from 'tui-lib/ui/primitives'
  4. export default class ScrollBar extends DisplayElement {
  5. constructor({
  6. getLayoutType,
  7. getCurrentScroll,
  8. getMaximumScroll,
  9. getTotalItems
  10. }) {
  11. super()
  12. this.getLayoutType = getLayoutType
  13. this.getCurrentScroll = getCurrentScroll
  14. this.getMaximumScroll = getMaximumScroll
  15. this.getTotalItems = getTotalItems
  16. }
  17. fixLayout() {
  18. // Normally we'd subtract one from contentW/contentH when setting the x/y
  19. // position, but the scroll-bar is actually displayed OUTSIDE of (adjacent
  20. // to) the parent's content area.
  21. if (this.getLayoutType() === 'vertical') {
  22. this.h = this.parent.contentH
  23. this.w = 1
  24. this.x = this.parent.contentW
  25. this.y = 0
  26. } else {
  27. this.h = 1
  28. this.w = this.parent.contentW
  29. this.x = 0
  30. this.y = this.parent.contentH
  31. }
  32. }
  33. drawTo(writable) {
  34. // Uuuurgh
  35. this.fixLayout()
  36. // TODO: Horizontal layout! Not functionally a lot different, but I'm too
  37. // lazy to write a test UI for it right now.
  38. const {
  39. backwards: canScrollBackwards,
  40. forwards: canScrollForwards
  41. } = this.getScrollableDirections()
  42. // - 2 for extra UI elements (arrows)
  43. const totalLength = this.h - 2
  44. // ..[-----]..
  45. // ^start|
  46. // ^end
  47. //
  48. // Start and end should correspond to how much of the scroll area
  49. // is currently visible. So, if you can see 60% of the full scroll length
  50. // at a time, and you are scrolled 10% down, the start position of the
  51. // handle should be 10% down, and it should extend 60% of the scrollbar
  52. // length, to the 70% mark.
  53. // NB: I think this math mixes the units for "items" and "lines".
  54. // edgeLength is measured in lines, while totalItems is a number of items.
  55. // This isn't a problem when the length of an item is equal to one line,
  56. // but it's still worth investigating at some point.
  57. const currentScroll = this.getCurrentScroll()
  58. const totalItems = this.getTotalItems()
  59. const edgeLength = this.parent.contentH
  60. const visibleAtOnce = Math.min(totalItems, edgeLength)
  61. const handleLength = visibleAtOnce / totalItems * totalLength
  62. let handlePosition = Math.floor(totalLength / totalItems * currentScroll)
  63. // Silly peeve of mine: The handle should only be visibly touching the top
  64. // or bottom of the scrollbar area if you're actually scrolled all the way
  65. // to the start or end. Otherwise, it shouldn't be touching! There should
  66. // visible space indicating that you can scroll in that direction
  67. // (in addition to the arrows we show at the ends).
  68. if (canScrollBackwards && handlePosition === 0) {
  69. handlePosition = 1
  70. }
  71. if (canScrollForwards && (handlePosition + handleLength) === edgeLength) {
  72. handlePosition--
  73. }
  74. if (this.getLayoutType() === 'vertical') {
  75. const start = this.absTop + handlePosition + 1
  76. for (let i = 0; i < handleLength; i++) {
  77. writable.write(ansi.moveCursor(start + i, this.absLeft))
  78. writable.write(unic.BOX_V_DOUBLE)
  79. }
  80. if (canScrollBackwards) {
  81. writable.write(ansi.moveCursor(this.absTop, this.absLeft))
  82. writable.write(unic.ARROW_UP_DOUBLE)
  83. }
  84. if (canScrollForwards) {
  85. writable.write(ansi.moveCursor(this.absBottom, this.absLeft))
  86. writable.write(unic.ARROW_DOWN_DOUBLE)
  87. }
  88. }
  89. }
  90. getScrollableDirections() {
  91. const currentScroll = this.getCurrentScroll()
  92. const maximumScroll = this.getMaximumScroll()
  93. return {
  94. backwards: (currentScroll > 0),
  95. forwards: (currentScroll < maximumScroll)
  96. }
  97. }
  98. canScrollAtAll() {
  99. const {backwards, forwards} = this.getScrollableDirections()
  100. return backwards || forwards
  101. }
  102. }