0.0.10 • Published 5 months ago

event-ops v0.0.10

Weekly downloads
-
License
-
Repository
-
Last release
5 months ago

Event Ops

Event Ops (Event Operations) is a library that provides very simple and minimalistic utils to create event-driven programs. Event Ops was designed to work in any JavaScript environment and doesn't have any dependencies. With that said, it does provide additional helpers to work with React. Here's an example of how one can use Event Ops to write an event-driven controller:

import { Memo, State } from 'event-ops';

export class Calculator {
  readonly num1 = new State(100);
  readonly num2 = new State(200);

  readonly sum = new Memo(() => {
    return num1.value + num2.value;
  }, [this.num1, this.num2]);
}

To use sumState in a React component, you can import some helpers from event-ops/react:

import { useValue } from 'event-ops/react';

export function SumLabel() {
  const calculator = useCalculator();
  const sum = useValue(calculator.sumState);

  return (
    <div>
      sum: {sum}
    </div>
  );
}

For a robust example, I recommend you to look at the parallax-chess project.

Installation

Event Ops can be installed via NPM:

npm install event-ops

API

core/Event

Event is the most basic object in Event Ops. With an Event, you can either listen to a change, or emit a change.

import { Event } from 'event-ops';

const clickEvent = new Event<[number, number]>();

const clearClickListener = clickEvent.listen(([x, y]) => {
  console.log(`Click coord: ${x},${y}`);
});

clickEvent.emit([100, 150]);

// ... some time later ...

clearClickListener();

// OR

clickEvent.unlisten(clickListener);

core/EmitterEvent

In case you have an existing emitter, let's say a web socket, you can construct an event that's bound to a key using the EmitterEvent object. This is particularly useful if the instance of the emitter can vary, but the event key and value remain the same, e.g. if you have multiple web socket clients that subscribe to the same event.

import { EmitterEvent } from 'event-ops';

const pingEvent = new EmitterEvent('ping');
const pongEvent = new EmitterEvent<number>('pong');

wsServer.on('connection', (wsClient) => {
  pingEvent.listen(wsClient, () => {
    const ts = Date.now();
    pongEvent.emit(wsClient, ts);
  });
});

core/State

A State is an object that can hold a persistant value. Similarly to an Event, you can listen to changes, except they're emitted automatically based on the State value.

import { State } from 'event-ops';

const colorState = new State('red');

const clearColorListener = colorState.listen((color) => {
  console.log(`Color changed to ${color}`)
});

colorState.value = 'green';

// ... some time later ...

clearColorListener();

// OR

colorState.unlisten(colorListener);

You can also force a State change using a Value object:

import { State, Value } from 'event-ops';

const colorState = new State('red');

colorState.listen(() => {
  console.log('Color is still red')
});

colorState.value = new Value('red');

core/Effect

With an Effect, you can listen to multiple Events using a unified callback.

import { State, Effect } from 'event-ops';

const num1State = new State(0);
const num2State = new State(0);
const sumEffect = new Effect([num1State, num2State]);

const clearSumListener = sumEffect.listen(() => {
  console.log(`New sum is ${num1State.value + num2State.value}`);
});

num1State.value = 100;
num2State.value = 200;

// ... some time later ...

clearSumListener();

// OR

sumEffect.unlisten(sumListener);

core/Memo

Memo provides an alternative to a LazyValue assignment in an Effect listener.

import { State, Memo } from 'event-ops';

const num1 = new State(0);
const num2 = new State(0);

const sum = new Memo(() => {
  return num1.value + num2.value;
}, [num1, num2]);

// Which is equivalent to

const num1 = new State(0);
const num2 = new State(0);
const sum = new State(0);

const effect = new Effect(() => {
  sum.value = new LazyValue(() => num1.value + num2.value);
}, [num1, num2]);

effect.emit();

You can stop listening to the underlying Memo Effect by calling the clearEffectListener() method.

const sum = new Memo(() => {
  return num1.value = num2.value;
}, [num1, num2]);

// ... some time later ...

sum.clearEffectListener();

core/Task

All listeners are scheduled to run using a Task system to prevent redundant computations. To tap into the Task system, you can use the Task object.

import { State, Effect, Task } from 'event-ops';

const num1State = new State(0);
const num2State = new State(0);

new Effect([num1State, num2State]).listen(() => {
  console.log(`New sum is ${num1State.value + num2State.value}`);
});

// Here we emit only a single change by scheduling a Task

new Task(() => {
  num1State.value = 100;
  num2State.value = 200;
}).schedule();

If some time later you would like to clear a scheduled Task, you can do so with the unschedule() method. This would only be relevant if the Task has not been run yet.

const task = new Task(() => {
  // perform task
});

const clearTask = task.schedule();

// ... some time later ...

clearTask();

// OR

task.unschedule();

You can also use the scheduleTask() util to create a task and schedule it immediately.

import { scheduleTask } from 'event-ops';

const clearTask = scheduleTask(() => {
  // perform task
});

core/Value

A Value object is useful if you would like to force-emit a State change, even if the underlying object remains the same:

import { Value, State } from 'event-ops';

const state = new State(1);

state.listen(() => {
  console.log('Value is still 1');
});

state.value = new Value(1);

core/LazyValue

Some listeners may include heavy computations, in which case you can use a LazyValue object to compute a value only once at runtime. If you reset the State value with a new LazyValue object, the cache of the underlying value would be invalidated and the computation would rerun the next time the LazyValue is consumed:

import { LazyValue, State } from 'event-ops';

const state = new State<unknown[]>([]);
const veryFrequentEvent = new Event<unknown[]>();

veryFrequentEvent.listen((value) => {
  state.value = new LazyValue(() => veryHeavyTransform(value));
});

// ... some time later ...

// veryHeavyTransform() should only be called once
console.log(state.value);

react/useListener

With useListener() you can add a listener to an Event and bind it to the lifetime of the component, meaning that once the component is unmounted, the listener will be automatically cleared.

import { Event } from 'event-ops';
import { useListener } from 'event-ops/react';

const clickEvent = new Event<[number, number]>();

function Component() {
  useListener(clickEvent, ([x, y]) => {
    console.log(`Mouse coord: ${x},${y}`);
  });

  return null;
}

// ... after render ...

clickEvent.emit([100, 150]);

For your convenience, you can use the useUpdate() hook to trigger updates once an Event change has been emitted.

import { Event } from 'event-ops';
import { useListener, useUpdate } from 'event-ops/react';

const clickEvent = new Event<[number, number]>();

function Component() {
  const update = useUpdate();
  useListener(clickEvent, update);

  return null;
}

There's also a voidEvent util that can be used as a default to an optional Event prop. This can be useful in React for conditional useListener() calls.

import { Event, voidEvent } from 'event-ops';
import { useListener, useUpdate } from 'event-ops/react';

type Props = {
  event?: Event,
};

function Component(props: Props) {
  const update = useUpdate();
  useListener(props.event ?? voidEvent, update);

  return null;
}

react/useEmitterListener

The useEmitterListener() hook is similar to useListener(), only it's based on EmitterEvent.

import { useEmitterListener } from 'event-ops/react';

const pingEvent = new EmitterEvent('ping');
const pongEvent = new EmitterEvent<number>('pong');

function Component() {
  const socket = useSocket();
  const [ts, setTs] = useState(0);

  useEmitterListener(pongEvent, socket, (ts) => {
    setTs(ts);
  });

  return (
    <div>
      timestamp: {ts}
    </div>
  );
}

react/useValue

With useValue() you can use the value of a State; the component will automatically be updated with every value change.

import { State } from 'event-ops';
import { useValue } from 'event-ops/react';

const colorState = new State('red');

function Component() {
  const color = useValue(colorState);

  return (
    <div>
      color: {color}
    </div>
  );
}

// ... after render ...

colorState.value = 'green';

License

MIT

0.0.10

5 months ago

0.0.9

8 months ago

0.0.8

8 months ago

0.0.7

8 months ago

0.0.6

8 months ago

0.0.5

8 months ago

0.0.4

8 months ago

0.0.3

8 months ago

0.0.2

8 months ago

0.0.1

8 months ago