import { ingestInCache } from '~/cache/api/ingestInCache';
import { getActiveClient } from '~/client/api/activeClient';
import { createEventSubscriber } from '~/core/events';

import { prepareChannelPayload } from '../utils';

type CallbackFn = (
  channel: Amity.StaticInternalChannel,
  member: Amity.Membership<'channel'>,
) => void;
const callbacks: CallbackFn[] = [];
let mainDisposer: (() => void) | null = null;

const dispose = (cb: CallbackFn) => {
  const index = callbacks.indexOf(cb);
  if (index > -1) {
    callbacks.splice(index, 1);
  }
  if (callbacks.length === 0) {
    mainDisposer?.();
  }
};

export const onChannelMemberAdded = (
  callback: (channel: Amity.StaticInternalChannel, member: Amity.Membership<'channel'>) => void,
): Amity.Unsubscriber => {
  if (callbacks.length === 0) {
    const client = getActiveClient();

    const filter = async (payload: Amity.ChannelMembershipPayload) => {
      const preparedPayload = await prepareChannelPayload(payload);
      const { channels, channelUsers } = preparedPayload;

      ingestInCache(preparedPayload);
      callbacks.forEach(cb =>
        cb(channels[0], channelUsers.find(cu => cu.membership === 'member')!),
      );
    };

    mainDisposer = createEventSubscriber(
      client,
      'onChannelMemberAdded',
      'channel.membersAdded',
      filter,
    );
  }

  callbacks.push(callback);
  return () => dispose(callback);
};
