/** * 3D Foundation Project * Copyright 2025 Smithsonian Institution * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import Subscriber from "@ff/core/Subscriber"; import { EReaderPosition } from "client/schema/setup"; import CVAnalytics from "../../components/CVAnalytics"; import CVAssetManager from "../../components/CVAssetManager"; import CVDocument from "../../components/CVDocument"; import SceneView from "../SceneView"; import "../Spinner"; import "./ActionPrompt" import "./ReaderView"; import "./CaptionView" import DocumentView, { customElement, html } from "./DocumentView"; import CRenderer from "client/../../libs/ff-scene/source/components/CRenderer"; import ARPrompt from "./ARPrompt"; import ARMenu from "./ARMenu"; import CVARManager from "client/components/CVARManager"; import CVAssetReader from "client/components/CVAssetReader"; import CaptionView from "./CaptionView"; import { EQuadViewLayout } from "client/../../libs/ff-scene/source/RenderQuadView"; //////////////////////////////////////////////////////////////////////////////// @customElement("sv-content-view") export default class ContentView extends DocumentView { protected sceneView: SceneView = null; protected captionView: CaptionView = null; protected documentProps = new Subscriber("value", this.onUpdate, this); protected isMobile: boolean = null; protected assetPath: string = ""; protected get analytics() { return this.system.getMainComponent(CVAnalytics); } protected get assetManager() { return this.system.getMainComponent(CVAssetManager); } protected get reader() { return this.activeDocument ? this.activeDocument.setup.reader : null; } protected get tours() { return this.activeDocument ? this.activeDocument.setup.tours : null; } protected get navigation() { return this.activeDocument ? this.activeDocument.setup.navigation : null; } protected get viewer() { return this.activeDocument ? this.activeDocument.setup.viewer : null; } protected get renderer() { return this.system.getMainComponent(CRenderer); } protected get arManager() { return this.system.getMainComponent(CVARManager); } protected get assetReader() { return this.system.getMainComponent(CVAssetReader); } protected firstConnected() { this.classList.add("sv-content-view"); this.sceneView = new SceneView(this.system); this.captionView = new CaptionView(this.system); this.isMobile = this.mobileCheck(); } protected connected() { super.connected(); this.assetManager.outs.busy.on("value", this.onUpdate, this); this.sceneView.on("layout", () => this.onUpdate()); this.assetPath = this.assetReader.getSystemAssetUrl(""); } protected disconnected() { this.sceneView.off("layout", () => this.onUpdate()); this.assetManager.outs.busy.off("value", this.onUpdate, this); super.disconnected(); } protected render() { const system = this.system; const isLoading = this.assetManager.outs.busy.value; const isInitialLoad = this.assetManager.outs.initialLoad.value; const sceneLoaded = this.viewer?.outs.sceneLoaded.value || false; let readerVisible = false; let readerPosition = EReaderPosition.Overlay; let tourMenuVisible = false; let promptVisible = false; const reader = this.reader; const tours = this.tours; const navigation = this.navigation; // TODO - Hack, figure out a better place for this. const overlayElement = this.arManager.shadowRoot.querySelector('ff-viewport-overlay'); if(overlayElement) { if(this.arManager.shadowRoot.querySelector('sv-ar-prompt-container') === null) { overlayElement.append(new ARPrompt(this.system)); } if(this.arManager.shadowRoot.querySelector('sv-ar-menu') === null) { overlayElement.append(new ARMenu(this.system)); } } if (tours) { tourMenuVisible = tours.ins.enabled.value && tours.outs.tourIndex.value === -1; } if (reader) { readerVisible = ! tourMenuVisible && reader.ins.enabled.value && reader.ins.visible.value; readerPosition = reader.ins.position.getValidatedValue(); // do not use right reader position on mobile if(this.isMobile === true) { readerPosition = EReaderPosition.Overlay; } } if(navigation) { const controls = navigation.ins.pointerEnabled.value; const promptEnabled = navigation.ins.promptEnabled.value; if(controls && promptEnabled) { const isInUse = navigation.ins.isInUse.value; promptVisible = !isLoading && !isInUse && !readerVisible && sceneLoaded; navigation.ins.promptActive.setValue(promptVisible); } } const sceneView = this.sceneView; const captionView = this.captionView; const blurContent = ((readerVisible && readerPosition === EReaderPosition.Overlay) || tourMenuVisible) && this.sceneView.getView().layout == EQuadViewLayout.Single; if (!blurContent) { sceneView.classList.remove("sv-blur"); } else { if(!sceneView.classList.contains("sv-blur")) { setTimeout(() => {sceneView.classList.add("sv-blur"); this.renderer.views[0].render()}, 1); // TODO: Extra for an apparent Android Firefox bug - remove when fixed } } if (readerVisible) { if (readerPosition === EReaderPosition.Right) { return html`