Skip to content

aneldev/dynadux-history-middleware

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

19 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Dynadux Logo

dynadux history middleware

Travel in time with dynadux.

This middleware offers the Undo/Redo feature.

It offers a history limit so it won't explode your memory.

Supports restore points so you can travel in time with named restore points.

Create a store

Let's create a store with dynadux.

Our store exposes the history with a few methods where dispatching events to the dynaduxHistoryMiddleware

If you are unfamiliar with dynadux learn it here.

const actions = {
  CLEAR: 'CLEAR',
  ADD_TODO: 'ADD_TODO',
  REMOVE_TODO: 'REMOVE_TODO',
};

const createTodoAppStore = () => {
  const store = createStore<ITodoAppState>({
    initialState: {
      todos: [],
    },
    middlewares: [
      dynaduxHistoryMiddleware(),
    ],
    reducers: {
      [actions.CLEAR]: () => {
        return {
          todos: [],
        };
      },
      [actions.ADD_TODO]: ({state: {todos}, payload}) => {
        return {
          todos: todos.concat(payload),
        };
      },
      [actions.REMOVE_TODO]: ({state: {todos}, payload: todoId}) => {
        return {
          todos: todos.filter(todo => todo.id !== todoId),
        };
      },
    },
  });

  return {
    get state() { return store.state; },
    clear: () => store.dispatch(actions.CLEAR),
    addTodo: (todo: ITodo) => store.dispatch(actions.ADD_TODO, todo),
    removeTodo: (todoId: string) => store.dispatch(actions.REMOVE_TODO, todoId),
    history: {
      prev: () => store.dispatch(EDynaduxHistoryMiddlewareActions.PREV),
      next: () => store.dispatch(EDynaduxHistoryMiddlewareActions.NEXT),
      setRestorePoint: (name: string) => store.dispatch(EDynaduxHistoryMiddlewareActions.SET_RESTORE_POINT, {name}),
      activateRestorePoint: (name: string) => store.dispatch(EDynaduxHistoryMiddlewareActions.ACTIVATE_RESTORE_POINT, {name}),
    },
  };
};

// create the store
const appStore = createTodoAppStore();

To get the todos at any time you evaluate appStore.state.todos.

History limit

The middleware accepts an object as an argument with the historySize property.

This property indicates how many history items will be available to travel in the past.

    middlewares: [
      dynaduxHistoryMiddleware({historySize: 100}),
    ],

If you don't define it, history is unlimited.

Travel in time

You can travel in time dispatching actions EDynaduxHistoryMiddlewareActions.PREV/NEXT.

Our store above is smart and we can use the appStore.history.prev/next() accordingly.

// Clear the store
appStore.clear();


// Add a couple of todos
appStore.addTodo({id: '301', label: 'Before work beers', done: false});
appStore.addTodo({id: '302', label: 'After work beers', done: false});
appStore.addTodo({id: '303', label: 'Evening beers', done: false});
// The todos now are: 301, 302, 303

// Let's go back in time
appStore.history.prev();
// The todos now are: 301, 302

// Let's go forward
appStore.history.next();
// The todos now are: 301, 302, 303

// Try to go forward
appStore.history.next();
// The todos still are: 301, 302, 303

Restore points

At any time you can set a restore point. To do that you call appStore.history.setRestorePoint('nice').

Nothing is changed in the state.

Later you can call the setRestorePoint with the same name and this will override the previous state.

After a time, when you want to go back, you can call appStore.history.activateRestorePoint('nice').

Restore points feature is extremely useful for apps with progress steps, editor apps, game apps, etc.

Restore points are introduced for the first time by this dynadux middleware. This was possible with the flexibility that dynadux offers.

Restore points feature doesn't duplicate data but it uses indexes. So adding restore points don't increase the used memory.


// Clear the store
appStore.clear();

// Add a couple of todos
appStore.addTodo({id: '301', label: 'Before work beers', done: false});
appStore.addTodo({id: '302', label: 'After work beers', done: false});
// The todos now are: 301, 302

// Create a restore point with name "basics"
appStore.history.setRestorePoint('basics');

// Add a todo
appStore.addTodo({id: '303', label: 'Evening beers', done: false});
// Now the todos are: 301, 302, 303

// Create a restore point with the name evening
appStore.history.setRestorePoint('evening');

// Activate the restore point basics
appStore.history.activateRestorePoint('basics');
// The todos now are: 301, 302, as it was at the set store point previously

// Activate the restore point evening
appStore.history.activateRestorePoint('evening');
// Now the todos are: 301, 302, 303


// Let's go back in time, make a change and try to activate a future restore point

// Go back in time to basics
appStore.history.activateRestorePoint('basics');
// Add a new todo
appStore.addTodo({id: '304', label: 'Sleep', done: false});    // This add deleted the future evening tag
// The todos now are 301, 302, 304
// Try to go to evening
appStore.history.activateRestorePoint('evening');
// The todos still are 301, 302, 304
// A console.error is occured
    

API

Middleware support the below actions

EDynaduxHistoryMiddlewareActions.PREV

Payload: none

It goes on dispatch back in time,

EDynaduxHistoryMiddlewareActions.NEXT

Payload: none

It goes on dispatch forward in time,

EDynaduxHistoryMiddlewareActions.SET_RESTORE_POINT

Payload: {name: string}

It creates a restore point with a name.

If the name is already used it overrides the last one.

EDynaduxHistoryMiddlewareActions.ACTIVATE_RESTORE_POINT

Payload: {name: string}

Activates a restore point.

If the restore point doesn't exist a console.error will be occured but not an exception.

EDynaduxHistoryMiddlewareActions.GET_HISTORY

Payload: {stateTargetPropertyName: string}

This will create a property in the state of the store, with the name, the value of the payload's stateTargetPropertyName property.

In the state it will save the collected history items.

About

Middleware for dynadux stores to support trips in time

Resources

License

Stars

Watchers

Forks

Packages

No packages published