blob: 5ffc8b36c04aadb8291013c1e608afea115a41c6 [file] [log] [blame]
/**
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
import undoable from 'redux-undo';
import { UNDO_LIMIT } from '../util/constants';
import {
UPDATE_COMPONENTS,
DELETE_COMPONENT,
CREATE_COMPONENT,
CREATE_TOP_LEVEL_TABS,
DELETE_TOP_LEVEL_TABS,
RESIZE_COMPONENT,
MOVE_COMPONENT,
HANDLE_COMPONENT_DROP,
} from '../actions/dashboardLayout';
import { HYDRATE_DASHBOARD } from '../actions/hydrate';
import dashboardLayout from './dashboardLayout';
// List of actions that should trigger undo history
const TRACKED_ACTIONS = [
HYDRATE_DASHBOARD,
UPDATE_COMPONENTS,
DELETE_COMPONENT,
CREATE_COMPONENT,
CREATE_TOP_LEVEL_TABS,
DELETE_TOP_LEVEL_TABS,
RESIZE_COMPONENT,
MOVE_COMPONENT,
HANDLE_COMPONENT_DROP,
];
/*
* WORKAROUND FOR REDUX-UNDO FILTER BUG
*
* PROBLEM:
* Redux-undo's filter functionality is broken in multiple versions. Users report that
* actions are either incorrectly filtered out or incorrectly included in undo history,
* making undo/redo buttons not work when they should.
*
* AFFECTED VERSIONS:
* - Beta versions (like 1.0.0-beta9-9-7 currently used by Superset)
* - Stable versions up to 1.1.0
* Both includeAction/excludeAction helpers AND custom filter functions fail.
*
* WHY FILTERING MATTERS:
* Without filtering, ALL Redux actions (API calls, toasts, theme changes, etc.) would
* create undo history entries, making undo/redo confusing and inefficient. We only want
* dashboard layout changes to be undoable.
*
* SOLUTION:
* Instead of using redux-undo's broken filter system, we filter at the reducer level
* by wrapping the base dashboardLayout reducer. This approach:
* 1. Prevents non-layout actions from reaching the reducer (so no state changes)
* 2. Allows redux-undo to work without any filtering (which works fine)
* 3. Results in only layout actions being tracked in undo history
* 4. Provides the same filtering result without hitting the redux-undo bug
*
* WHEN TO REMOVE:
* This can be reverted when redux-undo fixes their filter functionality by:
* 1. Removing the layoutOnlyReducer wrapper
* 2. Using the base dashboardLayout reducer directly
* 3. Adding back: filter: includeAction(TRACKED_ACTIONS)
*
* BUG REPORT:
* - https://github.com/omnidan/redux-undo/issues/306
*/
// Wrapper reducer that filters actions before they reach the dashboardLayout reducer
const layoutOnlyReducer = (state, action) => {
// Only allow layout-related actions to reach the dashboardLayout reducer
if (!TRACKED_ACTIONS.includes(action.type)) return state;
// For layout actions, proceed with normal dashboardLayout reduction
return dashboardLayout(state, action);
};
const undoableReducer = undoable(layoutOnlyReducer, {
// +1 because length of history seems max out at limit - 1
// +1 again so we can detect if we've exceeded the limit
limit: UNDO_LIMIT + 2,
ignoreInitialState: true,
});
export default undoableReducer;