72 |
73 |
91 |
94 |
97 |
100 |
103 |
104 |
105 |
106 |
107 |
108 |
109 |
110 |
111 |
112 |
Static:
113 |
114 | - ABYSS1.ANS
115 | - AN-D2.ANS
116 | - BK-KING.ANS
117 | - COMICS14.ANS
118 | - CT-DIE_HARD.ANS
119 | - CT-PIXELS.ANS
120 | - DONATELO.ANS
121 | - DW-FACES.ANS
122 | - DW-HAPPY_HOLIDAYS.ANS
123 | - ED-NS.ANS
124 | - JET.ANS
125 | - FROSTBBS.ANS
126 | - GAVEL30.ANS
127 | - GLOBE.ANS
128 | - LDA-GARFIELD.ANS
129 | - THEQ.ANS
130 | - US-UWU.ANS
131 | - WWANS54.ANS
132 |
133 |
Blinking:
134 |
135 | - CHRIST1.ANS
136 | - SMRFBONK.ANS
137 | - US-CANDLES.ANS
138 | - UTOPIA20.ANS
139 | - XMAS1.ANS
140 |
141 |
Animations:
142 |
143 | - BCACID7.ANS
144 | - BOGACID1.ANS
145 | - CC-ICE1.ICE
146 | - DT-GHETO.ANS
147 | - JD-BUTT.ANS
148 | - LM-OKC.ICE
149 | - SC-ACID5.ANS
150 | - SUBACID.ANS
151 | - UTOPIA86.ANS
152 |
153 |
154 |
157 |
158 | );
159 | }
160 |
--------------------------------------------------------------------------------
/src/dom-components.js:
--------------------------------------------------------------------------------
1 | import { useMemo, useRef, useEffect, createElement } from 'react';
2 | import { useAnsi } from './hooks.js';
3 | import { cgaPalette } from './dos-environment.js';
4 |
5 | export function AnsiText({ src, srcObject, palette = cgaPalette, className = 'AnsiText', ...options }) {
6 | // retrieve data if necessary
7 | const data = useMemo(() => srcObject ?? fetchBuffer(src), [ src, srcObject ]);
8 | // process data through hook
9 | const { lines, blinking, blinked } = useAnsi(data, options);
10 | // convert lines to spans
11 | const children = lines.map((segments) => {
12 | const spans = segments.map(({ text, fgColor, bgColor, blink }) => {
13 | const props = {};
14 | if (Array.isArray(palette)) {
15 | props.style = {
16 | backgroundColor: palette[bgColor],
17 | color: palette[(blink && blinked) ? bgColor : fgColor],
18 | };
19 | } else {
20 | const names = [];
21 | if (fgColor !== undefined) {
22 | names.push(`fgColor${fgColor}`)
23 | }
24 | if (bgColor !== undefined) {
25 | names.push(`bgColor${bgColor}`);
26 | }
27 | if (blink) {
28 | if (blinking === true) {
29 | // manual blinking
30 | if (blink && blinked) {
31 | names.push('blink');
32 | }
33 | } else {
34 | // blinking through css
35 | names.push('blinking');
36 | }
37 | }
38 | props.className = names.join(' ');
39 | }
40 | return createElement('span', props, text);
41 | });
42 | return createElement('div', {}, ...spans);
43 | });
44 | const style = {
45 | display: 'inline-block',
46 | whiteSpace: 'pre',
47 | width: 'fit-content',
48 | };
49 | return createElement('code', { className, style }, ...children);
50 | }
51 |
52 | /* c8 ignore start */
53 | export function AnsiCanvas({ src, srcObject, palette = cgaPalette, className = 'AnsiCanvas', ...options }) {
54 | const canvasRef = useRef();
55 | // retrieve data if necessary
56 | const data = useMemo(() => srcObject ?? fetchBuffer(src), [ src, srcObject ]);
57 | // process data through hook
58 | const { width, height, lines, blinked } = useAnsi(data, options);
59 | // draw into canvas in useEffect hook
60 | useEffect(() => {
61 | const canvas = canvasRef.current;
62 | if (!canvas) {
63 | return;
64 | }
65 | // get font applicable to canvas
66 | let { color, font } = getCanvasStyle(canvas);
67 | if (document.fonts.check(font)) {
68 | draw();
69 | // observe resizing of element so any font changes get applied
70 | const observer = new ResizeObserver(() => {
71 | const { font: newFont, color: newColor } = getCanvasStyle(canvas);
72 | if (newFont !== font || newColor !== color) {
73 | font = newFont;
74 | color = newColor;
75 | draw();
76 | }
77 | });
78 | observer.observe(canvas);
79 | return () => observer.disconnect();
80 | } else {
81 | // draw when the font has been loaded
82 | let cancelled = false;
83 | document.fonts.load(font).then(() => {
84 | if (!cancelled) {
85 | draw();
86 | }
87 | });
88 | return () => cancelled = true;
89 | }
90 |
91 | function draw() {
92 | const { charWidth, charHeight, ascent } = getFontMetrics(font);
93 | canvas.width = width * charWidth;
94 | canvas.height = height * charHeight;
95 | const cxt = canvas.getContext('2d');
96 | cxt.clearRect(0, 0, canvas.width, canvas.height);
97 | cxt.font = font;
98 | let x = 0, y = ascent;
99 | for (const line of lines) {
100 | for (const { text, bgColor, fgColor, blink } of line) {
101 | for (let i = 0; i < text.length; i++) {
102 | if (bgColor !== undefined) {
103 | // fill background with block character for more consistent appearance
104 | // if the full-block character doesn't quote fill the cell, then the gaps between
105 | // cells should appear everywhere
106 | cxt.fillStyle = palette[bgColor];
107 | cxt.fillText('\u2588', x, y);
108 | }
109 | if (!blink || !blinked) {
110 | // use black if foreground color isn't set
111 | cxt.fillStyle = (fgColor !== undefined) ? palette[fgColor] : color;
112 | cxt.fillText(text.charAt(i), x, y);
113 | }
114 | x += charWidth;
115 | }
116 | }
117 | y += charHeight;
118 | x = 0;
119 | }
120 | }
121 | }, [ width, height, lines, blinked, palette ]);
122 | return createElement('canvas', { ref: canvasRef, className });
123 | }
124 |
125 | function getCanvasStyle(node) {
126 | const { fontStyle, fontWeight, fontSize, fontFamily, color } = getComputedStyle(node);
127 | const font = `${fontStyle} ${fontWeight} ${fontSize} ${fontFamily}`;
128 | return { color, font };
129 | }
130 |
131 | const fontMetrics = {};
132 |
133 | function getFontMetrics(specifier) {
134 | let metrics = fontMetrics[specifier];
135 | if (!metrics) {
136 | const canvas = document.createElement('CANVAS');
137 | const cxt = canvas.getContext('2d');
138 | cxt.font = specifier;
139 | const m = cxt.measureText('\u2588');
140 | // support for fontBoundingBoxAscent and fontBoundingBoxDescent is spotty
141 | // (https://developer.mozilla.org/en-US/docs/Web/API/TextMetrics#browser_compatibility)
142 | //
143 | // actualBoundingBoxAscent and actualBoundingBoxDescent wouldn't yield the exactly result
144 | // since the bounding box of the full-block character (U+2588) will likely stick out just a little
145 | const ascent = m.fontBoundingBoxAscent ?? m.actualBoundingBoxAscent;
146 | const descent = m.fontBoundingBoxDescent ?? m.actualBoundingBoxDescent;
147 | const charWidth = m.width;
148 | const charHeight = ascent + descent;
149 | metrics = fontMetrics[specifier] = { ascent, descent, charWidth, charHeight };
150 | }
151 | return metrics;
152 | }
153 | /* c8 ignore stop */
154 |
155 | async function fetchBuffer(src, options) {
156 | if (src) {
157 | const res = await fetch(src, options);
158 | if (res.status !== 200) {
159 | throw new Error(`HTTP ${res.status} - ${res.statusText}`);
160 | }
161 | return await res.arrayBuffer();
162 | }
163 | }
164 |
165 |
--------------------------------------------------------------------------------
/demo/ink/bin/SelectBox.mjs:
--------------------------------------------------------------------------------
1 | import { useFocus, useInput, Text, Box } from 'ink';
2 | import InkSelectInput from 'ink-select-input';
3 | import { jsx as _jsx } from "react/jsx-runtime";
4 | import { jsxs as _jsxs } from "react/jsx-runtime";
5 | const {
6 | default: SelectInput
7 | } = InkSelectInput;
8 | export default function SelectBox({
9 | id,
10 | items,
11 | label: labelWithAmp,
12 | value,
13 | home,
14 | onSelect: onSelectCaller
15 | }) {
16 | const {
17 | isFocused,
18 | focus
19 | } = useFocus({
20 | id
21 | });
22 | const initialIndex = items.findIndex(i => i.value === value);
23 | const [label, hotkey] = extractHotkey(labelWithAmp);
24 | useInput(input => {
25 | if (input.toUpperCase() === hotkey) {
26 | focus(id);
27 | }
28 | }, {
29 | isActive: !isFocused
30 | });
31 | if (isFocused) {
32 | const onSelect = item => {
33 | onSelectCaller?.(item);
34 | if (home) {
35 | // refocus main content (after this component has updated)
36 | setTimeout(() => focus(home), 0);
37 | }
38 | };
39 | return /*#__PURE__*/_jsxs(Box, {
40 | borderStyle: "round",
41 | borderColor: "blue",
42 | children: [/*#__PURE__*/_jsx(Text, {
43 | children: label
44 | }), /*#__PURE__*/_jsx(SelectInput, {
45 | items,
46 | initialIndex,
47 | onSelect
48 | })]
49 | });
50 | } else {
51 | const minWidth = items.reduce((w, i) => Math.max(w, i.label.length + 2), 2);
52 | const item = items[initialIndex];
53 | return /*#__PURE__*/_jsxs(Box, {
54 | borderStyle: "round",
55 | children: [/*#__PURE__*/_jsx(Text, {
56 | children: label
57 | }), /*#__PURE__*/_jsx(Box, {
58 | minWidth: minWidth,
59 | children: /*#__PURE__*/_jsxs(Text, {
60 | children: [": ", item?.label]
61 | })
62 | })]
63 | });
64 | }
65 | }
66 | function extractHotkey(labelWithAmp) {
67 | const m = /(.*)&(\w)(.*)/.exec(labelWithAmp);
68 | if (!m) {
69 | return [labelWithAmp];
70 | }
71 | return [/*#__PURE__*/_jsxs(Text, {
72 | children: [m[1], /*#__PURE__*/_jsx(Text, {
73 | underline: true,
74 | children: m[2]
75 | }), m[3]]
76 | }), m[2].toUpperCase()];
77 | }
78 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJ1c2VGb2N1cyIsInVzZUlucHV0IiwiVGV4dCIsIkJveCIsIklua1NlbGVjdElucHV0IiwiZGVmYXVsdCIsIlNlbGVjdElucHV0IiwiU2VsZWN0Qm94IiwiaWQiLCJpdGVtcyIsImxhYmVsIiwibGFiZWxXaXRoQW1wIiwidmFsdWUiLCJob21lIiwib25TZWxlY3QiLCJvblNlbGVjdENhbGxlciIsImlzRm9jdXNlZCIsImZvY3VzIiwiaW5pdGlhbEluZGV4IiwiZmluZEluZGV4IiwiaSIsImhvdGtleSIsImV4dHJhY3RIb3RrZXkiLCJpbnB1dCIsInRvVXBwZXJDYXNlIiwiaXNBY3RpdmUiLCJpdGVtIiwic2V0VGltZW91dCIsIm1pbldpZHRoIiwicmVkdWNlIiwidyIsIk1hdGgiLCJtYXgiLCJsZW5ndGgiLCJtIiwiZXhlYyJdLCJzb3VyY2VzIjpbIlNlbGVjdEJveC5qc3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgdXNlRm9jdXMsIHVzZUlucHV0LCBUZXh0LCBCb3ggfSBmcm9tICdpbmsnO1xuaW1wb3J0IElua1NlbGVjdElucHV0IGZyb20gJ2luay1zZWxlY3QtaW5wdXQnOyBjb25zdCB7IGRlZmF1bHQ6IFNlbGVjdElucHV0IH0gPSBJbmtTZWxlY3RJbnB1dDtcblxuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gU2VsZWN0Qm94KHsgaWQsIGl0ZW1zLCBsYWJlbDogbGFiZWxXaXRoQW1wLCB2YWx1ZSwgaG9tZSwgb25TZWxlY3Q6IG9uU2VsZWN0Q2FsbGVyIH0pIHtcbiAgY29uc3QgeyBpc0ZvY3VzZWQsIGZvY3VzIH0gPSB1c2VGb2N1cyh7IGlkIH0pO1xuICBjb25zdCBpbml0aWFsSW5kZXggPSBpdGVtcy5maW5kSW5kZXgoaSA9PiBpLnZhbHVlID09PSB2YWx1ZSk7XG4gIGNvbnN0IFsgbGFiZWwsIGhvdGtleSBdID0gZXh0cmFjdEhvdGtleShsYWJlbFdpdGhBbXApO1xuICB1c2VJbnB1dCgoaW5wdXQpID0+IHtcbiAgICBpZiAoaW5wdXQudG9VcHBlckNhc2UoKSA9PT0gaG90a2V5KSB7XG4gICAgICBmb2N1cyhpZCk7XG4gICAgfVxuICB9LCB7IGlzQWN0aXZlOiAhaXNGb2N1c2VkIH0pO1xuICBpZiAoaXNGb2N1c2VkKSB7XG4gICAgY29uc3Qgb25TZWxlY3QgPSAoaXRlbSkgPT4ge1xuICAgICAgb25TZWxlY3RDYWxsZXI/LihpdGVtKTtcbiAgICAgIGlmIChob21lKSB7XG4gICAgICAgIC8vIHJlZm9jdXMgbWFpbiBjb250ZW50IChhZnRlciB0aGlzIGNvbXBvbmVudCBoYXMgdXBkYXRlZClcbiAgICAgICAgc2V0VGltZW91dCgoKSA9PiBmb2N1cyhob21lKSwgMCk7XG4gICAgICB9XG4gICAgfTtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveCBib3JkZXJTdHlsZT1cInJvdW5kXCIgYm9yZGVyQ29sb3I9XCJibHVlXCI+XG4gICAgICAgIDxUZXh0PntsYWJlbH08L1RleHQ+XG4gICAgICAgIDxTZWxlY3RJbnB1dCB7Li4ueyBpdGVtcywgaW5pdGlhbEluZGV4LCBvblNlbGVjdCB9fSAvPlxuICAgICAgPC9Cb3g+XG4gICAgKTtcbiAgfSBlbHNlIHtcbiAgICBjb25zdCBtaW5XaWR0aCA9IGl0ZW1zLnJlZHVjZSgodywgaSkgPT4gTWF0aC5tYXgodywgaS5sYWJlbC5sZW5ndGggKyAyKSwgMik7XG4gICAgY29uc3QgaXRlbSA9IGl0ZW1zW2luaXRpYWxJbmRleF07XG4gICAgcmV0dXJuIChcbiAgICAgIDxCb3ggYm9yZGVyU3R5bGU9XCJyb3VuZFwiPlxuICAgICAgICA8VGV4dD57bGFiZWx9PC9UZXh0PlxuICAgICAgICA8Qm94IG1pbldpZHRoPXttaW5XaWR0aH0+PFRleHQ+OiB7aXRlbT8ubGFiZWx9PC9UZXh0PjwvQm94PlxuICAgICAgPC9Cb3g+XG4gICAgKTtcbiAgfVxufVxuXG5mdW5jdGlvbiBleHRyYWN0SG90a2V5KGxhYmVsV2l0aEFtcCkge1xuICBjb25zdCBtID0gLyguKikmKFxcdykoLiopLy5leGVjKGxhYmVsV2l0aEFtcCk7XG4gIGlmICghbSkge1xuICAgIHJldHVybiBbIGxhYmVsV2l0aEFtcCBdO1xuICB9XG4gIHJldHVybiBbXG4gICAgPFRleHQ+e21bMV19PFRleHQgdW5kZXJsaW5lPnttWzJdfTwvVGV4dD57bVszXX08L1RleHQ+LFxuICAgIG1bMl0udG9VcHBlckNhc2UoKVxuICBdO1xufSJdLCJtYXBwaW5ncyI6IkFBQUEsU0FBU0EsUUFBUSxFQUFFQyxRQUFRLEVBQUVDLElBQUksRUFBRUMsR0FBRyxRQUFRLEtBQUs7QUFDbkQsT0FBT0MsY0FBYyxNQUFNLGtCQUFrQjtBQUFDO0FBQUE7QUFBQyxNQUFNO0VBQUVDLE9BQU8sRUFBRUM7QUFBWSxDQUFDLEdBQUdGLGNBQWM7QUFFOUYsZUFBZSxTQUFTRyxTQUFTLENBQUM7RUFBRUMsRUFBRTtFQUFFQyxLQUFLO0VBQUVDLEtBQUssRUFBRUMsWUFBWTtFQUFFQyxLQUFLO0VBQUVDLElBQUk7RUFBRUMsUUFBUSxFQUFFQztBQUFlLENBQUMsRUFBRTtFQUMzRyxNQUFNO0lBQUVDLFNBQVM7SUFBRUM7RUFBTSxDQUFDLEdBQUdqQixRQUFRLENBQUM7SUFBRVE7RUFBRyxDQUFDLENBQUM7RUFDN0MsTUFBTVUsWUFBWSxHQUFHVCxLQUFLLENBQUNVLFNBQVMsQ0FBQ0MsQ0FBQyxJQUFJQSxDQUFDLENBQUNSLEtBQUssS0FBS0EsS0FBSyxDQUFDO0VBQzVELE1BQU0sQ0FBRUYsS0FBSyxFQUFFVyxNQUFNLENBQUUsR0FBR0MsYUFBYSxDQUFDWCxZQUFZLENBQUM7RUFDckRWLFFBQVEsQ0FBRXNCLEtBQUssSUFBSztJQUNsQixJQUFJQSxLQUFLLENBQUNDLFdBQVcsRUFBRSxLQUFLSCxNQUFNLEVBQUU7TUFDbENKLEtBQUssQ0FBQ1QsRUFBRSxDQUFDO0lBQ1g7RUFDRixDQUFDLEVBQUU7SUFBRWlCLFFBQVEsRUFBRSxDQUFDVDtFQUFVLENBQUMsQ0FBQztFQUM1QixJQUFJQSxTQUFTLEVBQUU7SUFDYixNQUFNRixRQUFRLEdBQUlZLElBQUksSUFBSztNQUN6QlgsY0FBYyxHQUFHVyxJQUFJLENBQUM7TUFDdEIsSUFBSWIsSUFBSSxFQUFFO1FBQ1I7UUFDQWMsVUFBVSxDQUFDLE1BQU1WLEtBQUssQ0FBQ0osSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDO01BQ2xDO0lBQ0YsQ0FBQztJQUNELG9CQUNFLE1BQUMsR0FBRztNQUFDLFdBQVcsRUFBQyxPQUFPO01BQUMsV0FBVyxFQUFDLE1BQU07TUFBQSx3QkFDekMsS0FBQyxJQUFJO1FBQUEsVUFBRUg7TUFBSyxFQUFRLGVBQ3BCLEtBQUMsV0FBVztRQUFPRCxLQUFLO1FBQUVTLFlBQVk7UUFBRUo7TUFBUSxFQUFNO0lBQUEsRUFDbEQ7RUFFVixDQUFDLE1BQU07SUFDTCxNQUFNYyxRQUFRLEdBQUduQixLQUFLLENBQUNvQixNQUFNLENBQUMsQ0FBQ0MsQ0FBQyxFQUFFVixDQUFDLEtBQUtXLElBQUksQ0FBQ0MsR0FBRyxDQUFDRixDQUFDLEVBQUVWLENBQUMsQ0FBQ1YsS0FBSyxDQUFDdUIsTUFBTSxHQUFHLENBQUMsQ0FBQyxFQUFFLENBQUMsQ0FBQztJQUMzRSxNQUFNUCxJQUFJLEdBQUdqQixLQUFLLENBQUNTLFlBQVksQ0FBQztJQUNoQyxvQkFDRSxNQUFDLEdBQUc7TUFBQyxXQUFXLEVBQUMsT0FBTztNQUFBLHdCQUN0QixLQUFDLElBQUk7UUFBQSxVQUFFUjtNQUFLLEVBQVEsZUFDcEIsS0FBQyxHQUFHO1FBQUMsUUFBUSxFQUFFa0IsUUFBUztRQUFBLHVCQUFDLE1BQUMsSUFBSTtVQUFBLFdBQUMsSUFBRSxFQUFDRixJQUFJLEVBQUVoQixLQUFLO1FBQUE7TUFBUSxFQUFNO0lBQUEsRUFDdkQ7RUFFVjtBQUNGO0FBRUEsU0FBU1ksYUFBYSxDQUFDWCxZQUFZLEVBQUU7RUFDbkMsTUFBTXVCLENBQUMsR0FBRyxlQUFlLENBQUNDLElBQUksQ0FBQ3hCLFlBQVksQ0FBQztFQUM1QyxJQUFJLENBQUN1QixDQUFDLEVBQUU7SUFDTixPQUFPLENBQUV2QixZQUFZLENBQUU7RUFDekI7RUFDQSxPQUFPLGNBQ0wsTUFBQyxJQUFJO0lBQUEsV0FBRXVCLENBQUMsQ0FBQyxDQUFDLENBQUMsZUFBQyxLQUFDLElBQUk7TUFBQyxTQUFTO01BQUEsVUFBRUEsQ0FBQyxDQUFDLENBQUM7SUFBQyxFQUFRLEVBQUNBLENBQUMsQ0FBQyxDQUFDLENBQUM7RUFBQSxFQUFRLEVBQ3REQSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUNWLFdBQVcsRUFBRSxDQUNuQjtBQUNIIn0=
--------------------------------------------------------------------------------
/demo/ink/bin/main.mjs:
--------------------------------------------------------------------------------
1 | import { dirname } from 'path';
2 | import { Box, Text } from 'ink';
3 | import ModemSpeedSelect from "./ModemSpeedSelect.mjs";
4 | import BlinkingSelect from "./BlinkingSelect.mjs";
5 | import ScrollingSelect from "./ScrollingSelect.mjs";
6 | import TransparencySelect from "./TransparencySelect.mjs";
7 | import AnsiDisplay from "./AnsiDisplay.mjs";
8 | import AnsiSlideShow from "./AnsiSlideShow.mjs";
9 | import FileList from "./FileList.mjs";
10 | import { jsx as _jsx } from "react/jsx-runtime";
11 | import { jsxs as _jsxs } from "react/jsx-runtime";
12 | export default async function* main(methods) {
13 | const {
14 | wrap,
15 | manageRoute,
16 | manageEvents,
17 | replacing
18 | } = methods;
19 | const [parts] = manageRoute();
20 | const [on, eventual] = manageEvents();
21 | wrap(children => {
22 | return /*#__PURE__*/_jsxs(Box, {
23 | children: [/*#__PURE__*/_jsxs(Box, {
24 | flexDirection: "column",
25 | children: [/*#__PURE__*/_jsx(ModemSpeedSelect, {}), /*#__PURE__*/_jsx(BlinkingSelect, {}), /*#__PURE__*/_jsx(ScrollingSelect, {}), /*#__PURE__*/_jsx(TransparencySelect, {})]
26 | }), /*#__PURE__*/_jsx(Box, {
27 | children: children
28 | })]
29 | });
30 | });
31 | for (;;) {
32 | try {
33 | if (parts[0] === undefined) {
34 | replacing(() => parts[0] = 'list');
35 | } else if (parts[0] === 'list') {
36 | yield /*#__PURE__*/_jsx(FileList, {
37 | folder: parts[1],
38 | selected: parts[2],
39 | onFileSelect: on.file,
40 | onFolderSelect: on.folder
41 | });
42 | const {
43 | file,
44 | folder
45 | } = await eventual.file.or.folder;
46 | parts.splice(0);
47 | if (folder) {
48 | parts.push('list', folder);
49 | } else if (file) {
50 | parts.push('show', file);
51 | }
52 | } else if (parts[0] === 'show') {
53 | const path = parts[1];
54 | yield /*#__PURE__*/_jsx(AnsiDisplay, {
55 | src: parts[1],
56 | onExit: on.exit
57 | });
58 | await eventual.exit;
59 | parts.splice(0);
60 | parts.push('list', dirname(path), path);
61 | } else if (parts[0] === 'loop') {
62 | const paths = parts.slice(1);
63 | yield /*#__PURE__*/_jsx(AnsiSlideShow, {
64 | srcList: paths,
65 | onExit: on.exit
66 | });
67 | await eventual.exit;
68 | process.exit(0);
69 | } else {
70 | throw new Error(`Unrecognized command: ${parts[0]}`);
71 | }
72 | } catch (err) {
73 | // not sure why Ink complains when nothing is yielded prior to program exit
74 | yield null;
75 | console.error(err.message);
76 | process.exit(1);
77 | }
78 | }
79 | }
80 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJuYW1lcyI6WyJkaXJuYW1lIiwiQm94IiwiVGV4dCIsIk1vZGVtU3BlZWRTZWxlY3QiLCJCbGlua2luZ1NlbGVjdCIsIlNjcm9sbGluZ1NlbGVjdCIsIlRyYW5zcGFyZW5jeVNlbGVjdCIsIkFuc2lEaXNwbGF5IiwiQW5zaVNsaWRlU2hvdyIsIkZpbGVMaXN0IiwibWFpbiIsIm1ldGhvZHMiLCJ3cmFwIiwibWFuYWdlUm91dGUiLCJtYW5hZ2VFdmVudHMiLCJyZXBsYWNpbmciLCJwYXJ0cyIsIm9uIiwiZXZlbnR1YWwiLCJjaGlsZHJlbiIsInVuZGVmaW5lZCIsImZpbGUiLCJmb2xkZXIiLCJvciIsInNwbGljZSIsInB1c2giLCJwYXRoIiwiZXhpdCIsInBhdGhzIiwic2xpY2UiLCJwcm9jZXNzIiwiRXJyb3IiLCJlcnIiLCJjb25zb2xlIiwiZXJyb3IiLCJtZXNzYWdlIl0sInNvdXJjZXMiOlsibWFpbi5qc3giXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgZGlybmFtZSB9IGZyb20gJ3BhdGgnO1xuaW1wb3J0IHsgQm94LCBUZXh0IH0gZnJvbSAnaW5rJztcbmltcG9ydCBNb2RlbVNwZWVkU2VsZWN0IGZyb20gJy4vTW9kZW1TcGVlZFNlbGVjdC5qc3gnO1xuaW1wb3J0IEJsaW5raW5nU2VsZWN0IGZyb20gJy4vQmxpbmtpbmdTZWxlY3QuanN4JztcbmltcG9ydCBTY3JvbGxpbmdTZWxlY3QgZnJvbSAnLi9TY3JvbGxpbmdTZWxlY3QuanN4JztcbmltcG9ydCBUcmFuc3BhcmVuY3lTZWxlY3QgZnJvbSAnLi9UcmFuc3BhcmVuY3lTZWxlY3QuanN4JztcbmltcG9ydCBBbnNpRGlzcGxheSBmcm9tICcuL0Fuc2lEaXNwbGF5LmpzeCc7XG5pbXBvcnQgQW5zaVNsaWRlU2hvdyBmcm9tICcuL0Fuc2lTbGlkZVNob3cuanN4JztcbmltcG9ydCBGaWxlTGlzdCBmcm9tICcuL0ZpbGVMaXN0LmpzeCc7XG5cbmV4cG9ydCBkZWZhdWx0IGFzeW5jIGZ1bmN0aW9uKiBtYWluKG1ldGhvZHMpIHtcbiAgY29uc3QgeyB3cmFwLCBtYW5hZ2VSb3V0ZSwgbWFuYWdlRXZlbnRzLCByZXBsYWNpbmcgfSA9IG1ldGhvZHM7XG4gIGNvbnN0IFsgcGFydHMgXSA9IG1hbmFnZVJvdXRlKCk7XG4gIGNvbnN0IFsgb24sIGV2ZW50dWFsIF0gPSBtYW5hZ2VFdmVudHMoKTtcbiAgd3JhcCgoY2hpbGRyZW4pID0+IHtcbiAgICByZXR1cm4gKFxuICAgICAgPEJveD5cbiAgICAgICAgPEJveCBmbGV4RGlyZWN0aW9uPVwiY29sdW1uXCI+XG4gICAgICAgICAgPE1vZGVtU3BlZWRTZWxlY3QgLz5cbiAgICAgICAgICA8QmxpbmtpbmdTZWxlY3QgLz5cbiAgICAgICAgICA8U2Nyb2xsaW5nU2VsZWN0IC8+XG4gICAgICAgICAgPFRyYW5zcGFyZW5jeVNlbGVjdCAvPlxuICAgICAgICA8L0JveD5cbiAgICAgICAgPEJveD5cbiAgICAgICAgICB7Y2hpbGRyZW59XG4gICAgICAgIDwvQm94PlxuICAgICAgPC9Cb3g+XG4gICAgKTtcbiAgfSk7XG4gIGZvciAoOzspIHtcbiAgICB0cnkge1xuICAgICAgaWYgKHBhcnRzWzBdID09PSB1bmRlZmluZWQpIHtcbiAgICAgICAgcmVwbGFjaW5nKCgpID0+IHBhcnRzWzBdID0gJ2xpc3QnKVxuICAgICAgfSBlbHNlIGlmIChwYXJ0c1swXSA9PT0gJ2xpc3QnKSB7XG4gICAgICAgIHlpZWxkIDxGaWxlTGlzdCBmb2xkZXI9e3BhcnRzWzFdfSBzZWxlY3RlZD17cGFydHNbMl19IG9uRmlsZVNlbGVjdD17b24uZmlsZX0gb25Gb2xkZXJTZWxlY3Q9e29uLmZvbGRlcn0gLz47XG4gICAgICAgIGNvbnN0IHsgZmlsZSwgZm9sZGVyIH0gPSBhd2FpdCBldmVudHVhbC5maWxlLm9yLmZvbGRlcjtcbiAgICAgICAgcGFydHMuc3BsaWNlKDApO1xuICAgICAgICBpZiAoZm9sZGVyKSB7XG4gICAgICAgICAgcGFydHMucHVzaCgnbGlzdCcsIGZvbGRlcik7XG4gICAgICAgIH0gZWxzZSBpZiAoZmlsZSkge1xuICAgICAgICAgIHBhcnRzLnB1c2goJ3Nob3cnLCBmaWxlKTtcbiAgICAgICAgfVxuICAgICAgfSBlbHNlIGlmIChwYXJ0c1swXSA9PT0gJ3Nob3cnKSB7XG4gICAgICAgIGNvbnN0IHBhdGggPSBwYXJ0c1sxXTtcbiAgICAgICAgeWllbGQgPEFuc2lEaXNwbGF5IHNyYz17cGFydHNbMV19IG9uRXhpdD17b24uZXhpdH0gLz47XG4gICAgICAgIGF3YWl0IGV2ZW50dWFsLmV4aXQ7XG4gICAgICAgIHBhcnRzLnNwbGljZSgwKTtcbiAgICAgICAgcGFydHMucHVzaCgnbGlzdCcsIGRpcm5hbWUocGF0aCksIHBhdGgpO1xuICAgICAgfSBlbHNlIGlmIChwYXJ0c1swXSA9PT0gJ2xvb3AnKSB7XG4gICAgICAgIGNvbnN0IHBhdGhzID0gcGFydHMuc2xpY2UoMSk7XG4gICAgICAgIHlpZWxkIDxBbnNpU2xpZGVTaG93IHNyY0xpc3Q9e3BhdGhzfSBvbkV4aXQ9e29uLmV4aXR9IC8+O1xuICAgICAgICBhd2FpdCBldmVudHVhbC5leGl0O1xuICAgICAgICBwcm9jZXNzLmV4aXQoMCk7XG4gICAgICB9IGVsc2Uge1xuICAgICAgICB0aHJvdyBuZXcgRXJyb3IoYFVucmVjb2duaXplZCBjb21tYW5kOiAke3BhcnRzWzBdfWApO1xuICAgICAgfVxuICAgIH0gY2F0Y2ggKGVycikge1xuICAgICAgLy8gbm90IHN1cmUgd2h5IEluayBjb21wbGFpbnMgd2hlbiBub3RoaW5nIGlzIHlpZWxkZWQgcHJpb3IgdG8gcHJvZ3JhbSBleGl0XG4gICAgICB5aWVsZCBudWxsO1xuICAgICAgY29uc29sZS5lcnJvcihlcnIubWVzc2FnZSk7ICAgICAgXG4gICAgICBwcm9jZXNzLmV4aXQoMSk7XG4gICAgfVxuICB9XG59XG5cblxuIl0sIm1hcHBpbmdzIjoiQUFBQSxTQUFTQSxPQUFPLFFBQVEsTUFBTTtBQUM5QixTQUFTQyxHQUFHLEVBQUVDLElBQUksUUFBUSxLQUFLO0FBQy9CLE9BQU9DLGdC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
--------------------------------------------------------------------------------
/demo/ink/bin/cli.mjs:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env node
2 | import { render, useInput } from 'ink';
3 | import meow from 'meow';
4 | import meowhelp from 'cli-meow-help';
5 | import meowrev, { meowparse } from 'meow-reverse';
6 | import parse from 'shell-quote/parse.js';
7 | import quote from 'shell-quote/quote.js';
8 | import { createContext } from 'react';
9 | import { useSequential } from 'react-seq';
10 | import { useSequentialRouter } from 'array-router';
11 | import main from "./main.mjs";
12 | import { jsx as _jsx } from "react/jsx-runtime";
13 | const name = `ink-ansi-animation`;
14 | const commands = {
15 | 'show [FILE]': {
16 | desc: `Show an ANSI animation`
17 | },
18 | 'loop [FILE]...': {
19 | desc: `Show files in a loop`
20 | },
21 | 'list': {
22 | desc: `List ANSI files in current directory`
23 | }
24 | };
25 | const flags = {
26 | modemSpeed: {
27 | desc: `Emulate modem of specific baudrate`,
28 | alias: 'm',
29 | type: 'number',
30 | default: 56000
31 | },
32 | blinking: {
33 | desc: `Enable blinking text`,
34 | alias: 'b',
35 | type: 'boolean'
36 | },
37 | scrolling: {
38 | desc: `Enable scrolling`,
39 | alias: 's',
40 | type: 'boolean'
41 | },
42 | transparency: {
43 | desc: `Enable transparency`,
44 | alias: 't',
45 | type: 'boolean'
46 | }
47 | };
48 | const helpText = meowhelp({
49 | name,
50 | flags,
51 | commands
52 | });
53 | const options = {
54 | importMeta: import.meta,
55 | flags
56 | };
57 | const {
58 | input: parts,
59 | flags: query
60 | } = meow(helpText, options);
61 | function parseURL(_, {
62 | pathname
63 | }) {
64 | const argv = parse(pathname);
65 | const {
66 | input: parts,
67 | flags: query
68 | } = meowparse(argv, options);
69 | return {
70 | parts,
71 | query
72 | };
73 | }
74 | function createURL(_, {
75 | parts: input,
76 | query: flags
77 | }) {
78 | const argv = meowrev({
79 | input,
80 | flags
81 | }, options);
82 | const pathname = quote(argv);
83 | return new URL(`argv:${pathname}`);
84 | }
85 | function applyURL(currentURL) {
86 | globalThis.location.href = currentURL.href;
87 | }
88 | globalThis.location = createURL(null, {
89 | parts,
90 | query
91 | });
92 | const SpecialContext = createContext();
93 | function App() {
94 | useInput(input => {
95 | if (input.toLowerCase() === 'q') {
96 | process.exit(0);
97 | }
98 | });
99 | // use command-line URL
100 | const override = {
101 | createURL,
102 | parseURL,
103 | applyURL
104 | };
105 | const [parts, query, rMethods, {
106 | createContext,
107 | createBoundary
108 | }] = useSequentialRouter(override);
109 | return createContext(useSequential(sMethods => {
110 | const methods = {
111 | ...rMethods,
112 | ...sMethods
113 | };
114 | const {
115 | fallback,
116 | wrap,
117 | trap,
118 | reject
119 | } = methods;
120 | // default fallback (issue #142 in React-seq 0.9.0)
121 | fallback(null);
122 | // create error boundary
123 | wrap(children => createBoundary(children));
124 | // redirect error from boundary to generator function
125 | trap('error', err => reject(err));
126 | // method for managing route
127 | methods.manageRoute = () => [parts, query];
128 | return main(methods);
129 | }, [parts, query, rMethods, createBoundary]));
130 | }
131 | render( /*#__PURE__*/_jsx(App, {}));
132 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["render","useInput","meow","meowhelp","meowrev","meowparse","parse","quote","createContext","useSequential","useSequentialRouter","main","name","commands","desc","flags","modemSpeed","alias","type","default","blinking","scrolling","transparency","helpText","options","importMeta","import","meta","input","parts","query","parseURL","_","pathname","argv","createURL","URL","applyURL","currentURL","globalThis","location","href","SpecialContext","App","toLowerCase","process","exit","override","rMethods","createBoundary","sMethods","methods","fallback","wrap","trap","reject","children","err","manageRoute"],"sources":["cli.jsx"],"sourcesContent":["#!/usr/bin/env node\nimport { render, useInput } from 'ink';\nimport meow from 'meow';\nimport meowhelp from 'cli-meow-help';\nimport meowrev, { meowparse } from 'meow-reverse';\nimport parse from 'shell-quote/parse.js';\nimport quote from 'shell-quote/quote.js';\nimport { createContext } from 'react';\nimport { useSequential } from 'react-seq';\nimport { useSequentialRouter } from 'array-router';\nimport main from './main.jsx';\n\nconst name = `ink-ansi-animation`;\nconst commands = {\n  'show [FILE]': { desc: `Show an ANSI animation` },\n  'loop [FILE]...': { desc: `Show files in a loop` },\n\t'list': { desc: `List ANSI files in current directory` },\n};\nconst flags = {\n  modemSpeed: {\n    desc: `Emulate modem of specific baudrate`,\n    alias: 'm',\n    type: 'number',\n    default: 56000\n  },\n  blinking: {\n    desc: `Enable blinking text`,\n    alias: 'b',\n    type: 'boolean',\n  },\n  scrolling: {\n    desc: `Enable scrolling`,\n    alias: 's',\n    type: 'boolean',\n  },\n  transparency: {\n    desc: `Enable transparency`,\n    alias: 't',\n    type: 'boolean',\n  },\n};\n\nconst helpText = meowhelp({\tname,\tflags, commands });\nconst options = { importMeta: import.meta, flags };\nconst { input: parts, flags: query } = meow(helpText, options);\n\nfunction parseURL(_, { pathname }) {\n  const argv = parse(pathname);\n  const { input: parts, flags: query } = meowparse(argv, options);\n  return { parts, query };\n}\n\nfunction createURL(_, { parts: input, query: flags }) {\n  const argv = meowrev({ input, flags }, options);\n  const pathname = quote(argv);\n  return new URL(`argv:${pathname}`);\n}\n\nfunction applyURL(currentURL) {\n  globalThis.location.href = currentURL.href;\n}\n\nglobalThis.location = createURL(null, { parts, query });\n\nconst SpecialContext = createContext();\n\nfunction App() {\n  useInput((input) => {\n    if (input.toLowerCase() === 'q') {\n      process.exit(0);\n    }\n  })\n  // use command-line URL\n  const override = { createURL, parseURL, applyURL };\n  const [ parts, query, rMethods, { createContext, createBoundary } ] = useSequentialRouter(override);\n  return createContext(useSequential((sMethods) => {\n    const methods = { ...rMethods, ...sMethods };\n    const { fallback, wrap, trap, reject } = methods;\n    // default fallback (issue #142 in React-seq 0.9.0)\n    fallback(null);\n    // create error boundary\n    wrap(children => createBoundary(children));\n    // redirect error from boundary to generator function\n    trap('error', err => reject(err));\n    // method for managing route\n    methods.manageRoute = () => [ parts, query ];\n    return main(methods);\n  }, [ parts, query, rMethods, createBoundary ]));\n}\n\nrender(<App />);\n"],"mappings":"AAAA;AACA,SAASA,MAAM,EAAEC,QAAQ,QAAQ,KAAK;AACtC,OAAOC,IAAI,MAAM,MAAM;AACvB,OAAOC,QAAQ,MAAM,eAAe;AACpC,OAAOC,OAAO,IAAIC,SAAS,QAAQ,cAAc;AACjD,OAAOC,KAAK,MAAM,sBAAsB;AACxC,OAAOC,KAAK,MAAM,sBAAsB;AACxC,SAASC,aAAa,QAAQ,OAAO;AACrC,SAASC,aAAa,QAAQ,WAAW;AACzC,SAASC,mBAAmB,QAAQ,cAAc;AAClD,OAAOC,IAAI;AAAmB;AAE9B,MAAMC,IAAI,GAAI,oBAAmB;AACjC,MAAMC,QAAQ,GAAG;EACf,aAAa,EAAE;IAAEC,IAAI,EAAG;EAAwB,CAAC;EACjD,gBAAgB,EAAE;IAAEA,IAAI,EAAG;EAAsB,CAAC;EACnD,MAAM,EAAE;IAAEA,IAAI,EAAG;EAAsC;AACxD,CAAC;AACD,MAAMC,KAAK,GAAG;EACZC,UAAU,EAAE;IACVF,IAAI,EAAG,oCAAmC;IAC1CG,KAAK,EAAE,GAAG;IACVC,IAAI,EAAE,QAAQ;IACdC,OAAO,EAAE;EACX,CAAC;EACDC,QAAQ,EAAE;IACRN,IAAI,EAAG,sBAAqB;IAC5BG,KAAK,EAAE,GAAG;IACVC,IAAI,EAAE;EACR,CAAC;EACDG,SAAS,EAAE;IACTP,IAAI,EAAG,kBAAiB;IACxBG,KAAK,EAAE,GAAG;IACVC,IAAI,EAAE;EACR,CAAC;EACDI,YAAY,EAAE;IACZR,IAAI,EAAG,qBAAoB;IAC3BG,KAAK,EAAE,GAAG;IACVC,IAAI,EAAE;EACR;AACF,CAAC;AAED,MAAMK,QAAQ,GAAGpB,QAAQ,CAAC;EAAES,IAAI;EAAEG,KAAK;EAAEF;AAAS,CAAC,CAAC;AACpD,MAAMW,OAAO,GAAG;EAAEC,UAAU,EAAEC,MAAM,CAACC,IAAI;EAAEZ;AAAM,CAAC;AAClD,MAAM;EAAEa,KAAK,EAAEC,KAAK;EAAEd,KAAK,EAAEe;AAAM,CAAC,GAAG5B,IAAI,CAACqB,QAAQ,EAAEC,OAAO,CAAC;AAE9D,SAASO,QAAQ,CAACC,CAAC,EAAE;EAAEC;AAAS,CAAC,EAAE;EACjC,MAAMC,IAAI,GAAG5B,KAAK,CAAC2B,QAAQ,CAAC;EAC5B,MAAM;IAAEL,KAAK,EAAEC,KAAK;IAAEd,KAAK,EAAEe;EAAM,CAAC,GAAGzB,SAAS,CAAC6B,IAAI,EAAEV,OAAO,CAAC;EAC/D,OAAO;IAAEK,KAAK;IAAEC;EAAM,CAAC;AACzB;AAEA,SAASK,SAAS,CAACH,CAAC,EAAE;EAAEH,KAAK,EAAED,KAAK;EAAEE,KAAK,EAAEf;AAAM,CAAC,EAAE;EACpD,MAAMmB,IAAI,GAAG9B,OAAO,CAAC;IAAEwB,KAAK;IAAEb;EAAM,CAAC,EAAES,OAAO,CAAC;EAC/C,MAAMS,QAAQ,GAAG1B,KAAK,CAAC2B,IAAI,CAAC;EAC5B,OAAO,IAAIE,GAAG,CAAE,QAAOH,QAAS,EAAC,CAAC;AACpC;AAEA,SAASI,QAAQ,CAACC,UAAU,EAAE;EAC5BC,UAAU,CAACC,QAAQ,CAACC,IAAI,GAAGH,UAAU,CAACG,IAAI;AAC5C;AAEAF,UAAU,CAACC,QAAQ,GAAGL,SAAS,CAAC,IAAI,EAAE;EAAEN,KAAK;EAAEC;AAAM,CAAC,CAAC;AAEvD,MAAMY,cAAc,GAAGlC,aAAa,EAAE;AAEtC,SAASmC,GAAG,GAAG;EACb1C,QAAQ,CAAE2B,KAAK,IAAK;IAClB,IAAIA,KAAK,CAACgB,WAAW,EAAE,KAAK,GAAG,EAAE;MAC/BC,OAAO,CAACC,IAAI,CAAC,CAAC,CAAC;IACjB;EACF,CAAC,CAAC;EACF;EACA,MAAMC,QAAQ,GAAG;IAAEZ,SAAS;IAAEJ,QAAQ;IAAEM;EAAS,CAAC;EAClD,MAAM,CAAER,KAAK,EAAEC,KAAK,EAAEkB,QAAQ,EAAE;IAAExC,aAAa;IAAEyC;EAAe,CAAC,CAAE,GAAGvC,mBAAmB,CAACqC,QAAQ,CAAC;EACnG,OAAOvC,aAAa,CAACC,aAAa,CAAEyC,QAAQ,IAAK;IAC/C,MAAMC,OAAO,GAAG;MAAE,GAAGH,QAAQ;MAAE,GAAGE;IAAS,CAAC;IAC5C,MAAM;MAAEE,QAAQ;MAAEC,IAAI;MAAEC,IAAI;MAAEC;IAAO,CAAC,GAAGJ,OAAO;IAChD;IACAC,QAAQ,CAAC,IAAI,CAAC;IACd;IACAC,IAAI,CAACG,QAAQ,IAAIP,cAAc,CAACO,QAAQ,CAAC,CAAC;IAC1C;IACAF,IAAI,CAAC,OAAO,EAAEG,GAAG,IAAIF,MAAM,CAACE,GAAG,CAAC,CAAC;IACjC;IACAN,OAAO,CAACO,WAAW,GAAG,MAAM,CAAE7B,KAAK,EAAEC,KAAK,CAAE;IAC5C,OAAOnB,IAAI,CAACwC,OAAO,CAAC;EACtB,CAAC,EAAE,CAAEtB,KAAK,EAAEC,KAAK,EAAEkB,QAAQ,EAAEC,cAAc,CAAE,CAAC,CAAC;AACjD;AAEAjD,MAAM,eAAC,KAAC,GAAG,KAAG,CAAC"}
--------------------------------------------------------------------------------
/test/dom-components.test.js:
--------------------------------------------------------------------------------
1 | import { expect } from 'chai';
2 | import { readFile, stat } from 'fs/promises';
3 | import { createElement } from 'react';
4 | import { delay } from 'react-seq';
5 | import { withTestRenderer } from './test-renderer.js';
6 |
7 | import {
8 | AnsiText
9 | } from '../index.js';
10 |
11 | describe('DOM components', function() {
12 | describe('#AnsiText', function() {
13 | it('should produce empty block of text at min width and height when neither src or srcObject is provided', async function() {
14 | await withTestRenderer(async ({ render, toJSON }) => {
15 | const el = createElement(AnsiText, { minHeight: 4 });
16 | await render(el);
17 | const node = toJSON();
18 | expect(node).to.have.property('type', 'code');
19 | expect(node.props).to.have.property('className', 'AnsiText');
20 | expect(node.children).to.have.lengthOf(4);
21 | for (const { children: line } of node.children) {
22 | const segment = line[0];
23 | expect(segment).to.have.property('type', 'span');
24 | expect(segment.props.style).to.have.property('color', undefined);
25 | expect(segment.props.style).to.have.property('backgroundColor', undefined);
26 | const text = segment.children[0];
27 | expect(text).to.have.lengthOf(79);
28 | expect(text).to.match(/^\s+$/);
29 | }
30 | });
31 | })
32 | it('should output error message when fetch throws', async function() {
33 | await withTestRenderer(async ({ render, toJSON }) => {
34 | try {
35 | global.fetch = () => {
36 | // checking handling of non-ASCII characters
37 | throw new Error('Stało się coś strasznego');
38 | };
39 | const src = 'http://whatever';
40 | let error;
41 | const el = createElement(AnsiText, { src, minHeight: 4, onError: (err) => error = err });
42 | await render(el);
43 | const node = toJSON();
44 | expect(node).to.have.property('type', 'code');
45 | expect(node.props).to.have.property('className', 'AnsiText');
46 | expect(node.children).to.have.lengthOf(4);
47 | const segment = node.children[0].children[0];
48 | expect(segment).to.have.property('type', 'span');
49 | expect(segment.props.style).to.have.property('color', '#aaaaaa');
50 | expect(segment.props.style).to.have.property('backgroundColor', '#000000');
51 | const text = segment.children[0];
52 | expect(text).to.match(/^Sta\?o si\? co\? strasznego\s+/);
53 | expect(error).to.be.an('error');
54 | } finally {
55 | delete global.fetch;
56 | }
57 | });
58 | })
59 | it('should accept a buffer as srcObject', async function() {
60 | await withTestRenderer(async ({ render, toJSON }) => {
61 | const srcObject = await readFile(resolve('./ansi/LDA-GARFIELD.ANS'));
62 | const el = createElement(AnsiText, { srcObject, maxHeight: 1024 });
63 | await render(el);
64 | const node = toJSON();
65 | expect(node).to.have.property('type', 'code');
66 | expect(node.props).to.have.property('className', 'AnsiText');
67 | expect(node.children).to.have.lengthOf(40);
68 | });
69 | })
70 | it('should call onMetadata when file contains metadata', async function() {
71 | await withTestRenderer(async ({ render, toJSON }) => {
72 | const srcObject = await readFile(resolve('./ansi/LDA-GARFIELD.ANS'));
73 | let metadata = null;
74 | const el = createElement(AnsiText, { srcObject, modemSpeed: Infinity, onMetadata: (m) => metadata = m });
75 | await render(el);
76 | expect(metadata).to.be.an('array');
77 | });
78 | })
79 | it('should accept a promise as srcObject', async function() {
80 | await withTestRenderer(async ({ render, toJSON }) => {
81 | const srcObject = readFile(resolve('./ansi/LDA-GARFIELD.ANS'));
82 | const el = createElement(AnsiText, { srcObject, maxHeight: 1024 });
83 | await render(el);
84 | await delay(50);
85 | const node = toJSON();
86 | expect(node).to.have.property('type', 'code');
87 | expect(node.props).to.have.property('className', 'AnsiText');
88 | expect(node.children).to.have.lengthOf(40);
89 | });
90 | })
91 | it('should return status through onStatus', async function() {
92 | await withTestRenderer(async ({ render, toJSON }) => {
93 | const srcObject = await readFile(resolve('./ansi/LDA-GARFIELD.ANS'));
94 | let status = null;
95 | const el = createElement(AnsiText, {
96 | srcObject,
97 | maxHeight: 1024,
98 | onStatus: s => status = s,
99 | });
100 | await render(el);
101 | await delay(50);
102 | const node = toJSON();
103 | expect(node).to.have.property('type', 'code');
104 | expect(node.props).to.have.property('className', 'AnsiText');
105 | expect(node.children).to.have.lengthOf(40);
106 | expect(status.position).to.be.at.least(0).and.at.most(0.5);
107 | });
108 | })
109 | it('should display blinking text', async function() {
110 | await withTestRenderer(async ({ render, toJSON }) => {
111 | const srcObject = await readFile(resolve('./ansi/US-CANDLES.ANS'));
112 | const el = createElement(AnsiText, { srcObject, blinking: true, modemSpeed: Infinity, blinkDuration: 100 });
113 | await render(el);
114 | const node1 = toJSON();
115 | await delay(120);
116 | const node2 = toJSON();
117 | expect(node2).to.not.eql(node1);
118 | await delay(120);
119 | const node3 = toJSON();
120 | expect(node3).to.not.eql(node2);
121 | expect(node3).to.eql(node1);
122 | });
123 | })
124 | it('should leave out background color from undrawn area when transparency is on', async function() {
125 | await withTestRenderer(async ({ render, toJSON }) => {
126 | const srcObject = await readFile(resolve('./ansi/LDA-GARFIELD.ANS'));
127 | const el = createElement(AnsiText, { srcObject, transparency: true });
128 | await render(el);
129 | const node = toJSON();
130 | let transparentSegment;
131 | for (const line of node.children) {
132 | for (const segment of line.children) {
133 | if (!segment.props.style.backgroundColor) {
134 | transparentSegment = segment;
135 | }
136 | }
137 | }
138 | expect(transparentSegment).to.not.be.null;
139 | });
140 | })
141 | it('should load data through fetch when src is given', async function() {
142 | await withTestRenderer(async ({ render, toJSON }) => {
143 | let called = false;
144 | global.fetch = async function(path) {
145 | called = true;
146 | const data = await readFile(resolve(path));
147 | return {
148 | status: 200,
149 | statusText: 'OK',
150 | arrayBuffer: async () => data,
151 | };
152 | };
153 | try {
154 | const el = createElement(AnsiText, { src: './ansi/LDA-GARFIELD.ANS', maxHeight: 1024 });
155 | await render(el);
156 | await delay(10);
157 | expect(called).to.be.true;
158 | const node = toJSON();
159 | expect(node).to.have.property('type', 'code');
160 | expect(node.props).to.have.property('className', 'AnsiText');
161 | expect(node.children).to.have.lengthOf(40);
162 | } finally {
163 | delete global.fetch;
164 | }
165 | });
166 | })
167 | it('should display error when fetch does not return 200 OK', async function() {
168 | await withTestRenderer(async ({ render, toJSON }) => {
169 | let called = false;
170 | global.fetch = async function(path) {
171 | called = true;
172 | await delay(10);
173 | return {
174 | status: 404,
175 | statusText: 'Not Found',
176 | };
177 | };
178 | try {
179 | const el = createElement(AnsiText, { src: './ansi/LDA-GARFIELD.ANS' });
180 | await render(el);
181 | await delay(30);
182 | expect(called).to.be.true;
183 | const node = toJSON();
184 | expect(node).to.have.property('type', 'code');
185 | expect(node.props).to.have.property('className', 'AnsiText');
186 | expect(node.children).to.have.lengthOf(22);
187 | const segment = node.children[0].children[0];
188 | expect(segment).to.have.property('type', 'span');
189 | expect(segment.props.style).to.have.property('color', '#aaaaaa');
190 | expect(segment.props.style).to.have.property('backgroundColor', '#000000');
191 | const text = segment.children[0];
192 | expect(text).to.match(/^HTTP 404 \- Not Found\s+$/);
193 | } finally {
194 | delete global.fetch;
195 | }
196 | });
197 | })
198 | it('should blink text using CSS when palette is set to css', async function() {
199 | await withTestRenderer(async ({ render, toJSON }) => {
200 | const srcObject = await readFile(resolve('./ansi/US-CANDLES.ANS'));
201 | const el = createElement(AnsiText, {
202 | srcObject,
203 | blinking: true,
204 | modemSpeed: Infinity,
205 | blinkDuration: 100 ,
206 | palette: 'css',
207 | });
208 | await render(el);
209 | const node1 = toJSON();
210 | const segment = node1.children[0].children[0];
211 | expect(segment.props).to.not.have.property('style');
212 | expect(segment.props).to.have.property('className').that.matches(/fgColor\d+ bgColor\d+/);
213 | await delay(120);
214 | let blinkingSegment = null;
215 | for (const line of node1.children) {
216 | for (const segment of line.children) {
217 | if (/\bblink\b/.test(segment.props.className)) {
218 | blinkingSegment = segment;
219 | }
220 | }
221 | }
222 | expect(blinkingSegment).to.be.null;
223 | const node2 = toJSON();
224 | expect(node2).to.not.eql(node1);
225 | for (const line of node2.children) {
226 | for (const segment of line.children) {
227 | if (/\bblink\b/.test(segment.props.className)) {
228 | blinkingSegment = segment;
229 | }
230 | }
231 | }
232 | expect(blinkingSegment).to.not.be.null;
233 | expect(blinkingSegment.props.className).to.match(/fgColor\d+ bgColor\d+ blink\b/);
234 | });
235 | })
236 | it('should use CSS for blinking when blinking is also set to css', async function() {
237 | await withTestRenderer(async ({ render, toJSON }) => {
238 | const srcObject = await readFile(resolve('./ansi/US-CANDLES.ANS'));
239 | const el = createElement(AnsiText, {
240 | srcObject,
241 | blinking: 'css',
242 | modemSpeed: Infinity,
243 | blinkDuration: 100 ,
244 | palette: 'css',
245 | });
246 | await render(el);
247 | const node1 = toJSON();
248 | const segment = node1.children[0].children[0];
249 | expect(segment.props).to.not.have.property('style');
250 | expect(segment.props).to.have.property('className').that.matches(/fgColor\d+ bgColor\d+/);
251 | let blinkingSegment = null;
252 | for (const line of node1.children) {
253 | for (const segment of line.children) {
254 | if (/\bblinking\b/.test(segment.props.className)) {
255 | blinkingSegment = segment;
256 | }
257 | }
258 | }
259 | expect(blinkingSegment).to.not.be.null;
260 | expect(blinkingSegment.props.className).to.match(/fgColor\d+ bgColor\d+ blinking\b/);
261 | await delay(120);
262 | const node2 = toJSON();
263 | expect(node2).to.eql(node1);
264 | });
265 | })
266 | })
267 | })
268 |
--------------------------------------------------------------------------------
/demo/ink/bin/FileList.mjs:
--------------------------------------------------------------------------------
1 | import { readdir, open } from 'fs/promises';
2 | import { basename, normalize } from 'path';
3 | import { useProgressive } from 'react-seq';
4 | import { useFocus, Box, Text } from 'ink';
5 | import MulticolumnSelectInput from 'ink-multicolumn-select-input';
6 | import InkSpinner from 'ink-spinner';
7 | import { jsx as _jsx } from "react/jsx-runtime";
8 | import { jsxs as _jsxs } from "react/jsx-runtime";
9 | const {
10 | default: Spinner
11 | } = InkSpinner;
12 | export default function FileList({
13 | folder = '.',
14 | selected = '',
15 | onFileSelect,
16 | onFolderSelect
17 | }) {
18 | return useProgressive(async ({
19 | fallback,
20 | type,
21 | usable,
22 | defer
23 | }) => {
24 | fallback( /*#__PURE__*/_jsx(LoadingScreen, {}));
25 | type(FileListUI);
26 | usable(5);
27 | defer(100);
28 | return {
29 | folders: findSubfolder(folder),
30 | files: findAnsiFiles(folder),
31 | selected,
32 | onFileSelect,
33 | onFolderSelect
34 | };
35 | }, [folder]);
36 | }
37 |
38 | // typical ANSI files are 79x24
39 | const width = 81;
40 | const height = 26;
41 | const borderStyle = 'round';
42 | function FileListUI({
43 | folders = [],
44 | files = [],
45 | selected,
46 | onFileSelect,
47 | onFolderSelect
48 | }) {
49 | const {
50 | isFocused
51 | } = useFocus({
52 | id: 'main',
53 | autoFocus: true
54 | });
55 | const items = [];
56 | for (const folder of folders) {
57 | items.push({
58 | label: '\u{1F5C0} ' + basename(folder),
59 | value: folder,
60 | type: 'folder'
61 | });
62 | }
63 | for (const file of files) {
64 | items.push({
65 | label: '\u{1F5CF} ' + basename(file),
66 | value: file,
67 | type: 'file'
68 | });
69 | }
70 | // ensure that columns are wide enough for the longest filename
71 | const maxWidth = items.reduce((m, i) => m = Math.max(m, i.label.length), 0);
72 | const columnCount = Math.max(1, Math.floor(80 / (maxWidth + 2)));
73 | const limit = height - 2;
74 | const initialIndex = Math.max(0, items.findIndex(i => i.value === selected));
75 | const borderColor = isFocused ? 'blue' : undefined;
76 | const onSelect = ({
77 | value,
78 | type
79 | }) => {
80 | if (type === 'folder') {
81 | onFolderSelect?.(value);
82 | } else if (type === 'file') {
83 | onFileSelect?.(value);
84 | }
85 | };
86 | return /*#__PURE__*/_jsx(Box, {
87 | borderStyle,
88 | borderColor,
89 | width,
90 | children: /*#__PURE__*/_jsx(MulticolumnSelectInput, {
91 | items,
92 | limit,
93 | columnCount,
94 | isFocused,
95 | initialIndex,
96 | onSelect
97 | })
98 | });
99 | }
100 | export function LoadingScreen() {
101 | const alignItems = 'center';
102 | const justifyContent = 'center';
103 | return /*#__PURE__*/_jsx(Box, {
104 | borderStyle,
105 | width,
106 | height,
107 | alignItems,
108 | justifyContent,
109 | children: /*#__PURE__*/_jsxs(Text, {
110 | children: [" ", /*#__PURE__*/_jsx(Spinner, {}), " Loading"]
111 | })
112 | });
113 | }
114 | async function* findSubfolder(folder) {
115 | const list = await readdir(folder, {
116 | withFileTypes: true
117 | });
118 | yield normalize(`${folder}/..`);
119 | for (const entry of list) {
120 | if (entry.isDirectory() && !entry.name.startsWith('.')) {
121 | yield `${folder}/${entry.name}`;
122 | }
123 | }
124 | }
125 | async function* findAnsiFiles(folder) {
126 | const list = await readdir(folder, {
127 | withFileTypes: true
128 | });
129 | for (const entry of list) {
130 | if (entry.isFile() && !entry.name.startsWith('.')) {
131 | const path = `${folder}/${entry.name}`;
132 | if (await isAnsiFile(path)) {
133 | yield path;
134 | }
135 | }
136 | }
137 | }
138 | export async function isAnsiFile(path) {
139 | let file;
140 | try {
141 | file = await open(path);
142 | const buffer = Buffer.alloc(1024);
143 | const {
144 | bytesRead
145 | } = await file.read(buffer, 0, buffer.length);
146 | const array = new Uint8Array(buffer, 0, bytesRead);
147 | for (let i = 0; i < array.length - 1; i++) {
148 | if (array[i] === 0x1B && array[i + 1] === 0x5B) {
149 | return true;
150 | }
151 | }
152 | } catch (err) {} finally {
153 | await file?.close();
154 | }
155 | return false;
156 | }
157 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["readdir","open","basename","normalize","useProgressive","useFocus","Box","Text","MulticolumnSelectInput","InkSpinner","default","Spinner","FileList","folder","selected","onFileSelect","onFolderSelect","fallback","type","usable","defer","FileListUI","folders","findSubfolder","files","findAnsiFiles","width","height","borderStyle","isFocused","id","autoFocus","items","push","label","value","file","maxWidth","reduce","m","i","Math","max","length","columnCount","floor","limit","initialIndex","findIndex","borderColor","undefined","onSelect","LoadingScreen","alignItems","justifyContent","list","withFileTypes","entry","isDirectory","name","startsWith","isFile","path","isAnsiFile","buffer","Buffer","alloc","bytesRead","read","array","Uint8Array","err","close"],"sources":["FileList.jsx"],"sourcesContent":["import { readdir, open } from 'fs/promises';\nimport { basename, normalize } from 'path';\nimport { useProgressive } from 'react-seq';\nimport { useFocus, Box, Text } from 'ink';\nimport MulticolumnSelectInput from 'ink-multicolumn-select-input';\nimport InkSpinner from 'ink-spinner'; const { default: Spinner } = InkSpinner;\n\nexport default function FileList({ folder = '.', selected = '', onFileSelect, onFolderSelect }) {\n  return useProgressive(async ({ fallback, type, usable, defer }) => {\n    fallback(<LoadingScreen />);\n    type(FileListUI);\n    usable(5);\n    defer(100);\n    return {\n      folders: findSubfolder(folder),\n      files: findAnsiFiles(folder),\n      selected,\n      onFileSelect, \n      onFolderSelect,\n    };\n  }, [ folder ]);\n}\n\n// typical ANSI files are 79x24\nconst width = 81;\nconst height = 26;\nconst borderStyle = 'round';\n\nfunction FileListUI({ folders = [], files = [], selected, onFileSelect, onFolderSelect }) {\n  const { isFocused } = useFocus({ id: 'main', autoFocus: true });\n  const items = [];\n  for (const folder of folders) {\n    items.push({ label: '\\u{1F5C0} ' + basename(folder), value: folder, type: 'folder' })\n  }\n  for (const file of files) {\n    items.push({ label: '\\u{1F5CF} ' + basename(file), value: file, type: 'file' })\n  }\n  // ensure that columns are wide enough for the longest filename\n  const maxWidth = items.reduce((m, i) => m = Math.max(m, i.label.length), 0);\n  const columnCount = Math.max(1, Math.floor(80 / (maxWidth + 2)));\n  const limit = height - 2;\n  const initialIndex = Math.max(0, items.findIndex(i => i.value === selected));\n  const borderColor = (isFocused) ? 'blue' : undefined;\n  const onSelect = ({ value, type }) => {\n    if (type === 'folder') {\n      onFolderSelect?.(value);\n    } else if (type === 'file') {\n      onFileSelect?.(value);\n    }\n  };\n  return (\n    <Box {...{ borderStyle, borderColor, width }}>\n      <MulticolumnSelectInput {...{ items, limit, columnCount, isFocused, initialIndex, onSelect }} />\n    </Box>\n  );\n}\n\nexport function LoadingScreen() {\n  const alignItems = 'center';\n  const justifyContent = 'center';\n  return (\n    <Box {...{ borderStyle, width, height, alignItems, justifyContent }}>\n      <Text> <Spinner /> Loading</Text>\n    </Box>\n  );\n}\n\nasync function* findSubfolder(folder) {\n  const list = await readdir(folder, { withFileTypes: true });\n  yield normalize(`${folder}/..`);\n  for (const entry of list) {\n    if (entry.isDirectory() && !entry.name.startsWith('.')) {\n      yield `${folder}/${entry.name}`;\n    }\n  }\n}\n\nasync function* findAnsiFiles(folder) {\n  const list = await readdir(folder, { withFileTypes: true });\n  for (const entry of list) {\n    if (entry.isFile() && !entry.name.startsWith('.')) {\n      const path = `${folder}/${entry.name}`;\n      if (await isAnsiFile(path)) {\n        yield path;\n      }\n    }\n  }\n}\n\nexport async function isAnsiFile(path) {\n  let file;\n  try {\n    file = await open(path);\n    const buffer = Buffer.alloc(1024);\n    const { bytesRead } = await file.read(buffer, 0, buffer.length);\n    const array = new Uint8Array(buffer, 0, bytesRead);\n    for (let i = 0; i < array.length - 1; i++) {\n      if (array[i] === 0x1B && array[i + 1] === 0x5B) {\n        return true;\n      }\n    }\n  } catch (err) {\n  } finally {\n    await file?.close();\n  }\n  return false;\n}"],"mappings":"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"}
--------------------------------------------------------------------------------
/demo/ink/bin/MulticolumnSelectInput.mjs:
--------------------------------------------------------------------------------
1 | import { useState, useRef, useEffect, createElement } from 'react';
2 | import { Text, Box, useInput } from 'ink';
3 | export default function MulticolumnSelectInput(props) {
4 | const {
5 | items = [],
6 | initialIndex = 0,
7 | isFocused = true,
8 | limit = 24,
9 | columnCount = 4,
10 | width = '100%',
11 | indicatorComponent = Indicator,
12 | itemComponent = Item,
13 | onSelect,
14 | onHighlight
15 | } = props;
16 | const [selectedIndex, setSelectedIndex] = useState(initialIndex);
17 | const [columnOffset, setColumnOffset] = useState(calculateColumnOffset(selectedIndex, columnCount, limit));
18 | useInput((input, key) => {
19 | let newIndex = -1;
20 | if (input === 'k' || key.upArrow) {
21 | newIndex = selectedIndex - 1;
22 | } else if (input === 'j' || key.downArrow) {
23 | newIndex = selectedIndex + 1;
24 | } else if (input === 'h' || key.leftArrow) {
25 | newIndex = selectedIndex - limit;
26 | } else if (input === 'l' || key.rightArrow) {
27 | newIndex = selectedIndex + limit;
28 | if (newIndex >= items.length) {
29 | // jump to the last item only if we're in the second to last column
30 | const currentColumn = Math.floor((selectedIndex + 1) / limit);
31 | const lastColumn = Math.floor(items.length / limit);
32 | if (lastColumn > currentColumn) {
33 | newIndex = items.length - 1;
34 | }
35 | }
36 | } else if (input === '^' || input === '\u005BH') {
37 | newIndex = 0;
38 | } else if (input === '$' || input === '\u005BF') {
39 | newIndex = items.length - 1;
40 | } else if (key.return) {
41 | if (items[selectedIndex]) {
42 | onSelect?.(items[selectedIndex]);
43 | }
44 | }
45 | if (newIndex >= 0 && newIndex < items.length) {
46 | setSelectedIndex(newIndex);
47 | // adjust column offset so the selected item is in view
48 | setColumnOffset(calculateColumnOffset(initialIndex, columnCount, limit, columnOffset));
49 | onHighlight?.(items[newIndex]);
50 | }
51 | }, {
52 | isActive: isFocused
53 | });
54 | // reset selectIndex when items are different
55 | const previousItems = useRef();
56 | useEffect(() => {
57 | const {
58 | current
59 | } = previousItems;
60 | if (current && items.some((item, i) => item.value !== current[i].value)) {
61 | setSelectedIndex(initialIndex);
62 | setColumnOffset(calculateColumnOffset(initialIndex, columnCount, limit));
63 | }
64 | previousItems.current = items;
65 | }, [items, initialIndex, columnCount, limit]);
66 | const columns = [];
67 | for (let i = 0; i < columnCount; i++) {
68 | const rows = [];
69 | for (let j = 0, index = (columnOffset + i) * limit; j < limit && index < items.length; j++, index++) {
70 | const isSelected = index === selectedIndex;
71 | const indicator = createElement(indicatorComponent, {
72 | isSelected
73 | });
74 | const item = createElement(itemComponent, {
75 | ...items[index],
76 | isSelected
77 | });
78 | const row = createElement(Box, {
79 | key: index
80 | }, indicator, item);
81 | rows.push(row);
82 | }
83 | // use flex basis 0 so columns have the same width
84 | const column = createElement(Box, {
85 | key: i,
86 | flexDirection: 'column',
87 | flexGrow: 1,
88 | flexBasis: 0
89 | }, rows);
90 | columns.push(column);
91 | }
92 | return createElement(Box, {
93 | width
94 | }, columns);
95 | }
96 | function calculateColumnOffset(index, count, limit, offset = 0) {
97 | const columnIndex = Math.floor(Math.max(1, index + 1) / limit);
98 | let newOffset = offset;
99 | while (newOffset + count <= columnIndex) {
100 | newOffset++;
101 | }
102 | while (newOffset > columnIndex) {
103 | newOffset--;
104 | }
105 | return newOffset;
106 | }
107 | function Item({
108 | isSelected = false,
109 | label
110 | }) {
111 | return createElement(Text, {
112 | inverse: isSelected
113 | }, ` ${label} `);
114 | }
115 | function Indicator({
116 | isSelected
117 | }) {
118 | return null;
119 | }
120 | //# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["useState","useRef","useEffect","createElement","Text","Box","useInput","MulticolumnSelectInput","props","items","initialIndex","isFocused","limit","columnCount","width","indicatorComponent","Indicator","itemComponent","Item","onSelect","onHighlight","selectedIndex","setSelectedIndex","columnOffset","setColumnOffset","calculateColumnOffset","input","key","newIndex","upArrow","downArrow","leftArrow","rightArrow","length","currentColumn","Math","floor","lastColumn","return","isActive","previousItems","current","some","item","i","value","columns","rows","j","index","isSelected","indicator","row","push","column","flexDirection","flexGrow","flexBasis","count","offset","columnIndex","max","newOffset","label","inverse"],"sources":["MulticolumnSelectInput.jsx"],"sourcesContent":["import { useState, useRef, useEffect, createElement } from 'react';\nimport { Text, Box, useInput } from 'ink';\n\nexport default function MulticolumnSelectInput(props) {\n  const { \n    items = [],\n    initialIndex = 0,\n    isFocused = true,\n    limit = 24,\n    columnCount = 4,\n    width = '100%',\n    indicatorComponent = Indicator,\n    itemComponent = Item,    \n    onSelect,\n    onHighlight,\n  } = props;\n  const [ selectedIndex, setSelectedIndex ] = useState(initialIndex);\n  const [ columnOffset, setColumnOffset ] = useState(calculateColumnOffset(selectedIndex, columnCount, limit));\n  useInput((input, key) => {\n    let newIndex = -1;\n    if (input === 'k' || key.upArrow) {\n      newIndex = selectedIndex - 1;\n    } else if (input === 'j' || key.downArrow) {\n      newIndex = selectedIndex + 1;\n    } else if (input === 'h' || key.leftArrow) {\n      newIndex = selectedIndex - limit;\n    } else if (input === 'l' || key.rightArrow) {\n      newIndex = selectedIndex + limit;\n      if (newIndex >= items.length) {\n        // jump to the last item only if we're in the second to last column\n        const currentColumn = Math.floor((selectedIndex + 1) / limit);\n        const lastColumn = Math.floor(items.length / limit);\n        if (lastColumn > currentColumn) {\n          newIndex = items.length - 1;\n        }\n      }\n    } else if (input === '^' || input === '\\u005BH') {\n      newIndex = 0;\n    } else if (input === '$' || input === '\\u005BF') {\n      newIndex = items.length - 1;\n    } else if (key.return) {\n      if (items[selectedIndex]) {\n        onSelect?.(items[selectedIndex]);\n      }\n    }   \n    if (newIndex >= 0 && newIndex < items.length) {\n      setSelectedIndex(newIndex);\n      // adjust column offset so the selected item is in view\n      setColumnOffset(calculateColumnOffset(initialIndex, columnCount, limit, columnOffset));\n      onHighlight?.(items[newIndex]);\n    }\n  }, { isActive: isFocused });\n  // reset selectIndex when items are different \n\tconst previousItems = useRef();\n  useEffect(() => {\n    const { current } = previousItems;\n    if (current && items.some((item, i) => item.value !== current[i].value)) {\n      setSelectedIndex(initialIndex);\n      setColumnOffset(calculateColumnOffset(initialIndex, columnCount, limit));\n    }\n    previousItems.current = items;\n  }, [ items, initialIndex, columnCount, limit ]);\n  const columns = [];\n  for (let i = 0; i < columnCount; i++) {    \n    const rows = [];\n    for (let j = 0, index = (columnOffset + i) * limit; j < limit && index < items.length; j++, index++) {\n      const isSelected = (index === selectedIndex);\n      const indicator = createElement(indicatorComponent, { isSelected });\n      const item = createElement(itemComponent, { ...items[index], isSelected });\n      const row = createElement(Box, { key: index }, indicator, item);\n      rows.push(row);\n    }\n    // use flex basis 0 so columns have the same width\n    const column = createElement(Box, { key: i, flexDirection: 'column', flexGrow: 1, flexBasis: 0 }, rows);\n    columns.push(column);\n  }\n  return createElement(Box, { width }, columns);\n}\n\nfunction calculateColumnOffset(index, count, limit, offset = 0) {\n  const columnIndex = Math.floor(Math.max(1, index + 1) / limit);\n  let newOffset = offset;\n  while (newOffset + count <= columnIndex) {\n    newOffset++;\n  }\n  while (newOffset > columnIndex) {\n    newOffset--;\n  }\n  return newOffset;\n}\n\n\n\nfunction Item({ isSelected = false, label }) {\n\treturn createElement(Text, { inverse: isSelected }, ` ${label} `);\n}\n\nfunction Indicator({ isSelected }) {\n  return null;\n}"],"mappings":"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"}
--------------------------------------------------------------------------------
/src/hooks.js:
--------------------------------------------------------------------------------
1 | import { useRef, useEffect } from 'react';
2 | import { useSequentialState, delay } from 'react-seq';
3 | import { toCP437, cp437Chars } from './dos-environment.js';
4 |
5 | const defaultStatus = { position: 0, playing: true };
6 | const promisedData = new WeakMap();
7 |
8 | export function useAnsi(dataSource, options = {}) {
9 | const {
10 | modemSpeed = 56000,
11 | frameDuration = 50,
12 | blinkDuration = 500,
13 | blinking = false,
14 | transparency = false,
15 | minWidth = 79,
16 | minHeight = 22,
17 | maxWidth = 80,
18 | maxHeight = 25,
19 | initialStatus = defaultStatus,
20 | onStatus,
21 | onError,
22 | onMetadata,
23 | beep,
24 | } = options;
25 | const state = useSequentialState(async function*({ initial, mount, signal }) {
26 | // screen is at minimum dimensions and empty initially
27 | let state = {
28 | width: minWidth,
29 | height: minHeight,
30 | blinked: false,
31 | lines: Array(minHeight).fill([ { text: ' '.repeat(minWidth), fgColor: undefined, bgColor: undefined, blinking, blink: false } ]),
32 | willBlink: false,
33 | status: initialStatus,
34 | metadata: null,
35 | error: null,
36 | };
37 | let data = null, initialized = false, error = null;
38 | if (typeof(dataSource?.then) === 'function') {
39 | data = promisedData.get(dataSource);
40 | if (!data) {
41 | try {
42 | // set initial state now, since we need to wait for data to show up
43 | initial(state);
44 | initialized = true;
45 | data = await dataSource;
46 | } catch (err) {
47 | data = err.message;
48 | error = err;
49 | }
50 | // remember the data
51 | promisedData.set(dataSource, data);
52 | }
53 | } else {
54 | data = dataSource;
55 | }
56 | if (typeof(data) === 'string') {
57 | data = toCP437(data);
58 | }
59 | let chars = new Uint8Array(data);
60 | let detectedWidth = 0, detectedHeight = 0;
61 | // process data in two passes: the first determines the maximum extent of the contents
62 | // while the second pass actually outputs them
63 | for (let pass = 1; pass <= 2; pass++) {
64 | // screen states
65 | let width = detectedWidth, height = detectedHeight;
66 | let cursorX = 0, cursorY = 0, savedCursorX = 0, savedCursorY = 0, maxCursorX = 0, maxCursorY = 0;
67 | let bgColor = 0, fgColor = 7, bgColorBase = 0, fgColorBase = 7, bgBright = false, fgBright = false;
68 | let transparencyFlags = 0, bgSet = false, fgSet = false;
69 | let buffer = null, blinked = false, willBlink = false;
70 | let escapeSeq = null, eof = false, metadata = null, metaString = '';
71 | if (pass === 1) {
72 | // there's no need to do the first pass if the minimum dimensions match the maximum
73 | if (minWidth !== maxWidth || minHeight !== maxHeight) {
74 | chars.map(processCharacter);
75 | }
76 | detectedWidth = Math.max(minWidth, maxCursorX + 1);
77 | detectedHeight = Math.max(minHeight, maxCursorY + 1);
78 | } else if (pass === 2) {
79 | // calculate the number of frames during which blinking text stays visible or invisible
80 | const blinkFrameCount = Math.ceil(blinkDuration / frameDuration);
81 | let blinkFramesRemaining = blinkFrameCount;
82 | // create buffer, using 32-bit integers when handling transparency
83 | buffer = (transparency) ? new Uint32Array(width * height) : new Uint16Array(width * height);
84 | // fill buffer with default attributes
85 | buffer.fill(cell(0));
86 | metadata = [];
87 | // process data in a multiple chunks
88 | const animationSpeed = modemSpeed / 10 / 1000;
89 | const chunks = [];
90 | let i = 0;
91 | if (initialStatus.position > 0) {
92 | // add initial chunk
93 | i = Math.floor(initialStatus.position * chars.length);
94 | chunks.push(chars.subarray(0, i));
95 | }
96 | if (initialStatus.playing) {
97 | // add remaining chunks
98 | const chunkLength = Math.floor(animationSpeed * frameDuration);
99 | while (i < chars.length) {
100 | chunks.push(chars.subarray(i, i + chunkLength));
101 | i += chunkLength;
102 | }
103 | }
104 | let processed = 0;
105 | for (const [ index, chunk ] of chunks.entries()) {
106 | chunk.map(processCharacter);
107 | // time to output what's held in the screen buffer to the hook consumer,
108 | // consolidating characters with identical attributes into segments
109 | const lines = scanBuffer();
110 | // calculate status
111 | processed += chunk.length;
112 | const playing = (index !== chunks.length - 1);
113 | const position = processed / chars.length;
114 | const status = { position, playing };
115 | state = { width, height, blinking, blinked, lines, willBlink, status, metadata, error };
116 | if (!initialized) {
117 | // initialize with real contents
118 | initial(state);
119 | initialized = true;
120 | await mount();
121 | } else {
122 | yield state;
123 | }
124 | if (playing) {
125 | // wait for frame to end
126 | await delay(frameDuration, { signal });
127 | if (blinking) {
128 | // update blink states
129 | blinkFramesRemaining--;
130 | if (blinkFramesRemaining === 0) {
131 | blinked = !blinked;
132 | blinkFramesRemaining = blinkFrameCount;
133 | }
134 | }
135 | }
136 | }
137 | data = chars = buffer = null;
138 |
139 | // go into an endless loop if there's blinking text (unless blinking is just truthy and not true)
140 | if (state.willBlink && blinking === true) {
141 | // wait out the remaining blink period
142 | await delay(frameDuration * blinkFramesRemaining, { signal });
143 | for (;;) {
144 | blinked = !blinked;
145 | yield { ...state, blinked };
146 | await delay(blinkDuration, { signal });
147 | }
148 | }
149 | }
150 |
151 | // --- helper functions below ----
152 |
153 | function cell(c) {
154 | // pack text attributes and codepoint into 16-bit cell
155 | return (bgColor << 8) | (fgColor << 12) | c | transparencyFlags;
156 | }
157 |
158 | function setCharacter(c) {
159 | if (cursorY >= maxHeight) {
160 | if (buffer) {
161 | // scroll up
162 | processCommand('S', `${cursorY - maxHeight + 1}`);
163 | }
164 | cursorY = maxHeight - 1;
165 | }
166 | if (!buffer) {
167 | if (cursorX > maxCursorX) {
168 | maxCursorX = cursorX;
169 | }
170 | if (cursorY > maxCursorY) {
171 | maxCursorY = cursorY;
172 | }
173 | } else {
174 | buffer[cursorY * width + cursorX] = cell(c);
175 | }
176 | cursorX++;
177 | if (cursorX >= maxWidth) {
178 | cursorX = 0;
179 | cursorY++;
180 | }
181 | }
182 |
183 | function parseOne(text, def1) {
184 | return (text) ? parseInt(text) : def1;
185 | }
186 |
187 | function parseTwo(text, def1, def2) {
188 | const parts = text.split(';');
189 | return [ parseOne(parts[0], def1), parseOne(parts[1], def2) ];
190 | }
191 |
192 | function parseMultiple(text, def) {
193 | const parts = text.split(';');
194 | return parts.map(p => parseOne(p, def));
195 | }
196 |
197 | function processCharacter(c) {
198 | if (escapeSeq) {
199 | if (escapeSeq.length === 1) {
200 | escapeSeq.push(c);
201 | if (c !== 0x5b) {
202 | // invalid sequence
203 | for (const c of escapeSeq) {
204 | setCharacter(c);
205 | }
206 | escapeSeq = null;
207 | }
208 | } else {
209 | if (c >= 0x40 && c <= 0x7e) {
210 | // @ to ~
211 | const cmd = cp437Chars[c];
212 | const params = escapeSeq.slice(2).map(c => cp437Chars[c]).join('');
213 | processCommand(cmd, params);
214 | escapeSeq = null;
215 | } else {
216 | escapeSeq.push(c);
217 | }
218 | }
219 | } else if (!eof) {
220 | if (c === 0x07) {
221 | beep?.();
222 | } else if (c === 0x08) {
223 | // backspace
224 | cursorX--;
225 | if (cursorX < 0) {
226 | cursorX = 0;
227 | }
228 | } else if (c === 0x09) {
229 | // tabs
230 | cursorX = ((cursorX >> 3) << 3) + 8;
231 | } else if (c === 0x0a) {
232 | // linefeed
233 | cursorY++;
234 | } else if (c === 0x0c) {
235 | // clear screen
236 | processCommand('J', 2);
237 | } else if (c === 0x0d) {
238 | // carriage return
239 | cursorX = 0;
240 | } else if (c === 0x1a) {
241 | eof = true;
242 | } else if (c === 0x1b) {
243 | escapeSeq = [ c ];
244 | } else {
245 | setCharacter(c);
246 | }
247 | } else {
248 | // metadata
249 | if (metadata) {
250 | if (c === 0 || c === 0x1a) {
251 | if (metaString) {
252 | metadata.push(metaString);
253 | metaString = '';
254 | }
255 | } else {
256 | metaString += cp437Chars[c];
257 | }
258 | }
259 | }
260 | }
261 |
262 | function processCommand(cmd, params = '') {
263 | if (cmd === 'A') {
264 | const count = parseOne(params, 1);
265 | cursorY -= count;
266 | if (cursorY < 0) {
267 | cursorY = 0;
268 | }
269 | } else if (cmd === 'B') {
270 | const count = parseOne(params, 1);
271 | cursorY += count;
272 | if (cursorY >= maxHeight) {
273 | cursorY = maxHeight - 1;
274 | }
275 | } else if (cmd === 'C') {
276 | const count = parseOne(params, 1);
277 | cursorX += count;
278 | if (cursorX >= maxWidth) {
279 | cursorX = maxWidth - 1;
280 | }
281 | } else if (cmd === 'D') {
282 | const count = parseOne(params, 1);
283 | cursorX -= count;
284 | if (cursorX < 0) {
285 | cursorX = 0;
286 | }
287 | } else if (cmd === 'H' || cmd === 'f') {
288 | const [ row, col ] = parseTwo(params, 1, 1);
289 | cursorX = Math.min(col, maxWidth) - 1;
290 | cursorY = Math.min(row, maxHeight)- 1;
291 | } else if (cmd === 'J') {
292 | // clear screen
293 | const mode = parseOne(params, 0);
294 | if (buffer) {
295 | let start, end;
296 | if (mode === 0) {
297 | start = cursorY * width + cursorX;
298 | end = width * height;
299 | } else if (mode === 1) {
300 | start = 0;
301 | end = cursorY * width + cursorX;
302 | } else if (mode === 2) {
303 | start = 0;
304 | end = width * height;
305 | }
306 | buffer.fill(cell(0), start, end);;
307 | }
308 | if (mode === 2) {
309 | cursorX = 0;
310 | cursorY = 0;
311 | }
312 | } else if (cmd === 'K') {
313 | // clear line to end
314 | const mode = parseOne(params, 0);
315 | if (buffer) {
316 | let start, end;
317 | if (mode === 0) {
318 | start = cursorY * width + cursorX;
319 | end = (cursorY + 1) * width;
320 | } else if (mode === 1) {
321 | start = cursorY * width;
322 | end = start + cursorX;
323 | } else if (mode === 2) {
324 | start = cursorY * width;
325 | end = start + width;
326 | }
327 | buffer.fill(cell(0), start, end);
328 | }
329 | } else if (cmd === 'L') {
330 | // insert line
331 | const count = parseOne(params, 1);
332 | if (buffer) {
333 | const target = (cursorY + count) * width;
334 | const source = cursorY * width;
335 | buffer.copyWithin(target, source);
336 | const start = source;
337 | const end = target;
338 | buffer.fill(cell(0), start, end);
339 | }
340 | if (cursorY <= maxCursorY) {
341 | maxCursorY += count;
342 | }
343 | } else if (cmd === 'M') {
344 | // delete line
345 | const count = parseOne(params, 1);
346 | if (buffer) {
347 | const target = cursorY * width;
348 | const source = (cursorY + count) * width;
349 | buffer.copyWithin(target, source);
350 | const start = source;
351 | const end = width * height;
352 | buffer.fill(cell(0), start, end);
353 | }
354 | } else if (cmd === 'P') {
355 | // delete characters
356 | const count = parseOne(params, 1);
357 | if (buffer) {
358 | const target = cursorY * width + cursorX;
359 | const source = target + count;
360 | const last = (cursorY + 1) * width;
361 | buffer.copyWithin(target, source, last);
362 | const start = last - count;
363 | const end = last;
364 | buffer.fill(cell(0), start, end);
365 | }
366 | } else if (cmd === 'S') {
367 | // scroll up
368 | const count = parseOne(params, 1);
369 | if (buffer) {
370 | const target = 0;
371 | const source = count * width;
372 | buffer.copyWithin(target, source);
373 | const start = width * (height - count);
374 | const end = width * height;
375 | buffer.fill(cell(0), start, end);
376 | }
377 | } else if (cmd === 'T') {
378 | // scroll down
379 | const count = parseOne(params, 1);
380 | if (buffer) {
381 | const target = count * width;
382 | const source = 0;
383 | buffer.copyWithin(target, source);
384 | const start = 0;
385 | const end = count * width;
386 | buffer.fill(cell(0), start, end);
387 | }
388 | } else if (cmd === 'X') {
389 | // clear characters
390 | const count = parseOne(params, 1);
391 | if (buffer) {
392 | const start = cursorY * width + cursorX;
393 | const end = Math.min(start + count, (cursorY + 1) * width);
394 | buffer.fill(cell(0), start, end);
395 | }
396 | } else if (cmd === 'm') {
397 | // modify text properties
398 | const modifiers = parseMultiple(params, 0);
399 | for (const m of modifiers) {
400 | if (m === 0) {
401 | fgBright = false;
402 | bgBright = false;
403 | fgColorBase = 7;
404 | bgColorBase = 0;
405 | fgSet = false;
406 | bgSet = false;
407 | } else if (m === 1) {
408 | fgBright = true;
409 | } else if (m === 2 || m === 22) {
410 | fgBright = false;
411 | } else if (m === 5 || m === 6) {
412 | bgBright = true;
413 | } else if (m === 7) {
414 | const fgColorBefore = fgColorBase;
415 | fgColorBase = bgColorBase;
416 | bgColorBase = fgColorBefore;
417 | fgSet = true;
418 | bgSet = true;
419 | } else if (m === 8) {
420 | fgColorBase = bgColorBase;
421 | fgSet = true;
422 | } else if (m === 25) {
423 | bgBright = false;
424 | } else if (m >= 30 && m <= 37) {
425 | fgColorBase = m - 30;
426 | fgSet = true;
427 | } else if (m >= 40 && m <= 47) {
428 | bgColorBase = m - 40;
429 | bgSet = true;
430 | }
431 | }
432 | fgColor = fgColorBase + (fgBright ? 8 : 0);
433 | bgColor = bgColorBase + (bgBright ? 8 : 0);
434 | if (transparency) {
435 | transparencyFlags = (fgSet ? 0x00010000 : 0) | (bgSet ? 0x00020000 : 0);
436 | }
437 | } else if (cmd === 's') {
438 | savedCursorX = cursorX;
439 | savedCursorY = cursorY;
440 | } else if (cmd === 'u') {
441 | cursorX = savedCursorX;
442 | cursorY = savedCursorY;
443 | }
444 | }
445 |
446 | function scanBuffer() {
447 | const lines = [];
448 | const blinkMask = (blinking) ? 0x0800 : 0x0000;
449 | const bgColorMask = (blinking) ? 0x0700 : 0x0F00;
450 | const fgColorMask = 0xF000;
451 | const fgMask = 0x00010000;
452 | const bgMask = 0x00020000;
453 | for (let row = 0; row < height; row++) {
454 | const segments = [];
455 | const first = row * width;
456 | const last = first + width;
457 | let attr = 0x00FF; // invalid attributes
458 | let text = '';
459 | // find where there's a change in attributes
460 | for (let i = first; i < last; i++) {
461 | const cp = buffer[i] & 0x00FF;
462 | const newAttr = buffer[i] & 0x000FFF00;
463 | if (attr !== newAttr) {
464 | // add preceding text
465 | if (text.length > 0) {
466 | segments.push({ attr, text });
467 | }
468 | attr = newAttr;
469 | text = '';
470 | }
471 | // map codepoint 0 to space
472 | text += cp437Chars[cp || 0x20];
473 | }
474 | // add leftover at end of line
475 | if (text.length > 0) {
476 | segments.push({ attr, text });
477 | }
478 | const line = [];
479 | for (const { attr, text } of segments) {
480 | const blink = (attr & blinkMask) !== 0;
481 | const fgColor = (!transparency || (attr & fgMask)) ? (attr & fgColorMask) >> 12 : undefined;
482 | const bgColor = (!transparency || (attr & bgMask)) ? (attr & bgColorMask) >> 8 : undefined;
483 | line.push({ text, fgColor, bgColor, blink });
484 | willBlink = willBlink || blink;
485 | }
486 | lines.push(line);
487 | }
488 | return lines;
489 | }
490 | }
491 | if (!initialized) {
492 | // the data source was empty--initialize with empty screen
493 | initial(state);
494 | }
495 | }, [ dataSource, modemSpeed, frameDuration, blinkDuration, blinking, minWidth, minHeight, maxWidth, maxHeight, transparency, initialStatus, beep ]);
496 | // saving handlers into a ref so we don't trigger useEffect when they're different
497 | const handlerRef = useRef();
498 | handlerRef.current = { onStatus, onMetadata, onError };
499 | // relay events to event handlers
500 | const { status, metadata, error } = state;
501 | useEffect(() => {
502 | handlerRef.current.onStatus?.(status);
503 | }, [ status ]);
504 | useEffect(() => {
505 | handlerRef.current.onMetadata?.(metadata);
506 | }, [ metadata ]);
507 | useEffect(() => {
508 | if (error) {
509 | handlerRef.current.onError?.(error);
510 | }
511 | }, [ error ]);
512 | return state;
513 | }
514 |
515 |
--------------------------------------------------------------------------------