Skip to content

Enable click, hover, drag and scroll within ink components

License

Notifications You must be signed in to change notification settings

zenobi-us/ink-mouse

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

18 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

ink-mouse

Provides mouse support for Ink components.

simplescreenrecorder-2024-06-26_00.29.29.mp4

Features

  • Mouse position tracking
  • Mouse hover tracking
  • Mouse click tracking
  • Mouse drag tracking
  • Element position tracking

Todo

Usage

import React, { useMemo, useRef, useState } from 'react';
import type { ComponentProps } from 'react';
import { Box, DOMElement, Text, render } from 'ink';
import {
  MouseProvider,
  useOnMouseHover,
  useMousePosition,
  useElementPosition,
  useOnMouseClick,
} from '@zenobius/ink-mouse';

function App() {
  return (
    <MouseProvider>
      <MyComponent />
    </MouseProvider>
  );
}

function MyComponent() {
  const mouse = useMousePosition();

  return (
    <Box>
      <Button label="Button 1" />
      <Text>
        {mouse.x}, {mouse.y}
      </Text>
    </Box>
  );
}

function Button({ label, onClick }: { label: string; onClick?: () => void }) {
  const ref = useRef<DOMElement>(null);

  const [hovering, setHovering] = useState(false);
  const [clicking, setClicking] = useState(false);

  useOnMouseClick(ref, (event) => {
    setClicking(event);
    if (event && typeof onClick === 'function') {
      onClick();
    }
  });
  useOnMouseHover(ref, setHovering);

  const border = useMemo((): ComponentProps<typeof Box>['borderStyle'] => {
    if (clicking) {
      return 'double';
    }

    if (hovering) {
      return 'singleDouble';
    }

    return 'single';
  }, [clicking, hovering]);

  return (
    <Box
      gap={1}
      paddingX={1}
      ref={ref}
      borderStyle={border}
    >
      <Text>{label}</Text>
    </Box>
  );
}

render(<App />);

Contributing

This project uses ASDF and Yarn.

  1. clone repo
  2. make branch: git checkout [fix|feat|docs|chore]/blah-blah (see Conventional Commits
  3. run ./tools.sh
  4. run just setup
  5. run just demo
  6. run just unittest
  7. run just lint

Todo

  • Support absolute elements
    • Elements positioned absolutely that occupy same space as other elements will mean they both recieve click and hover events.
    • Ink supports absolute positioning. I think z order is based on order rendered.
    • This means to simluate knowing the z order, we might need to register the order in which elements subscribe to events?
  • Add tests.
    • testing a device may be difficult; but the implementation is sufficiently abstracted from the device that it should be possible to mock the device input stream.
      • stdin event stream parsing
      • position tracking
      • click tracking
      • drag tracking
      • z order priority
  • Add Drag examples
  • Research Windows support
    • Apparently old CMD.exe only supports rudimentary ansii escape codes for colours.
    • New Windows Terminal does support ansi escape codes, so we'd have to explore what works and what doesn't.
    • We might have to fall back to GPM or some other library. Seems a bit complex. want to avoid it if possible.
  • Add support for right and middle click.
    • I think these are supported by the terminal, but I'm not sure how to detect them. Is it lowercase M and R?