/**
 * Copyright (c) Meta Platforms, Inc. and affiliates.
 *
 * This source code is licensed under the MIT license found in the
 * LICENSE file in the root directory of this source tree.
 *
 * @emails react-core
 */

'use strict';

let React;
let ReactDOMClient;
let act;

let idCallOrder;
const recordID = function (id) {
  idCallOrder.push(id);
};
const recordIDAndStopPropagation = function (id, event) {
  recordID(id);
  event.stopPropagation();
};
const recordIDAndReturnFalse = function (id, event) {
  recordID(id);
  return false;
};
const LISTENER = jest.fn();
const ON_CLICK_KEY = 'onClick';

let GRANDPARENT;
let PARENT;
let CHILD;
let BUTTON;

let renderTree;
let putListener;
let deleteAllListeners;

let container;

// This test is written in a bizarre way because it was previously using internals.
// It should probably be rewritten but we're keeping it for some extra coverage.
describe('ReactBrowserEventEmitter', () => {
  beforeEach(() => {
    jest.resetModules();
    LISTENER.mockClear();

    React = require('react');
    ReactDOMClient = require('react-dom/client');
    act = require('internal-test-utils').act;
    container = document.createElement('div');
    document.body.appendChild(container);

    let GRANDPARENT_PROPS = {};
    let PARENT_PROPS = {};
    let CHILD_PROPS = {};
    let BUTTON_PROPS = {};

    function Child(props) {
      return <div ref={c => (CHILD = c)} {...props} />;
    }

    class ChildWrapper extends React.PureComponent {
      render() {
        return <Child {...this.props} />;
      }
    }

    const root = ReactDOMClient.createRoot(container);

    renderTree = async function () {
      await act(() => {
        root.render(
          <div ref={c => (GRANDPARENT = c)} {...GRANDPARENT_PROPS}>
            <div ref={c => (PARENT = c)} {...PARENT_PROPS}>
              <ChildWrapper {...CHILD_PROPS} />
              <button
                disabled={true}
                ref={c => (BUTTON = c)}
                {...BUTTON_PROPS}
              />
            </div>
          </div>,
        );
      });
    };

    putListener = async function (node, eventName, listener) {
      switch (node) {
        case CHILD:
          CHILD_PROPS[eventName] = listener;
          break;
        case PARENT:
          PARENT_PROPS[eventName] = listener;
          break;
        case GRANDPARENT:
          GRANDPARENT_PROPS[eventName] = listener;
          break;
        case BUTTON:
          BUTTON_PROPS[eventName] = listener;
          break;
      }
      // Rerender with new event listeners
      await renderTree();
    };

    deleteAllListeners = async function (node) {
      switch (node) {
        case CHILD:
          CHILD_PROPS = {};
          break;
        case PARENT:
          PARENT_PROPS = {};
          break;
        case GRANDPARENT:
          GRANDPARENT_PROPS = {};
          break;
        case BUTTON:
          BUTTON_PROPS = {};
          break;
      }
      await renderTree();
    };

    idCallOrder = [];
  });

  afterEach(() => {
    document.body.removeChild(container);
    container = null;
  });

  it('should bubble simply', async () => {
    await renderTree();
    await putListener(CHILD, ON_CLICK_KEY, recordID.bind(null, CHILD));
    await putListener(PARENT, ON_CLICK_KEY, recordID.bind(null, PARENT));
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(3);
    expect(idCallOrder[0]).toBe(CHILD);
    expect(idCallOrder[1]).toBe(PARENT);
    expect(idCallOrder[2]).toBe(GRANDPARENT);
  });

  it('should bubble to the right handler after an update', async () => {
    await renderTree();
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, 'GRANDPARENT'),
    );
    await putListener(PARENT, ON_CLICK_KEY, recordID.bind(null, 'PARENT'));
    await putListener(CHILD, ON_CLICK_KEY, recordID.bind(null, 'CHILD'));
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder).toEqual(['CHILD', 'PARENT', 'GRANDPARENT']);

    idCallOrder = [];

    // Update just the grand parent without updating the child.
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, 'UPDATED_GRANDPARENT'),
    );

    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder).toEqual(['CHILD', 'PARENT', 'UPDATED_GRANDPARENT']);
  });

  it('should continue bubbling if an error is thrown', async () => {
    await renderTree();
    await putListener(CHILD, ON_CLICK_KEY, recordID.bind(null, CHILD));
    await putListener(PARENT, ON_CLICK_KEY, function (event) {
      recordID(PARENT);
      throw new Error('Handler interrupted');
    });
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    const errorHandler = jest.fn(event => {
      event.preventDefault();
    });
    window.addEventListener('error', errorHandler);
    try {
      CHILD.click();
      expect(idCallOrder.length).toBe(3);
      expect(idCallOrder[0]).toBe(CHILD);
      expect(idCallOrder[1]).toBe(PARENT);
      expect(idCallOrder[2]).toBe(GRANDPARENT);
      expect(errorHandler).toHaveBeenCalledTimes(1);
      expect(errorHandler.mock.calls[0][0]).toEqual(
        expect.objectContaining({
          error: expect.any(Error),
          message: 'Handler interrupted',
        }),
      );
    } finally {
      window.removeEventListener('error', errorHandler);
    }
  });

  it('should set currentTarget', async () => {
    await renderTree();
    await putListener(CHILD, ON_CLICK_KEY, function (event) {
      recordID(CHILD);
      expect(event.currentTarget).toBe(CHILD);
    });
    await putListener(PARENT, ON_CLICK_KEY, function (event) {
      recordID(PARENT);
      expect(event.currentTarget).toBe(PARENT);
    });
    await putListener(GRANDPARENT, ON_CLICK_KEY, function (event) {
      recordID(GRANDPARENT);
      expect(event.currentTarget).toBe(GRANDPARENT);
    });
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(3);
    expect(idCallOrder[0]).toBe(CHILD);
    expect(idCallOrder[1]).toBe(PARENT);
    expect(idCallOrder[2]).toBe(GRANDPARENT);
  });

  it('should support stopPropagation()', async () => {
    await renderTree();
    await putListener(CHILD, ON_CLICK_KEY, recordID.bind(null, CHILD));
    await putListener(
      PARENT,
      ON_CLICK_KEY,
      recordIDAndStopPropagation.bind(null, PARENT),
    );
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(2);
    expect(idCallOrder[0]).toBe(CHILD);
    expect(idCallOrder[1]).toBe(PARENT);
  });

  it('should support overriding .isPropagationStopped()', async () => {
    await renderTree();
    // Ew. See D4504876.
    await putListener(CHILD, ON_CLICK_KEY, recordID.bind(null, CHILD));
    await putListener(PARENT, ON_CLICK_KEY, function (e) {
      recordID(PARENT, e);
      // This stops React bubbling but avoids touching the native event
      e.isPropagationStopped = () => true;
    });
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(2);
    expect(idCallOrder[0]).toBe(CHILD);
    expect(idCallOrder[1]).toBe(PARENT);
  });

  it('should stop after first dispatch if stopPropagation', async () => {
    await renderTree();
    await putListener(
      CHILD,
      ON_CLICK_KEY,
      recordIDAndStopPropagation.bind(null, CHILD),
    );
    await putListener(PARENT, ON_CLICK_KEY, recordID.bind(null, PARENT));
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(1);
    expect(idCallOrder[0]).toBe(CHILD);
  });

  it('should not stopPropagation if false is returned', async () => {
    await renderTree();
    await putListener(
      CHILD,
      ON_CLICK_KEY,
      recordIDAndReturnFalse.bind(null, CHILD),
    );
    await putListener(PARENT, ON_CLICK_KEY, recordID.bind(null, PARENT));
    await putListener(
      GRANDPARENT,
      ON_CLICK_KEY,
      recordID.bind(null, GRANDPARENT),
    );
    await act(() => {
      CHILD.click();
    });
    expect(idCallOrder.length).toBe(3);
    expect(idCallOrder[0]).toBe(CHILD);
    expect(idCallOrder[1]).toBe(PARENT);
    expect(idCallOrder[2]).toBe(GRANDPARENT);
  });

  /**
   * The entire event registration state of the world should be "locked-in" at
   * the time the event occurs. This is to resolve many edge cases that come
   * about from a listener on a lower-in-DOM node causing structural changes at
   * places higher in the DOM. If this lower-in-DOM node causes new content to
   * be rendered at a place higher-in-DOM, we need to be careful not to invoke
   * these new listeners.
   */

  it('should invoke handlers that were removed while bubbling', async () => {
    await renderTree();
    const handleParentClick = jest.fn();
    const handleChildClick = async function (event) {
      await deleteAllListeners(PARENT);
    };
    await putListener(CHILD, ON_CLICK_KEY, handleChildClick);
    await putListener(PARENT, ON_CLICK_KEY, handleParentClick);
    await act(() => {
      CHILD.click();
    });
    expect(handleParentClick).toHaveBeenCalledTimes(1);
  });

  it('should not invoke newly inserted handlers while bubbling', async () => {
    await renderTree();
    const handleParentClick = jest.fn();
    const handleChildClick = async function (event) {
      await putListener(PARENT, ON_CLICK_KEY, handleParentClick);
    };
    await putListener(CHILD, ON_CLICK_KEY, handleChildClick);
    await act(() => {
      CHILD.click();
    });
    expect(handleParentClick).toHaveBeenCalledTimes(0);
  });
});