ply_viewer_in_js / index_2.js
bilca's picture
Update index_2.js
6ba23b4 verified
raw
history blame
13.7 kB
(async function() {
// Retrieve the current script tag and load the JSON configuration file from the data-config attribute.
const scriptTag = document.currentScript;
const configUrl = scriptTag.getAttribute("data-config");
let config = {};
if (configUrl) {
try {
const response = await fetch(configUrl);
config = await response.json();
} catch (error) {
console.error("Error loading config file:", error);
return;
}
} else {
console.error("No config file provided. Please set a data-config attribute on the script tag.");
return;
}
// --- Outer scope variables for camera state ---
let cameraInstance = null;
let controlsInstance = null;
let initialCameraPosition = null;
let initialCameraRotation = null;
// Generate a unique identifier for this widget instance.
const instanceId = Math.random().toString(36).substr(2, 8);
// Read configuration values from the JSON file.
var gifUrl = config.gif_url;
var plyUrl = config.ply_url;
var minZoom = parseFloat(config.minZoom || "0");
var maxZoom = parseFloat(config.maxZoom || "20");
var minAngle = parseFloat(config.minAngle || "0");
var maxAngle = parseFloat(config.maxAngle || "360");
var minAzimuth = config.minAzimuth !== undefined ? parseFloat(config.minAzimuth) : -Infinity;
var maxAzimuth = config.maxAzimuth !== undefined ? parseFloat(config.maxAzimuth) : Infinity;
// Determine the aspect ratio.
var aspectPercent = "100%";
if (config.aspect) {
if (config.aspect.indexOf(":") !== -1) {
var parts = config.aspect.split(":");
var w = parseFloat(parts[0]);
var h = parseFloat(parts[1]);
if (!isNaN(w) && !isNaN(h) && w > 0) {
aspectPercent = (h / w * 100) + "%";
}
} else {
var aspectValue = parseFloat(config.aspect);
if (!isNaN(aspectValue) && aspectValue > 0) {
aspectPercent = (100 / aspectValue) + "%";
}
}
} else {
var parentContainer = scriptTag.parentNode;
var containerWidth = parentContainer.offsetWidth;
var containerHeight = parentContainer.offsetHeight;
if (containerWidth > 0 && containerHeight > 0) {
aspectPercent = (containerHeight / containerWidth * 100) + "%";
}
}
// Detect if the device is iOS.
var isIOS = /iPad|iPhone|iPod/.test(navigator.userAgent) && !window.MSStream;
// Also detect Android devices.
const isMobile = isIOS || /Android/i.test(navigator.userAgent);
// Inject CSS styles into the document head, scoped with the unique id.
var styleEl = document.createElement('style');
styleEl.textContent = `
/* Widget container styling */
#ply-widget-container-${instanceId} {
position: relative;
width: 100%;
height: 0;
padding-bottom: ${aspectPercent};
}
/* When in fake fullscreen mode (iOS fallback) */
#ply-widget-container-${instanceId}.fake-fullscreen {
position: fixed !important;
top: 0 !important;
left: 0 !important;
width: 100vw !important;
height: 100vh !important;
padding-bottom: 0 !important;
z-index: 9999 !important;
}
/* GIF Preview styling */
#gif-preview-container-${instanceId} {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
border: 1px solid #474558;
border-radius: 10px;
overflow: hidden;
cursor: pointer;
}
#gif-preview-container-${instanceId} img {
width: 100%;
height: 100%;
object-fit: cover;
}
/* Viewer Container styling */
#viewer-container-${instanceId} {
display: none;
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
background: #FEFEFD;
border: 1px solid #474558;
border-radius: 10px;
}
/* Canvas fills the viewer container */
#canvas-${instanceId} {
width: 100%;
height: 100%;
display: block;
}
/* Progress dialog styling (as a centered div) */
#progress-dialog-${instanceId} {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
border: none;
background: rgba(255,255,255,0.9);
padding: 20px;
border-radius: 5px;
z-index: 1000;
display: none;
}
/* Menu (instructions) content styling */
#menu-content-${instanceId} {
display: none;
position: absolute;
top: 70px;
right: 15px;
background: #FFFEF4;
border: 1px solid #474558;
border-radius: 5px;
padding: 10px;
font-size: 15px;
line-height: 1.4;
color: #474558;
}
/* Button styling */
.widget-button {
position: absolute;
width: 45px;
height: 45px;
background-color: #FFFEF4;
border: 1px solid #474558;
border-radius: 50%;
cursor: pointer;
font-size: 14px;
color: #474558;
display: flex;
align-items: center;
justify-content: center;
}
/* Positions: Close at top-left, fullscreen at top-right, help and reset buttons */
#close-btn-${instanceId} {
top: 17px;
left: 15px;
}
#fullscreen-toggle-${instanceId} {
top: 17px;
right: 15px;
}
#help-toggle-${instanceId} {
top: 17px;
right: 70px;
font-size: 22px;
}
#reset-camera-btn-${instanceId} {
top: 17px;
right: 123px;
font-size: 22px;
line-height: normal;
padding: 0;
}
/* Adjust the reset icon position: move it slightly upward */
.reset-icon {
display: inline-block;
transform: translateY(-3px);
}
`;
document.head.appendChild(styleEl);
// Create the widget container and set its inner HTML.
var widgetContainer = document.createElement('div');
widgetContainer.id = 'ply-widget-container-' + instanceId;
widgetContainer.innerHTML = `
<!-- GIF Preview Container -->
<div id="gif-preview-container-${instanceId}">
<img id="preview-image-${instanceId}" alt="Preview" crossorigin="anonymous">
</div>
<!-- Viewer Container -->
<div id="viewer-container-${instanceId}">
<canvas id="canvas-${instanceId}"></canvas>
<div id="progress-dialog-${instanceId}">
<progress id="progress-indicator-${instanceId}" max="100" value="0"></progress>
</div>
<button id="close-btn-${instanceId}" class="widget-button">X</button>
<button id="fullscreen-toggle-${instanceId}" class="widget-button">⇱</button>
<button id="help-toggle-${instanceId}" class="widget-button">?</button>
<button id="reset-camera-btn-${instanceId}" class="widget-button">
<span class="reset-icon">⟲</span>
</button>
<div id="menu-content-${instanceId}"></div>
</div>
`;
scriptTag.parentNode.appendChild(widgetContainer);
// Grab element references.
var gifPreview = document.getElementById('gif-preview-container-' + instanceId);
var viewerContainer = document.getElementById('viewer-container-' + instanceId);
var previewImage = document.getElementById('preview-image-' + instanceId);
var closeBtn = document.getElementById('close-btn-' + instanceId);
var fullscreenToggle = document.getElementById('fullscreen-toggle-' + instanceId);
var helpToggle = document.getElementById('help-toggle-' + instanceId);
var resetCameraBtn = document.getElementById('reset-camera-btn-' + instanceId);
var menuContent = document.getElementById('menu-content-' + instanceId);
var canvas = document.getElementById('canvas-' + instanceId);
var progressDialog = document.getElementById('progress-dialog-' + instanceId);
var progressIndicator = document.getElementById('progress-indicator-' + instanceId);
// Set help instructions based on device type.
if (isMobile) {
menuContent.innerHTML = `
- Pour vous déplacer, glissez deux doigts sur l'écran.<br>
- Pour orbiter, utilisez un doigt.<br>
- Pour zoomer, pincez avec deux doigts.
`;
} else {
menuContent.innerHTML = `
- orbitez avec le clic droit<br>
- zoomez avec la molette<br>
- déplacez vous avec le clic gauche
`;
}
// If a gif_url is provided, set the preview image.
// Otherwise, hide the preview container, show the viewer immediately,
// and hide the "close" button since there's no preview to return to.
if (gifUrl) {
previewImage.src = gifUrl;
} else {
gifPreview.style.display = 'none';
viewerContainer.style.display = 'block';
closeBtn.style.display = 'none';
// Start the viewer immediately.
initializeViewer();
}
// --- Button Event Handlers ---
if (gifUrl) {
gifPreview.addEventListener('click', function() {
gifPreview.style.display = 'none';
viewerContainer.style.display = 'block';
initializeViewer();
});
}
closeBtn.addEventListener('click', function() {
if (document.fullscreenElement === widgetContainer) {
if (document.exitFullscreen) {
document.exitFullscreen();
}
}
widgetContainer.classList.remove('fake-fullscreen');
viewerContainer.style.display = 'none';
gifPreview.style.display = 'block';
});
fullscreenToggle.addEventListener('click', function() {
if (isIOS) {
if (!widgetContainer.classList.contains('fake-fullscreen')) {
widgetContainer.classList.add('fake-fullscreen');
} else {
widgetContainer.classList.remove('fake-fullscreen');
}
fullscreenToggle.textContent = widgetContainer.classList.contains('fake-fullscreen') ? '⇲' : '⇱';
} else {
if (!document.fullscreenElement) {
if (widgetContainer.requestFullscreen) {
widgetContainer.requestFullscreen();
} else if (widgetContainer.webkitRequestFullscreen) {
widgetContainer.webkitRequestFullscreen();
} else if (widgetContainer.mozRequestFullScreen) {
widgetContainer.mozRequestFullScreen();
} else if (widgetContainer.msRequestFullscreen) {
widgetContainer.msRequestFullscreen();
}
} else {
if (document.exitFullscreen) {
document.exitFullscreen();
}
}
}
});
document.addEventListener('fullscreenchange', function() {
if (document.fullscreenElement === widgetContainer) {
fullscreenToggle.textContent = '⇲';
} else {
fullscreenToggle.textContent = '⇱';
}
});
helpToggle.addEventListener('click', function(e) {
e.stopPropagation();
menuContent.style.display = (menuContent.style.display === 'block') ? 'none' : 'block';
});
resetCameraBtn.addEventListener('click', function() {
console.log("Reset camera button clicked.");
if (cameraInstance && initialCameraPosition && initialCameraRotation) {
cameraInstance.position = initialCameraPosition.clone();
cameraInstance.rotation = initialCameraRotation.clone();
if (typeof cameraInstance.update === 'function') {
cameraInstance.update();
}
if (controlsInstance && typeof controlsInstance.update === 'function') {
controlsInstance.update();
}
}
});
// --- Initialize the 3D PLY Viewer ---
async function initializeViewer() {
const SPLAT = await import("https://bilca-gsplat-library.static.hf.space/dist/index.js");
progressDialog.style.display = 'block';
const renderer = new SPLAT.WebGLRenderer(canvas);
const scene = new SPLAT.Scene();
const camera = new SPLAT.Camera();
const controls = new SPLAT.OrbitControls(camera, canvas);
cameraInstance = camera;
controlsInstance = controls;
initialCameraPosition = camera.position.clone();
initialCameraRotation = camera.rotation.clone();
canvas.style.background = "#FEFEFD";
controls.maxZoom = maxZoom;
controls.minZoom = minZoom;
controls.minAngle = minAngle;
controls.maxAngle = maxAngle;
controls.minAzimuth = minAzimuth;
controls.maxAzimuth = maxAzimuth;
// Set panSpeed to 0.1 for all devices.
controls.panSpeed = 0.1;
// Set moveSpeed based on device: 0.5 on mobile, 1 on computer.
const moveSpeed = isMobile ? 0.5 : 1;
// Update the controls' internal move speed in the update loop.
// (Assuming your OrbitControls.update() function uses a moveSpeed variable)
// We'll override the moveSpeed in the update function via closure.
const originalUpdate = controls.update;
controls.update = function() {
// In your update() implementation, use the new moveSpeed instead of a hardcoded value.
// (This assumes your library's OrbitControls code uses a moveSpeed constant inside update,
// you may need to modify that file directly.)
originalUpdate.call(controls, moveSpeed);
};
controls.update();
try {
await SPLAT.PLYLoader.LoadAsync(
plyUrl,
scene,
(progress) => {
progressIndicator.value = progress * 100;
}
);
progressDialog.style.display = 'none';
} catch (error) {
console.error("Error loading PLY file:", error);
progressDialog.innerHTML = `<p style="color: red">Error loading model: ${error.message}</p>`;
}
const frame = () => {
controls.update();
renderer.render(scene, camera);
requestAnimationFrame(frame);
};
const handleResize = () => {
renderer.setSize(canvas.clientWidth, canvas.clientHeight);
};
handleResize();
window.addEventListener("resize", handleResize);
requestAnimationFrame(frame);
}
// If a gif_url exists, the viewer is started on preview click;
// otherwise, it was already started above.
})();