All Downloads are FREE. Search and download functionalities are using the official Maven repository.

lucuma.ui.components.UserSelectionForm.scala Maven / Gradle / Ivy

// Copyright (c) 2016-2023 Association of Universities for Research in Astronomy, Inc. (AURA)
// For license information see LICENSE or https://opensource.org/licenses/BSD-3-Clause

package lucuma.ui.components

import cats.syntax.all.*
import crystal.react.*
import crystal.react.hooks.*
import eu.timepit.refined.types.string.NonEmptyString
import japgolly.scalajs.react.*
import japgolly.scalajs.react.util.DefaultEffects.Async as DefaultA
import japgolly.scalajs.react.vdom.html_<^.*
import lucuma.core.util.NewType
import lucuma.react.common.*
import lucuma.react.primereact.Button
import lucuma.react.primereact.Dialog
import lucuma.react.primereact.Image
import lucuma.react.primereact.Message
import lucuma.ui.Resources
import lucuma.ui.primereact.*
import lucuma.ui.sso.SSOClient
import lucuma.ui.sso.UserVault
import lucuma.ui.syntax.all.*
import lucuma.ui.utils.UAParser
import org.scalajs.dom
import org.typelevel.log4cats.Logger

case class UserSelectionForm(
  systemName:      NonEmptyString,
  systemNameStyle: Css,
  ssoClient:       SSOClient[DefaultA],
  vault:           View[Option[UserVault]],
  message:         View[Option[NonEmptyString]],
  allowGuest:      Boolean
)(using val logger: Logger[DefaultA])
    extends ReactFnProps(UserSelectionForm.component)

object UserSelectionForm:
  private type Props = UserSelectionForm

  private object IsOpen extends NewType[Boolean]

  private case class BrowserInfo(supportedOrcidBrowser: Boolean, warnBrowser: Boolean):
    inline def showButtons: Boolean = supportedOrcidBrowser

  private object BrowserInfo:
    def supportedOrcidBrowser: DefaultA[BrowserInfo] =
      DefaultA.handleError(
        DefaultA.delay:
          val browser  = UAParser(dom.window.navigator.userAgent).getBrowser()
          val verRegex = raw"(\d{0,3}).(\d{0,3})\.?(.*)?".r

          (browser.name, browser.version) match {
            case ("Safari", verRegex(major, _, _)) if major.toInt <= 13 =>
              BrowserInfo(false, false)
            case ("Safari", _)                                          => BrowserInfo(true, true)
            case _                                                      => BrowserInfo(true, false)
          }
      )(_ => DefaultA.delay(BrowserInfo(true, true)))

  private val component =
    ScalaFnComponent
      .withHooks[Props]
      .useState(IsOpen(true))
      .useEffectResultOnMount(BrowserInfo.supportedOrcidBrowser)
      .render: (props, isOpen, browserInfoPot) =>
        import props.given

        val guest: Callback =
          props.ssoClient.guest.flatMap(v => props.vault.set(v.some).toAsync).runAsync

        val login: Callback = props.ssoClient.redirectToLogin.runAsync

        Dialog(
          closable = false,
          visible = isOpen.value.value,
          onHide = isOpen.setState(IsOpen(false)),
          resizable = false,
          draggable = false,
          showHeader = false,
          clazz = LucumaPrimeStyles.Dialog.Small
        )(
          browserInfoPot.renderPot(browserInfo =>
            React.Fragment(
              <.div(
                LoginStyles.LoginBoxLayout,
                Logo(props.systemName, props.systemNameStyle),
                <.div(
                  LoginStyles.UserSelectionButtons,
                  Button(
                    label = "Login with ORCID",
                    icon = Image(src = Resources.OrcidLogo, clazz = LoginStyles.LoginOrcidIcon),
                    clazz = LoginStyles.LoginBoxButton,
                    severity = Button.Severity.Secondary,
                    onClick = login >> props.message.set(none) >> isOpen.setState(IsOpen(false))
                  ).big.when(browserInfo.showButtons),
                  Button(
                    label = "Continue as Guest",
                    icon = LoginIcons.UserAstronaut.withClass(LoginStyles.LoginOrcidIcon),
                    clazz = LoginStyles.LoginBoxButton,
                    severity = Button.Severity.Secondary,
                    onClick = guest >> props.message.set(none) >> isOpen.setState(IsOpen(false))
                  ).big.when(browserInfo.showButtons && props.allowGuest)
                )
              ),
              <.div(LoginStyles.LoginMessagesLayout)(
                props.message.get.whenDefined(message =>
                  Message(text = message.value, severity = Message.Severity.Error)
                ),
                Message(
                  text =
                    "This version of Safari isn't supported. Try a newer version (≥14.0.1) or a recent version of Chrome or Firefox.",
                  severity = Message.Severity.Error,
                  icon = LoginIcons.SkullCrossBones
                ).unless(browserInfo.supportedOrcidBrowser),
                Message(
                  text =
                    "ORCID authentication does not work with some configurations of Safari and MacOS. If it doesn't work for you please try Chrome or Firefox.",
                  severity = Message.Severity.Warning,
                  icon = LoginIcons.ExclamationTriangle
                ).when(browserInfo.warnBrowser)
              )
            )
          )
        )




© 2015 - 2025 Weber Informatics LLC | Privacy Policy