import arrayContainsObject from './arrayContainsObject';
import decode from './decode';
import encode from './encode';
import CoreManager from './CoreManager';
import type ParseObject from './ParseObject';
import type Pointer from './ParseObject';
import ParseRelation from './ParseRelation';
import unique from './unique';

export function opFromJSON(json: Record<string, any>): Op | null {
  if (!json || !json.__op) {
    return null;
  }
  switch (json.__op) {
  case 'Delete':
    return new UnsetOp();
  case 'Increment':
    return new IncrementOp(json.amount);
  case 'Add':
    return new AddOp(decode(json.objects));
  case 'AddUnique':
    return new AddUniqueOp(decode(json.objects));
  case 'Remove':
    return new RemoveOp(decode(json.objects));
  case 'AddRelation': {
    const toAdd = decode(json.objects);
    if (!Array.isArray(toAdd)) {
      return new RelationOp([], []);
    }
    return new RelationOp(toAdd, []);
  }
  case 'RemoveRelation': {
    const toRemove = decode(json.objects);
    if (!Array.isArray(toRemove)) {
      return new RelationOp([], []);
    }
    return new RelationOp([], toRemove);
  }
  case 'Batch': {
    let toAdd = [];
    let toRemove = [];
    for (let i = 0; i < json.ops.length; i++) {
      if (json.ops[i].__op === 'AddRelation') {
        toAdd = toAdd.concat(decode(json.ops[i].objects));
      } else if (json.ops[i].__op === 'RemoveRelation') {
        toRemove = toRemove.concat(decode(json.ops[i].objects));
      }
    }
    return new RelationOp(toAdd, toRemove);
  }
  }
  return null;
}

export class Op {
  // Empty parent class
  applyTo(value: any): any {} /* eslint-disable-line @typescript-eslint/no-unused-vars */
  mergeWith(previous: Op): Op | void {} /* eslint-disable-line */
  toJSON(offline?: boolean): any {} /* eslint-disable-line @typescript-eslint/no-unused-vars */
}

export class SetOp extends Op {
  _value: any;

  constructor(value: any) {
    super();
    this._value = value;
  }

  applyTo(): any {
    return this._value;
  }

  mergeWith(): SetOp {
    return new SetOp(this._value);
  }

  toJSON(offline?: boolean): any {
    return encode(this._value, false, true, undefined, offline);
  }
}

export class UnsetOp extends Op {
  applyTo() {
    return undefined;
  }

  mergeWith(): UnsetOp {
    return new UnsetOp();
  }

  toJSON(): { __op: string } {
    return { __op: 'Delete' };
  }
}

export class IncrementOp extends Op {
  _amount: number;

  constructor(amount: number) {
    super();
    if (typeof amount !== 'number') {
      throw new TypeError('Increment Op must be initialized with a numeric amount.');
    }
    this._amount = amount;
  }

  applyTo(value: any): number {
    if (typeof value === 'undefined') {
      return this._amount;
    }
    if (typeof value !== 'number') {
      throw new TypeError('Cannot increment a non-numeric value.');
    }
    return this._amount + value;
  }

  mergeWith(previous: Op): Op {
    if (!previous) {
      return this;
    }
    if (previous instanceof SetOp) {
      return new SetOp(this.applyTo(previous._value));
    }
    if (previous instanceof UnsetOp) {
      return new SetOp(this._amount);
    }
    if (previous instanceof IncrementOp) {
      return new IncrementOp(this.applyTo(previous._amount));
    }
    throw new Error('Cannot merge Increment Op with the previous Op');
  }

  toJSON(): { __op: string; amount: number } {
    return { __op: 'Increment', amount: this._amount };
  }
}

export class AddOp extends Op {
  _value: any[];

  constructor(value: any | any[]) {
    super();
    this._value = Array.isArray(value) ? value : [value];
  }

  applyTo(value: any): any[] {
    if (value == null) {
      return this._value;
    }
    if (Array.isArray(value)) {
      return value.concat(this._value);
    }
    throw new Error('Cannot add elements to a non-array value');
  }

  mergeWith(previous: Op): Op {
    if (!previous) {
      return this;
    }
    if (previous instanceof SetOp) {
      return new SetOp(this.applyTo(previous._value));
    }
    if (previous instanceof UnsetOp) {
      return new SetOp(this._value);
    }
    if (previous instanceof AddOp) {
      return new AddOp(this.applyTo(previous._value));
    }
    throw new Error('Cannot merge Add Op with the previous Op');
  }

  toJSON(): { __op: string; objects: any } {
    return { __op: 'Add', objects: encode(this._value, false, true) };
  }
}

export class AddUniqueOp extends Op {
  _value: any[];

  constructor(value: any | any[]) {
    super();
    this._value = unique(Array.isArray(value) ? value : [value]);
  }

  applyTo(value: any | any[]): any[] {
    if (value == null) {
      return this._value || [];
    }
    if (Array.isArray(value)) {
      const ParseObject = CoreManager.getParseObject();
      const toAdd: any[] = [];
      this._value.forEach(v => {
        if (v instanceof ParseObject) {
          if (!arrayContainsObject(value, v)) {
            toAdd.push(v);
          }
        } else {
          if (value.indexOf(v) < 0) {
            toAdd.push(v);
          }
        }
      });
      return value.concat(toAdd);
    }
    throw new Error('Cannot add elements to a non-array value');
  }

  mergeWith(previous: Op): Op {
    if (!previous) {
      return this;
    }
    if (previous instanceof SetOp) {
      return new SetOp(this.applyTo(previous._value));
    }
    if (previous instanceof UnsetOp) {
      return new SetOp(this._value);
    }
    if (previous instanceof AddUniqueOp) {
      return new AddUniqueOp(this.applyTo(previous._value));
    }
    throw new Error('Cannot merge AddUnique Op with the previous Op');
  }

  toJSON(): { __op: string; objects: any } {
    return { __op: 'AddUnique', objects: encode(this._value, false, true) };
  }
}

export class RemoveOp extends Op {
  _value: any[];

  constructor(value: any | any[]) {
    super();
    this._value = unique(Array.isArray(value) ? value : [value]);
  }

  applyTo(value: any | any[]): any[] {
    if (value == null) {
      return [];
    }
    if (Array.isArray(value)) {
      const ParseObject = CoreManager.getParseObject();
      // var i = value.indexOf(this._value);
      const removed = value.concat([]);
      for (let i = 0; i < this._value.length; i++) {
        let index = removed.indexOf(this._value[i]);
        while (index > -1) {
          removed.splice(index, 1);
          index = removed.indexOf(this._value[i]);
        }
        if (this._value[i] instanceof ParseObject && this._value[i].id) {
          for (let j = 0; j < removed.length; j++) {
            if (removed[j] instanceof ParseObject && this._value[i].id === removed[j].id) {
              removed.splice(j, 1);
              j--;
            }
          }
        }
      }
      return removed;
    }
    throw new Error('Cannot remove elements from a non-array value');
  }

  mergeWith(previous: Op): Op {
    if (!previous) {
      return this;
    }
    if (previous instanceof SetOp) {
      return new SetOp(this.applyTo(previous._value));
    }
    if (previous instanceof UnsetOp) {
      return new UnsetOp();
    }
    if (previous instanceof RemoveOp) {
      const ParseObject = CoreManager.getParseObject();
      const uniques = previous._value.concat([]);
      for (let i = 0; i < this._value.length; i++) {
        if (this._value[i] instanceof ParseObject) {
          if (!arrayContainsObject(uniques, this._value[i])) {
            uniques.push(this._value[i]);
          }
        } else {
          if (uniques.indexOf(this._value[i]) < 0) {
            uniques.push(this._value[i]);
          }
        }
      }
      return new RemoveOp(uniques);
    }
    throw new Error('Cannot merge Remove Op with the previous Op');
  }

  toJSON(): { __op: string; objects: any } {
    return { __op: 'Remove', objects: encode(this._value, false, true) };
  }
}

export class RelationOp extends Op {
  _targetClassName: string | null;
  relationsToAdd: string[];
  relationsToRemove: string[];

  constructor(adds: (ParseObject | string)[], removes: (ParseObject | string)[]) {
    super();
    this._targetClassName = null;

    if (Array.isArray(adds)) {
      this.relationsToAdd = unique(adds.map(this._extractId, this));
    }

    if (Array.isArray(removes)) {
      this.relationsToRemove = unique(removes.map(this._extractId, this));
    }
  }

  _extractId(obj: string | ParseObject): string {
    if (typeof obj === 'string') {
      return obj;
    }
    if (!obj.id) {
      throw new Error('You cannot add or remove an unsaved Parse Object from a relation');
    }
    if (!this._targetClassName) {
      this._targetClassName = obj.className;
    }
    if (this._targetClassName !== obj.className) {
      throw new Error(
        'Tried to create a Relation with 2 different object types: ' +
          this._targetClassName +
          ' and ' +
          obj.className +
          '.'
      );
    }
    return obj.id;
  }

  applyTo(value: any, parent?: ParseObject, key?: string): ParseRelation {
    if (!value) {
      if (!parent || !key) {
        throw new Error(
          'Cannot apply a RelationOp without either a previous value, or an object and a key'
        );
      }
      const relation = new ParseRelation(parent, key);
      relation.targetClassName = this._targetClassName;
      return relation;
    }
    if (value instanceof ParseRelation) {
      if (this._targetClassName) {
        if (value.targetClassName) {
          if (this._targetClassName !== value.targetClassName) {
            throw new Error(
              'Related object must be a ' +
                value.targetClassName +
                ', but a ' +
                this._targetClassName +
                ' was passed in.'
            );
          }
        } else {
          value.targetClassName = this._targetClassName;
        }
      }
      return value;
    } else {
      throw new Error('Relation cannot be applied to a non-relation field');
    }
  }

  mergeWith(previous: Op): Op {
    if (!previous) {
      return this;
    } else if (previous instanceof UnsetOp) {
      throw new Error('You cannot modify a relation after deleting it.');
    } else if (previous instanceof SetOp && previous._value instanceof ParseRelation) {
      return this;
    } else if (previous instanceof RelationOp) {
      if (previous._targetClassName && previous._targetClassName !== this._targetClassName) {
        throw new Error(
          'Related object must be of class ' +
            previous._targetClassName +
            ', but ' +
            (this._targetClassName || 'null') +
            ' was passed in.'
        );
      }
      const newAdd = previous.relationsToAdd.concat([]);
      this.relationsToRemove.forEach(r => {
        const index = newAdd.indexOf(r);
        if (index > -1) {
          newAdd.splice(index, 1);
        }
      });
      this.relationsToAdd.forEach(r => {
        const index = newAdd.indexOf(r);
        if (index < 0) {
          newAdd.push(r);
        }
      });

      const newRemove = previous.relationsToRemove.concat([]);
      this.relationsToAdd.forEach(r => {
        const index = newRemove.indexOf(r);
        if (index > -1) {
          newRemove.splice(index, 1);
        }
      });
      this.relationsToRemove.forEach(r => {
        const index = newRemove.indexOf(r);
        if (index < 0) {
          newRemove.push(r);
        }
      });

      const newRelation = new RelationOp(newAdd, newRemove);
      newRelation._targetClassName = this._targetClassName;
      return newRelation;
    }
    throw new Error('Cannot merge Relation Op with the previous Op');
  }

  toJSON(): { __op?: string; objects?: any; ops?: any } {
    const idToPointer = (id: string) => {
      return {
        __type: 'Pointer',
        className: this._targetClassName,
        objectId: id,
      };
    };

    let pointers: any = null;
    let adds: null | { __op: string; objects: null | Pointer[] } = null;
    let removes: null | { __op: string; objects: null | Pointer[] } = null;

    if (this.relationsToAdd.length > 0) {
      pointers = this.relationsToAdd.map(idToPointer);
      adds = { __op: 'AddRelation', objects: pointers };
    }
    if (this.relationsToRemove.length > 0) {
      pointers = this.relationsToRemove.map(idToPointer);
      removes = { __op: 'RemoveRelation', objects: pointers };
    }

    if (adds && removes) {
      return { __op: 'Batch', ops: [adds, removes] };
    }

    return adds || removes || {};
  }
}
CoreManager.setParseOp({
  Op,
  opFromJSON,
  SetOp,
  UnsetOp,
  IncrementOp,
  AddOp,
  RelationOp,
  RemoveOp,
  AddUniqueOp,
});