/*
 * Copyright (c) All respective contributors to the Peridot Project. All rights reserved.
 * Copyright (c) 2021-2022 Rocky Enterprise Software Foundation, Inc. All rights reserved.
 * Copyright (c) 2021-2022 Ctrl IQ, Inc. All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice,
 * this list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * 3. Neither the name of the copyright holder nor the names of its contributors
 * may be used to endorse or promote products derived from this software without
 * specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 */

import Alert from '@mui/material/Alert';
import { H1 } from 'dotui/H1';
import { H2 } from 'dotui/H2';
import { H4 } from 'dotui/H4';
import { H5 } from 'dotui/H5';
import { PageWrapper } from 'dotui/PageWrapper';
import React from 'react';
import { reqap } from './reqap';
import { RemoteState, AccessDeniedError } from './types';

export var loadingElement: React.ReactElement = <div>Loading...</div>;

export const setLoadingElement = (elem: React.ReactElement) => {
  loadingElement = elem;
};

export function fetchRemoteResource<T, X>(
  fetchFunction: () => Promise<T>,
  setFunction: (val: RemoteState<T>) => void,
  disableEffect: boolean = false,
  effectArray: any[] = []
) {
  const remoteMethod = async () => {
    const [err, res] = await reqap(() => fetchFunction());
    if (err) {
      if (err.status === 403) {
        setFunction('access_denied');
      } else {
        setFunction(null);
      }
    }

    if (res) {
      setFunction(res);
    }
  };

  if (disableEffect) {
    remoteMethod().then();
  } else {
    React.useEffect(() => {
      remoteMethod().then();
    }, effectArray);
  }
}

export function suspenseRemoteResource<T>(
  state: RemoteState<T>,
  render: (res: T) => React.ReactNode,
  wrap: ((elem: React.ReactNode) => React.ReactNode) | undefined = undefined
): React.ReactNode {
  const wrapHelper = (elem: React.ReactNode) => {
    if (wrap) {
      return wrap(elem);
    }

    return elem;
  };

  if (state === undefined) {
    return wrapHelper(loadingElement);
  } else if (state === 'access_denied') {
    return wrapHelper(
      <Alert severity="error">
        Access to resource has been denied. Contact an administrator if you
        think this is a mistake
      </Alert>
    );
  } else if (state === null) {
    return wrapHelper(
      <div className="flex items-center justify-center flex-col mt-4">
        <H2>Could not fetch remote resource</H2>
        <H4>Please try again later</H4>
      </div>
    );
  } else {
    return render(state);
  }
}