import { observable, action, makeObservable, runInAction } from "mobx";
// types
import { RootStore } from "../root";
// services
import { InboxService } from "services/inbox.service";
// types
import { IInboxIssue, IIssue, TInboxStatus } from "types";
// constants
import { INBOX_ISSUE_SOURCE } from "constants/inbox";

export interface IInboxIssueDetailsStore {
  // states
  loader: boolean;
  error: any | null;

  // observables
  issueDetails: {
    [issueId: string]: IInboxIssue;
  };

  // actions
  fetchIssueDetails: (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    issueId: string
  ) => Promise<IInboxIssue>;
  createIssue: (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    data: Partial<IIssue>
  ) => Promise<IInboxIssue>;
  updateIssue: (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    issueId: string,
    data: Partial<IInboxIssue>
  ) => Promise<void>;
  updateIssueStatus: (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    issueId: string,
    data: TInboxStatus
  ) => Promise<void>;
  deleteIssue: (workspaceSlug: string, projectId: string, inboxId: string, issueId: string) => Promise<void>;
}

export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
  // states
  loader: boolean = false;
  error: any | null = null;

  // observables
  issueDetails: { [issueId: string]: IInboxIssue } = {};

  // root store
  rootStore;

  // services
  inboxService;

  constructor(_rootStore: RootStore) {
    makeObservable(this, {
      // states
      loader: observable.ref,
      error: observable.ref,

      // observables
      issueDetails: observable.ref,

      // actions
      fetchIssueDetails: action,
      createIssue: action,
      updateIssueStatus: action,
      deleteIssue: action,
    });

    this.rootStore = _rootStore;
    this.inboxService = new InboxService();
  }

  fetchIssueDetails = async (workspaceSlug: string, projectId: string, inboxId: string, issueId: string) => {
    try {
      runInAction(() => {
        this.loader = true;
      });

      const issueResponse = await this.inboxService.getInboxIssueById(workspaceSlug, projectId, inboxId, issueId);

      runInAction(() => {
        this.loader = false;
        this.issueDetails = {
          ...this.issueDetails,
          [issueId]: issueResponse,
        };
      });

      return issueResponse;
    } catch (error) {
      runInAction(() => {
        this.loader = false;
        this.error = error;
      });

      throw error;
    }
  };

  createIssue = async (workspaceSlug: string, projectId: string, inboxId: string, data: Partial<IIssue>) => {
    const payload = {
      issue: {
        name: data.name,
        description: data.description,
        description_html: data.description_html,
        priority: data.priority,
      },
      source: INBOX_ISSUE_SOURCE,
    };

    try {
      const response = await this.inboxService.createInboxIssue(workspaceSlug, projectId, inboxId, payload);

      runInAction(() => {
        this.issueDetails = {
          ...this.issueDetails,
          [response.id]: response,
        };
        this.rootStore.inboxIssues.inboxIssues = {
          ...this.rootStore.inboxIssues.inboxIssues,
          [inboxId]: [response, ...this.rootStore.inboxIssues.inboxIssues[inboxId]],
        };
      });

      return response;
    } catch (error) {
      runInAction(() => {
        this.error = error;
      });

      throw error;
    }
  };

  updateIssue = async (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    issueId: string,
    data: Partial<IInboxIssue>
  ) => {
    const updatedIssue = { ...this.issueDetails[issueId], ...data };

    try {
      runInAction(() => {
        this.issueDetails = {
          ...this.issueDetails,
          [issueId]: updatedIssue,
        };
        this.rootStore.inboxIssues.inboxIssues = {
          ...this.rootStore.inboxIssues.inboxIssues,
          [inboxId]: this.rootStore.inboxIssues.inboxIssues[inboxId].map((issue) => {
            if (issue.issue_inbox[0].id === issueId) return updatedIssue;

            return issue;
          }),
        };
      });

      await this.inboxService.patchInboxIssue(workspaceSlug, projectId, inboxId, issueId, { issue: data });
    } catch (error) {
      runInAction(() => {
        this.error = error;
      });

      this.rootStore.inboxIssues.fetchInboxIssues(workspaceSlug, projectId, inboxId);
      this.fetchIssueDetails(workspaceSlug, projectId, inboxId, issueId);

      throw error;
    }
  };

  updateIssueStatus = async (
    workspaceSlug: string,
    projectId: string,
    inboxId: string,
    issueId: string,
    data: TInboxStatus
  ) => {
    const updatedIssue = { ...this.issueDetails[issueId] };
    updatedIssue.issue_inbox[0] = {
      ...updatedIssue.issue_inbox[0],
      ...data,
    };

    try {
      runInAction(() => {
        this.issueDetails = {
          ...this.issueDetails,
          [issueId]: updatedIssue,
        };
        this.rootStore.inboxIssues.inboxIssues = {
          ...this.rootStore.inboxIssues.inboxIssues,
          [inboxId]: this.rootStore.inboxIssues.inboxIssues[inboxId].map((issue) => {
            if (issue.issue_inbox[0].id === issueId) return updatedIssue as IInboxIssue;

            return issue;
          }),
        };
      });

      await this.inboxService.markInboxStatus(workspaceSlug, projectId, inboxId, issueId, data);
    } catch (error) {
      runInAction(() => {
        this.error = error;
      });

      this.rootStore.inboxIssues.fetchInboxIssues(workspaceSlug, projectId, inboxId);
      this.fetchIssueDetails(workspaceSlug, projectId, inboxId, issueId);

      throw error;
    }
  };

  deleteIssue = async (workspaceSlug: string, projectId: string, inboxId: string, issueId: string) => {
    const updatedIssues = { ...this.issueDetails };
    delete updatedIssues[issueId];

    try {
      runInAction(() => {
        this.issueDetails = updatedIssues;
        this.rootStore.inboxIssues.inboxIssues = {
          ...this.rootStore.inboxIssues.inboxIssues,
          [inboxId]: this.rootStore.inboxIssues.inboxIssues[inboxId]?.filter(
            (issue) => issue.issue_inbox[0].id !== issueId
          ),
        };
      });

      await this.inboxService.deleteInboxIssue(workspaceSlug, projectId, inboxId, issueId);
    } catch (error) {
      runInAction(() => {
        this.error = error;
      });

      this.rootStore.inboxIssues.fetchInboxIssues(workspaceSlug, projectId, inboxId);
      this.fetchIssueDetails(workspaceSlug, projectId, inboxId, issueId);

      throw error;
    }
  };
}