Source: misc_controls_arcball.js

/**
 * @file
 *
 * Summary.
 *
 * <p>WebGL - {@link THREE.ArcballControls arcball controls} demo.</p>
 *
 * <p><b>For educational purposes only.</b></p>
 *
 * @since 17/09/2024
 * @license Licensed under the {@link https://www.opensource.org/licenses/mit-license.php MIT license}.
 * @copyright © 2025 Paulo R Cavalcanti.
 * @author {@link https://x.com/_artisaverb?lang=en Andrew Maximov}
 * @author modified by {@link https://krotalias.github.io Paulo Roma}
 *
 * @see <a href="/cwdc/13-webgl/examples/three/content/misc_controls_arcball.html?gui=1">link</a>
 * @see <a href="/cwdc/13-webgl/examples/three/content/misc_controls_arcball.js">source</a>
 * @see {@link https://threejs.org/examples/misc_controls_arcball.html link threejs}
 * @see {@link https://github.com/mrdoob/three.js/blob/master/examples/misc_controls_arcball.html#L184 source threejs}
 * @see {@link https://finalfantasy.fandom.com/wiki/Cerberus_(weapon) Cerberus} (weapon)
 * @see {@link https://en.wikipedia.org/wiki/Cerberus Cerberus} (underworld guardian)
 * @see {@link https://discourse.threejs.org/t/arcballcontrol-problems-on-mobile/70160/3 ArcballControls problems on mobile}
 * @see <img src="/cwdc/13-webgl/examples/three/content/Cerberus.jpg" alt="ArcballControls" width="256">
 */

"use strict";

import * as THREE from "https://unpkg.com/three@latest/build/three.module.js?module";
import { ArcballControls } from "https://unpkg.com/three@latest/examples/jsm/controls/ArcballControls.js?module";
import { OBJLoader } from "https://unpkg.com/three@latest/examples/jsm/loaders/OBJLoader.js?module";
import { RGBELoader } from "https://unpkg.com/three@latest/examples/jsm/loaders/RGBELoader.js?module";
import { GUI } from "https://unpkg.com/three@latest/examples/jsm/libs/lil-gui.module.min.js?module";

/**
 * Array of camera types.
 * @type {String[]}
 */
const cameras = ["Orthographic", "Perspective"];

/**
 * Object with the current camera type.
 * @type {Object<{type: String}>}
 */
const cameraType = { type: "Perspective" };

/**
 * Perspective camera distance.
 * @type {Number}
 */
const perspectiveDistance = 5;

/**
 * Orthographic camera distance.
 * @type {Number}
 */
const orthographicDistance = 120;

let camera, controls, scene, renderer;
let folderOptions, folderAnimations;

/**
 * Three.js module.
 * @external three
 * @author Ricardo Cabello ({@link https://coopermrdoob.weebly.com/ Mr.doob})
 * @since 24/04/2010
 * @license Licensed under the {@link https://www.opensource.org/licenses/mit-license.php MIT license}
 * @see {@link https://threejs.org/docs/#manual/en/introduction/Installation Installation}
 * @see {@link https://discoverthreejs.com DISCOVER three.js}
 * @see {@link https://riptutorial.com/ebook/three-js Learning three.js}
 * @see {@link https://github.com/mrdoob/three.js github}
 * @see {@link http://cindyhwang.github.io/interactive-design/Mrdoob/index.html An interview with Mr.doob}
 * @see {@link https://experiments.withgoogle.com/search?q=Mr.doob Experiments with Google}
 */

/**
 * <p>Main three.js namespace.</p>
 * <a href="/cwdc/13-webgl/examples/three/content/doc-example/index.html">Imported</a> from {@link external:three three.module.js}
 *
 * @namespace THREE
 */

/**
 * <p>lil-gui module.</p>
 *
 * <p>lil-gui gives you an interface for changing the properties of any JavaScript object at runtime.
 * It's intended as a drop-in replacement for dat.gui,
 * implemented with more modern web standards and some new quality of life features.</p>
 * @external GUI
 * @see {@link https://lil-gui.georgealways.com lil-gui}
 */

/**
 * Global {@link GUI.GUI GUI}.
 * @type {GUI}
 */
let gui;

/**
 * <p>ArcballGui is a class that contains the properties and methods for the user interface.</p>
 * @class ArcballGui
 * @see {@link https://developer.mozilla.org/en-US/docs/Web/JavaScript/Reference/Global_Objects/Object/Object MDN Object}
 */
const ArcballGui = {
  /**
   * Show gizmo if set to true
   * @type {Boolean}
   */
  gizmoVisible: true,

  /**
   * Enable GUI if set to true.
   * @type {Boolean}
   */
  useGUI: false,

  /**
   * Create an Arcballcontrols object and activate its gizmo.
   */
  setArcballControls: function () {
    /**
     * Arcball controls allow the camera to be controlled by a virtual trackball
     * with full touch support and advanced navigation functionality.
     * Cursor/finger positions and movements are mapped over a virtual trackball surface
     * represented by a gizmo and mapped in intuitive and consistent camera movements.
     * Dragging cursor/fingers will cause camera to orbit around the center of the trackball
     * in a conservative way (returning to the starting point will make the camera
     * to return to its starting orientation).
     * @class ArcballControls
     * @memberof THREE
     * @see {@link https://threejs.org/docs/#examples/en/controls/ArcballControls ArcballControls}
     */
    controls = new ArcballControls(camera, renderer.domElement, scene);
    controls.addEventListener("change", render);
    this.gizmoVisible = true;
    if (ArcballGui.useGUI) this.populateGui();
  },

  /**
   * Populate GUI
   */
  populateGui: function () {
    folderOptions.add(controls, "enabled").name("Enable controls");
    folderOptions.add(controls, "enableGrid").name("Enable Grid");
    folderOptions.add(controls, "enableRotate").name("Enable rotate");
    folderOptions.add(controls, "enablePan").name("Enable pan");
    folderOptions.add(controls, "enableZoom").name("Enable zoom");
    folderOptions.add(controls, "cursorZoom").name("Cursor zoom");
    folderOptions.add(controls, "adjustNearFar").name("adjust near/far");
    folderOptions
      .add(controls, "scaleFactor", 1.1, 10, 0.1)
      .name("Scale factor");
    folderOptions.add(controls, "minDistance", 0, 50, 0.5).name("Min distance");
    folderOptions.add(controls, "maxDistance", 0, 50, 0.5).name("Max distance");
    folderOptions.add(controls, "minZoom", 0, 50, 0.5).name("Min zoom");
    folderOptions.add(controls, "maxZoom", 0, 50, 0.5).name("Max zoom");
    folderOptions
      .add(ArcballGui, "gizmoVisible")
      .name("Show gizmos")
      .onChange(function () {
        controls.setGizmosVisible(ArcballGui.gizmoVisible);
      });
    folderOptions.add(controls, "copyState").name("Copy state(ctrl+c)");
    folderOptions.add(controls, "pasteState").name("Paste state(ctrl+v)");
    folderOptions.add(controls, "reset").name("Reset");
    folderAnimations.add(controls, "enableAnimations").name("Enable anim.");
    folderAnimations.add(controls, "dampingFactor", 0, 100, 1).name("Damping");
    folderAnimations.add(controls, "wMax", 0, 100, 1).name("Angular spd");
  },
};

/**
 * <p>Initializes the scene, camera, renderer, and the arcball controls.</p>
 * @summary Loads the viewer and starts the {@link runAnimation animation}.
 */
function init() {
  const container = document.createElement("div");
  document.body.appendChild(container);

  /**
   * The WebGL renderer displays your beautifully crafted scenes using WebGL.
   * @class WebGLRenderer
   * @memberof THREE
   * @see {@link https://threejs.org/docs/#api/en/renderers/WebGLRenderer WebGLRenderer}
   */
  renderer = new THREE.WebGLRenderer({
    antialias: true,
    alpha: true,
  });
  renderer.setPixelRatio(window.devicePixelRatio);
  renderer.setSize(window.innerWidth, window.innerHeight);
  renderer.toneMapping = THREE.ReinhardToneMapping;
  renderer.toneMappingExposure = 3;
  renderer.domElement.style.background =
    "linear-gradient( 180deg, rgba( 0,0,0,1 ) 0%, rgba( 128,128,255,1 ) 100% )";
  container.appendChild(renderer.domElement);

  scene = new THREE.Scene();

  /**
   * Camera that uses perspective projection.
   * @class PerspectiveCamera
   * @memberof THREE
   * @see {@link https://threejs.org/docs/#api/en/cameras/PerspectiveCamera PerspectiveCamera}
   */
  camera = makePerspectiveCamera();
  camera.position.set(0, 0, perspectiveDistance); ///// <---------------------

  const material = new THREE.MeshStandardMaterial();

  const rgbeLoader = new RGBELoader().setPath("textures/equirectangular/");
  (async () => {
    const hdrEquirect = await rgbeLoader.loadAsync("venice_sunset_1k.hdr");
    hdrEquirect.mapping = THREE.EquirectangularReflectionMapping;
    scene.environment = hdrEquirect;
  })();

  /**
   * <p>A loader for loading a .obj resource.</p>
   * The OBJ file format is a simple data-format that represents 3D geometry in a
   * human readable format as the position of each vertex,
   * the UV position of each texture coordinate vertex, vertex normals,
   * and the faces that make each polygon defined as a list of vertices, and texture vertices.
   * @class OBJLoader
   * @memberof THREE
   * @see {@link https://threejs.org/docs/#examples/en/loaders/OBJLoader obj_loader Model Loader}
   * @see {@link https://imagetostl.com/convert/file/glb/to/obj Convert Your 3D Mesh/Model GLB Files to OBJ}
   */
  const objLoader = new OBJLoader().setPath("models/obj/cerberus/");
  objLoader.loadAsync("Cerberus.obj").then((group) => {
    const textureLoader = new THREE.TextureLoader().setPath(
      "models/obj/cerberus/",
    );

    material.roughness = 1;
    material.metalness = 1;

    const diffuseMap = textureLoader.load("Cerberus_A.jpg", render);
    diffuseMap.colorSpace = THREE.SRGBColorSpace;
    material.map = diffuseMap;

    material.metalnessMap = material.roughnessMap = textureLoader.load(
      "Cerberus_RM.jpg",
      render,
    );
    material.normalMap = textureLoader.load("Cerberus_N.jpg", render);

    material.map.wrapS = THREE.RepeatWrapping;
    material.roughnessMap.wrapS = THREE.RepeatWrapping;
    material.metalnessMap.wrapS = THREE.RepeatWrapping;
    material.normalMap.wrapS = THREE.RepeatWrapping;

    group.traverse(function (child) {
      if (child.isMesh) {
        child.material = material;
      }
    });

    group.rotation.y = Math.PI / 2;
    group.position.x += 0.25;
    scene.add(group);

    if (ArcballGui.useGUI) {
      /**
       * @summary Makes a floating panel for controllers on the web.
       * Works as a drop-in replacement for dat.gui in most projects.
       * @class GUI
       * @memberof GUI
       */
      gui = new GUI();

      gui
        .add(cameraType, "type", cameras)
        .name("Choose Camera")
        .onChange(function () {
          setCamera(cameraType.type);
        });

      folderOptions = gui.addFolder("Arcball parameters");
      folderAnimations = folderOptions.addFolder("Animations");
    }

    /**
     * There is a problem when the camera position is set after the ArcballControls is created
     * using a mobile device. The model vanishes as soon as a Pan or Zoom is started.
     * One just need to move camera.position.set() after ArcballGui.setArcballControls() to trigger it.
     */
    ArcballGui.setArcballControls();
    // camera.position.set(0, 0, perspectiveDistance); //// <------
    // controls.update(); //// <----------

    /**
     * <p>Fired when a key is pressed.</p>
     *
     * <p>The {@link onKeyDown callback}
     * argument sets the callback that will be invoked when the event is dispatched.</p>
     *
     * @summary Appends an event listener for events whose type attribute value is keydown.
     * @param {KeyboardEvent} event a UIEvent.
     * @param {callback} function function to run when the event occurs.
     * @event keydown
     * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Element/keydown_event Element: keydown event}
     */
    window.addEventListener("keydown", onKeyDown);

    /**
     * <p>Fires when the document view (window) has been resized.</p>
     * <p>The {@link onWindowResize callback} argument sets the callback
     * that will be invoked when the event is dispatched.</p>
     * @summary Appends an event listener for events whose type attribute value is resize.
     * @param {Event} event a generic event.
     * @param {callback} function function to run when the event occurs.
     * @param {Boolean} useCapture handler is executed in the bubbling or capturing phase.
     * @event resize
     * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/resize_event Window: resize event}
     */
    window.addEventListener("resize", onWindowResize);
  });
}

/**
 * Creates an orthographic camera.
 * @returns {THREE.OrthographicCamera} newCamera
 */
function makeOrthographicCamera() {
  const halfFovV = THREE.MathUtils.DEG2RAD * 45 * 0.5;
  const aspect = window.innerWidth / window.innerHeight;
  const halfFovH = Math.atan(aspect * Math.tan(halfFovV));

  const halfW = perspectiveDistance * Math.tan(halfFovH);
  const halfH = perspectiveDistance * Math.tan(halfFovV);
  const near = 0.01;
  const far = 2000;
  /**
   * Camera that uses othographic projection.
   * @class OrthographicCamera
   * @memberof THREE
   * @see {@link https://threejs.org/docs/#api/en/cameras/OrthographicCamera OrthographicCamera}
   */
  const newCamera = new THREE.OrthographicCamera(
    -halfW,
    halfW,
    halfH,
    -halfH,
    near,
    far,
  );
  return newCamera;
}

/**
 * Creates a perspective camera.
 * @returns {THREE.PerspectiveCamera} newCamera
 */
function makePerspectiveCamera() {
  const fov = 45;
  const aspect = window.innerWidth / window.innerHeight;
  const near = 0.01;
  const far = 2000;
  const newCamera = new THREE.PerspectiveCamera(fov, aspect, near, far);
  return newCamera;
}

/**
 * <p>Fires when the document view (window) has been resized.</p>
 * Also resizes the canvas and viewport.
 * @callback onWindowResize
 * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/resize_event Window: resize event}
 */
function onWindowResize() {
  const h = window.innerHeight;
  const w = window.innerWidth;
  const aspect = w / h;
  if (camera.type == "OrthographicCamera") {
    const halfFovV = THREE.MathUtils.DEG2RAD * 45 * 0.5;
    const halfFovH = Math.atan(aspect * Math.tan(halfFovV));

    const halfW = perspectiveDistance * Math.tan(halfFovH);
    const halfH = perspectiveDistance * Math.tan(halfFovV);
    camera.left = -halfW;
    camera.right = halfW;
    camera.top = halfH;
    camera.bottom = -halfH;
  } else if (camera.type == "PerspectiveCamera") {
    camera.aspect = aspect;
  }

  camera.updateProjectionMatrix();
  renderer.setSize(w, h);
  render();
}

/**
 * Render a scene or another type of object using a camera.
 * @see {@link THREE.WebGLRenderer}
 */
function render() {
  renderer.render(scene, camera);
}

/**
 * <p>Copy the current state to clipboard (as a readable JSON text) when the "ctrl-c" key is pressed or <br>
 * set the controls state from the clipboard, assumming that the clipboard holds a JSON text file <br>
 * previously saved from .copyState when the "ctrl-v" key is pressed.</p>
 *
 * @param {KeyboardEvent} event a UIEvent.
 */
function onKeyDown(event) {
  if (event.key === "c") {
    if (event.ctrlKey || event.metaKey) {
      controls.copyState();
    }
  } else if (event.key === "v") {
    if (event.ctrlKey || event.metaKey) {
      controls.pasteState();
    }
  }
}

/**
 * Sets the camera type to either "Orthographic" or "Perspective".
 * @param {String} type camera type: "Orthographic" or "Perspective".
 */
function setCamera(type) {
  if (type == "Orthographic") {
    camera = makeOrthographicCamera();
    camera.position.set(0, 0, orthographicDistance);
  } else if (type == "Perspective") {
    camera = makePerspectiveCamera();
    camera.position.set(0, 0, perspectiveDistance);
  }
  controls.setCamera(camera);
  render();
}

/**
 * <p>Fires after both the mousedown and
 * mouseup events have fired (in that order).</p>
 * Reset button must be pressed and released while the pointer is located inside it.
 * <p>The {@link https://threejs.org/docs/#examples/en/controls/ArcballControls.reset callback}
 *   argument sets the callback that will be invoked when the event is dispatched.</p>
 * @summary Appends an event listener for events whose type attribute value is click.
 * @event clickReset
 * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Element/click_event Element: click event}
 */
document.querySelector(".buttonToLink").addEventListener("click", () => {
  controls.reset();
});

/**
 * <p>Fired when the whole page has loaded, including all dependent resources
 * such as stylesheets, scripts, iframes, and images, except those that are loaded lazily.</p>
 * @summary Sets the {@link init entry point} of the application.
 * @param {Event} event load event.
 * @callback WindowLoadCallback
 * @see {@link https://developer.mozilla.org/en-US/docs/Web/API/Window/load_event Window: load event}
 * @event load
 */
window.addEventListener("load", () => {
  const queryString = window.location.search;
  const urlParams = new URLSearchParams(queryString);
  ArcballGui.useGUI = urlParams.get("gui");
  init();
});