mirror of
https://github.com/overte-org/community-apps.git
synced 2025-04-05 21:22:00 +02:00
Add the application CAM360
Add the application CAM360 (Formerly known as RPO360, but with significant modifications)
This commit is contained in:
parent
b86532f43e
commit
b8da54c053
22 changed files with 1491 additions and 0 deletions
406
applications/cam360/cam360.html
Normal file
406
applications/cam360/cam360.html
Normal file
|
@ -0,0 +1,406 @@
|
|||
<!DOCTYPE html>
|
||||
<!--
|
||||
cam360.html
|
||||
|
||||
Created by Alezia Kurdis, August 27th 2022.
|
||||
Copyright 2022, Overte e.V.
|
||||
|
||||
UI for an application to take 360 degrees photo by throwing a camera in the air.
|
||||
|
||||
Distributed under the Apache License, Version 2.0.
|
||||
See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
||||
-->
|
||||
<html>
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<script src="resources/marzipano/marzipano.js" ></script>
|
||||
<script>
|
||||
|
||||
//Paths
|
||||
var thisPageName = "cam360.html";
|
||||
var currentPath = window.location.protocol + "//" + window.location.host + window.location.pathname;
|
||||
var ROOTPATH = currentPath.replace(thisPageName, "");
|
||||
var channel = "cam360.ak.overte.com";
|
||||
|
||||
var last360photo = "";
|
||||
var isCameraActive = false;
|
||||
var isPhotoProcessing = false;
|
||||
var useFlash = false;
|
||||
var isThrowMode = true;
|
||||
var DEG_TO_RAD = Math.PI/180;
|
||||
|
||||
//LISTENER FROM JS FILE:
|
||||
EventBridge.scriptEventReceived.connect(function (message) {
|
||||
var messageObj = JSON.parse(message);
|
||||
if (messageObj.channel === channel && messageObj.method === "last360ThumbnailURL") {
|
||||
last360photo = messageObj.last360ThumbnailURL;
|
||||
renderPreview();
|
||||
} else if (messageObj.channel === channel && messageObj.method === "initializeUI") {
|
||||
isCameraActive = messageObj.masterSwitchOn;
|
||||
last360photo = messageObj.last360ThumbnailURL;
|
||||
isPhotoProcessing = messageObj.processing360Snapshot;
|
||||
useFlash = messageObj.useFlash;
|
||||
isThrowMode = messageObj.isThrowMode;
|
||||
renderCameraStatus();
|
||||
setFlashButton();
|
||||
renderPreview();
|
||||
renderMode();
|
||||
|
||||
} else if (messageObj.channel === channel && messageObj.method === "finishedProcessing360Snapshot") {
|
||||
document.getElementById("processing").innerHTML = "";
|
||||
} else if (messageObj.channel === channel && messageObj.method === "startedProcessing360Snapshot") {
|
||||
document.getElementById("processing").innerHTML = "<img src='resources/images/processing.gif'>";
|
||||
} else if (messageObj.channel === channel && messageObj.method === "yawPitchRoll") {
|
||||
setView(messageObj.yaw, messageObj.pitch, messageObj.roll);
|
||||
}
|
||||
});
|
||||
|
||||
function setFlashButton() {
|
||||
if (useFlash) {
|
||||
document.getElementById("flashOnOff").className = "flashBtnOn";
|
||||
} else {
|
||||
document.getElementById("flashOnOff").className = "flashBtnOff";
|
||||
}
|
||||
}
|
||||
|
||||
function toggleFlash() {
|
||||
var messageToSend;
|
||||
if (useFlash) {
|
||||
useFlash = false;
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "disableFlash"
|
||||
};
|
||||
} else {
|
||||
useFlash = true;
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "enableFlash"
|
||||
};
|
||||
}
|
||||
setFlashButton();
|
||||
EventBridge.emitWebEvent(JSON.stringify(messageToSend));
|
||||
}
|
||||
|
||||
function openSetting() {
|
||||
var messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "openSettings"
|
||||
};
|
||||
EventBridge.emitWebEvent(JSON.stringify(messageToSend));
|
||||
}
|
||||
|
||||
function renderPreview() {
|
||||
if (last360photo === "") {
|
||||
setScene("resources/images/default.jpg");
|
||||
} else {
|
||||
setScene(last360photo);
|
||||
}
|
||||
}
|
||||
|
||||
function activateRpo360() {
|
||||
var messageToSend;
|
||||
if (isCameraActive) {
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "rpo360Off"
|
||||
};
|
||||
isCameraActive = false;
|
||||
} else {
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "rpo360On"
|
||||
};
|
||||
isCameraActive = true;
|
||||
}
|
||||
EventBridge.emitWebEvent(JSON.stringify(messageToSend));
|
||||
renderCameraStatus();
|
||||
}
|
||||
|
||||
function renderCameraStatus() {
|
||||
if (isCameraActive) {
|
||||
document.getElementById("cameraIndicator").src = "resources/images/on.png";
|
||||
document.getElementById("cameraOnOff").innerHTML = "STOP CAMERA";
|
||||
} else {
|
||||
document.getElementById("cameraIndicator").src = "resources/images/off.png";
|
||||
document.getElementById("cameraOnOff").innerHTML = "START CAMERA";
|
||||
}
|
||||
}
|
||||
|
||||
function setMode(mode) {
|
||||
var messageToSend;
|
||||
if (isThrowMode) {
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "PositionMode"
|
||||
};
|
||||
isThrowMode = false;
|
||||
} else {
|
||||
messageToSend = {
|
||||
"channel": channel,
|
||||
"method": "ThrowMode"
|
||||
};
|
||||
isThrowMode = true;
|
||||
}
|
||||
EventBridge.emitWebEvent(JSON.stringify(messageToSend));
|
||||
renderMode();
|
||||
}
|
||||
|
||||
function renderMode() {
|
||||
if (isThrowMode) {
|
||||
document.getElementById("modeThrow").className = "modeBtnOn";
|
||||
document.getElementById("modePosition").className = "modeBtnOff";
|
||||
document.getElementById("throwModeControls").style.display = "block";
|
||||
document.getElementById("positionModeControls").style.display = "none";
|
||||
} else {
|
||||
document.getElementById("modeThrow").className = "modeBtnOff";
|
||||
document.getElementById("modePosition").className = "modeBtnOn";
|
||||
document.getElementById("throwModeControls").style.display = "none";
|
||||
document.getElementById("positionModeControls").style.display = "flex";
|
||||
}
|
||||
}
|
||||
|
||||
function capture() {
|
||||
var messageToSend= {
|
||||
"channel": channel,
|
||||
"method": "Capture"
|
||||
};
|
||||
EventBridge.emitWebEvent(JSON.stringify(messageToSend));
|
||||
}
|
||||
</script>
|
||||
<style>
|
||||
@font-face {
|
||||
font-family: FiraSans-SemiBold;
|
||||
src: url(resources/fonts/FiraSans-SemiBold.ttf);
|
||||
}
|
||||
html {
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
|
||||
body {
|
||||
background-color: #21293d;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
color: #FFFFFF;
|
||||
font-weight: 600;
|
||||
text-decoration: none;
|
||||
font-style: normal;
|
||||
font-variant: normal;
|
||||
text-transform: none;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
min-height: 100%;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
overflow: hidden;
|
||||
}
|
||||
h1 {
|
||||
padding: 0px 10px 0px 10px;
|
||||
font-size: 18px;
|
||||
}
|
||||
#cameraOnOff{
|
||||
border: 1px solid #4f659c;
|
||||
color: #7d9ae3;
|
||||
background-color: #36415e;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 14px;
|
||||
padding: 6px 10px 6px 10px;
|
||||
min-width: 120px;
|
||||
margin: 2px;
|
||||
border-radius: 6px;
|
||||
}
|
||||
#pano {
|
||||
position: relative;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
}
|
||||
#previewContainer {
|
||||
width: 100%;
|
||||
height: 550px;
|
||||
}
|
||||
#cameraIndicator {
|
||||
margin: 2px 5px 2px 5px;
|
||||
vertical-align: middle;
|
||||
}
|
||||
.flashBtnOn {
|
||||
border: 1px solid #ffe657;
|
||||
color: #ffe657;
|
||||
background-color: #8f8029;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
padding: 2px 7px 2px 7px;
|
||||
min-width: 90px;
|
||||
margin: 2px;
|
||||
}
|
||||
.flashBtnOff {
|
||||
border: 1px solid #444444;
|
||||
color: #444444;
|
||||
background-color: #222222;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
padding: 2px 7px 2px 7px;
|
||||
min-width: 90px;
|
||||
margin: 2px;
|
||||
}
|
||||
.folderSetting {
|
||||
border: 1px solid #4f659c;
|
||||
color: #7d9ae3;
|
||||
background-color: #36415e;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
padding: 2px 7px 2px 7px;
|
||||
min-width: 90px;
|
||||
margin: 2px;
|
||||
border-radius: 6px;
|
||||
}
|
||||
*:focus {
|
||||
outline: none;
|
||||
}
|
||||
.modeBtnOn {
|
||||
border: 1px solid #ffe657;
|
||||
color: #ffe657;
|
||||
background-color: #8f8029;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
padding: 2px 7px 2px 7px;
|
||||
min-width: 110px;
|
||||
margin: 2px;
|
||||
}
|
||||
.modeBtnOff {
|
||||
border: 1px solid #444444;
|
||||
color: #444444;
|
||||
background-color: #222222;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 12px;
|
||||
padding: 2px 7px 2px 7px;
|
||||
min-width: 110px;
|
||||
margin: 2px;
|
||||
}
|
||||
#throwModeControls {
|
||||
display: block;
|
||||
width: 100%;
|
||||
padding: 10px;
|
||||
}
|
||||
#positionModeControls {
|
||||
display: none;
|
||||
vertical-align: middle;
|
||||
width: 100%;
|
||||
}
|
||||
#capture {
|
||||
border: 1px solid #ff5429;
|
||||
color: #ff5429;
|
||||
background-color: #9c290c;
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 16px;
|
||||
padding: 10px 5px 10px 5px;
|
||||
min-width: 110px;
|
||||
margin: 6px;
|
||||
border-radius: 13px;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<table style = "width: 100%; border-collapse: collapse;">
|
||||
<tr>
|
||||
<td style = "width: 40%;">
|
||||
<h1>CAM360 v2.0</h1>
|
||||
</td>
|
||||
<td style = "width: 60%;">
|
||||
<div id="processing"></div>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td colspan = '2' style = "width: 100%;">
|
||||
<div id= "previewContainer">
|
||||
<div id="pano"></div>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<td style = "width: 40%; vertical-align: middle;">
|
||||
<img id = "cameraIndicator" src="resources/images/off.png"> <button id="cameraOnOff" onClick='activateRpo360();'>OFF</button>
|
||||
</td>
|
||||
<td style = "width: 60%; vertical-align: middle; text-align: right;">
|
||||
<div style="width: 100%; text-align: right;">
|
||||
<button id="modeThrow" class="modeBtnOn" onClick='setMode("throw");'>THROW MODE</button>
|
||||
<button id="modePosition" class="modeBtnOff" onClick='setMode("position");'>POSITION MODE</button>
|
||||
</div>
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
<table style = "width: 100%; border-collapse: collapse;">
|
||||
<tr>
|
||||
<td style = "width: 75%; padding: 5px; text-align: left;">
|
||||
<div id="throwModeControls">
|
||||
Throw the camera in the air to take a 360 degrees snapshot!
|
||||
</div>
|
||||
<div id="positionModeControls">
|
||||
<div> </div>
|
||||
<div><button id="capture" onClick='capture();'>CAPTURE</button></div>
|
||||
<div> </div>
|
||||
<div><i><br>You can also click on the thumbstick<br>on your right controller.</i></div>
|
||||
</div>
|
||||
</td>
|
||||
<td style = "width: 25%; padding: 5px; text-align: right;">
|
||||
<button id="flashOnOff" class="flashBtnOff" onClick='toggleFlash();'>FLASH 🗲</button><br>
|
||||
<button id="folderSetting" class="folderSetting" onClick='openSetting();'>SETTINGS 🖿</button>
|
||||
</td>
|
||||
</tr>
|
||||
</table>
|
||||
<script>
|
||||
EventBridge.emitWebEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "uiReady"
|
||||
}));
|
||||
|
||||
//Panoramic preview related code:
|
||||
// Create viewer.
|
||||
var panoElement = document.getElementById('pano');
|
||||
var viewerOpts = {
|
||||
controls: {
|
||||
mouseViewMode: 'drag' // drag|qtvr
|
||||
}
|
||||
};
|
||||
var viewer = new Marzipano.Viewer(panoElement, viewerOpts);
|
||||
|
||||
// Create source.
|
||||
var source = Marzipano.ImageUrlSource.fromString("resources/images/blank.jpg");
|
||||
|
||||
// Create geometry.
|
||||
var geometry = new Marzipano.EquirectGeometry([{ width: 4000 }]);
|
||||
|
||||
// Create view.
|
||||
var limiter = Marzipano.RectilinearView.limit.traditional(1024, 100*Math.PI/180);
|
||||
var view = new Marzipano.RectilinearView({ yaw: Math.PI }, limiter);
|
||||
|
||||
var scene;
|
||||
|
||||
// Create scene.
|
||||
function setScene(photoUrl) {
|
||||
source = Marzipano.ImageUrlSource.fromString(photoUrl);
|
||||
|
||||
scene = viewer.createScene({
|
||||
source: source,
|
||||
geometry: geometry,
|
||||
view: view,
|
||||
pinFirstLevel: true
|
||||
});
|
||||
|
||||
// Display scene.
|
||||
scene.switchTo({transitionDuration: 1000});
|
||||
}
|
||||
|
||||
function setView(targetYaw, targetPitch, targetRoll){
|
||||
|
||||
view.setYaw(targetYaw * DEG_TO_RAD);
|
||||
view.setPitch(targetPitch* DEG_TO_RAD);
|
||||
view.setRoll(targetRoll* DEG_TO_RAD);
|
||||
|
||||
}
|
||||
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
605
applications/cam360/cam360.js
Normal file
605
applications/cam360/cam360.js
Normal file
|
@ -0,0 +1,605 @@
|
|||
"use strict";
|
||||
//
|
||||
// cam360.js
|
||||
//
|
||||
// Created by Zach Fox on 2018-10-26
|
||||
// Copyright 2018 High Fidelity, Inc.
|
||||
// Copyright 2022, Overte e.V.
|
||||
//
|
||||
// Application to take 360 degrees photo by throwing a camera in the air (as in Ready Player One (RPO)) or as a standard positionned camera.
|
||||
// version 2.0
|
||||
//
|
||||
// Distributed under the Apache License, Version 2.0
|
||||
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
||||
//
|
||||
|
||||
(function () { // BEGIN LOCAL_SCOPE
|
||||
|
||||
// Function Name: inFrontOf()
|
||||
// Description:
|
||||
// - Returns the position in front of the given "position" argument, where the forward vector is based off
|
||||
// the "orientation" argument and the amount in front is based off the "distance" argument.
|
||||
function inFrontOf(distance, position, orientation) {
|
||||
return Vec3.sum(position || MyAvatar.position,
|
||||
Vec3.multiply(distance, Quat.getForward(orientation || MyAvatar.orientation)));
|
||||
}
|
||||
|
||||
// Function Name: rpo360On()
|
||||
var CAMERA_NAME = "CAM360 Camera";
|
||||
var SETTING_LAST_360_CAPTURE = "overte_app_cam360_last_capture";
|
||||
var secondaryCameraConfig = Render.getConfig("SecondaryCamera");
|
||||
var camera = false;
|
||||
var cameraRotation;
|
||||
var cameraPosition;
|
||||
var cameraGravity = {x: 0, y: -5, z: 0};
|
||||
var velocityLoopInterval = false;
|
||||
var isThrowMode = true;
|
||||
|
||||
function rpo360On() {
|
||||
// Rez the camera model, and attach
|
||||
// the secondary camera to the rezzed model.
|
||||
cameraRotation = MyAvatar.orientation;
|
||||
cameraPosition = inFrontOf(1.0, Vec3.sum(MyAvatar.position, { x: 0, y: 0.3, z: 0 }));
|
||||
var properties;
|
||||
var hostType = "";
|
||||
if (isThrowMode) {
|
||||
properties = {
|
||||
"angularDamping": 0.08,
|
||||
"canCastShadow": false,
|
||||
"damping": 0.01,
|
||||
"collisionMask": 7,
|
||||
"modelURL": Script.resolvePath("resources/models/cam360white.fst"),
|
||||
"name": CAMERA_NAME,
|
||||
"rotation": cameraRotation,
|
||||
"position": cameraPosition,
|
||||
"shapeType": "simple-compound",
|
||||
"type": "Model",
|
||||
"grab": {
|
||||
"grabbable": true
|
||||
},
|
||||
"script": Script.resolvePath("grabDetection.js"),
|
||||
"userData": "",
|
||||
"isVisibleInSecondaryCamera": false,
|
||||
"gravity": cameraGravity,
|
||||
"dynamic": true
|
||||
};
|
||||
hostType = "local";
|
||||
} else {
|
||||
properties = {
|
||||
"canCastShadow": false,
|
||||
"collisionMask": 7,
|
||||
"modelURL": Script.resolvePath("resources/models/cam360black.fst"),
|
||||
"name": CAMERA_NAME,
|
||||
"rotation": cameraRotation,
|
||||
"position": cameraPosition,
|
||||
"shapeType": "sphere",
|
||||
"type": "Model",
|
||||
"grab": {
|
||||
"grabbable": true
|
||||
},
|
||||
"userData": "",
|
||||
"isVisibleInSecondaryCamera": false
|
||||
};
|
||||
hostType = "avatar";
|
||||
}
|
||||
|
||||
camera = Entities.addEntity(properties, hostType);
|
||||
secondaryCameraConfig.attachedEntityId = camera;
|
||||
|
||||
// Play a little sound to let the user know we've rezzed the camera
|
||||
Audio.playSound(SOUND_CAMERA_ON, {
|
||||
"volume": 0.15,
|
||||
"position": cameraPosition,
|
||||
"localOnly": true
|
||||
});
|
||||
|
||||
// Remove the existing camera model from the domain if one exists.
|
||||
// It's easy for this to happen if the user crashes while the RPO360 Camera is on.
|
||||
// We do this down here (after the new one is rezzed) so that we don't accidentally delete
|
||||
// the newly-rezzed model.
|
||||
var entityIDs = Entities.findEntitiesByName(CAMERA_NAME, MyAvatar.position, 100, false);
|
||||
entityIDs.forEach(function (currentEntityID) {
|
||||
var currentEntityOwner = Entities.getEntityProperties(currentEntityID, ['owningAvatarID']).owningAvatarID;
|
||||
if (currentEntityOwner === MyAvatar.sessionUUID && currentEntityID !== camera) {
|
||||
Entities.deleteEntity(currentEntityID);
|
||||
}
|
||||
});
|
||||
|
||||
setTakePhotoControllerMappingStatus();
|
||||
|
||||
// Start the velocity loop interval at 70ms
|
||||
// This is used to determine when the 360 photo should be snapped
|
||||
if (isThrowMode) {
|
||||
velocityLoopInterval = Script.setInterval(velocityLoop, 70);
|
||||
}
|
||||
}
|
||||
|
||||
// Function Name: velocityLoop()
|
||||
var hasBeenThrown = false;
|
||||
var snapshotVelocity = false;
|
||||
var snapshotAngularVelocity = false;
|
||||
var velocityWasPositive = false;
|
||||
var cameraReleaseTime = false;
|
||||
var MIN_AIRTIME_MS = 500;
|
||||
var flash = false;
|
||||
var useFlash = false;
|
||||
function velocityLoop() {
|
||||
// Get the velocity and angular velocity of the camera model
|
||||
var properties = Entities.getEntityProperties(camera, ["velocity", "angularVelocity", "userData"]);
|
||||
var velocity = properties.velocity;
|
||||
var angularVelocity = properties.angularVelocity;
|
||||
var releasedState = properties.userData;
|
||||
|
||||
if (releasedState === "RELEASED" && !hasBeenThrown) {
|
||||
hasBeenThrown = true;
|
||||
// Record the time at which a user has thrown the camera
|
||||
cameraReleaseTime = Date.now();
|
||||
}
|
||||
|
||||
// If we've been thrown UP...
|
||||
if (hasBeenThrown && velocity.y > 0) {
|
||||
// Set this flag to true
|
||||
velocityWasPositive = true;
|
||||
}
|
||||
|
||||
// If we've been thrown UP in the past, but now we're coming DOWN...
|
||||
if (hasBeenThrown && velocityWasPositive && velocity.y < 0) {
|
||||
// Reset the state machine
|
||||
hasBeenThrown = false;
|
||||
velocityWasPositive = false;
|
||||
// Don't take a snapshot if the camera hasn't been in the air for very long
|
||||
if (Date.now() - cameraReleaseTime <= MIN_AIRTIME_MS) {
|
||||
Entities.editEntity(camera, {
|
||||
"userData": ""
|
||||
});
|
||||
return;
|
||||
}
|
||||
// Save these properties so that the camera falls realistically
|
||||
// after it's taken the 360 snapshot
|
||||
snapshotVelocity = velocity;
|
||||
snapshotAngularVelocity = angularVelocity;
|
||||
// Freeze the camera model and make it not grabbable
|
||||
Entities.editEntity(camera, {
|
||||
"velocity": {"x": 0, "y": 0, "z": 0},
|
||||
"angularVelocity": {"x": 0, "y": 0, "z": 0},
|
||||
"gravity": {"x": 0, "y": 0, "z": 0},
|
||||
"grab": {
|
||||
"grabbable": false
|
||||
},
|
||||
"userData": ""
|
||||
});
|
||||
// Add a "flash" to the camera that illuminates the ground below the camera
|
||||
if (useFlash) {
|
||||
flash = Entities.addEntity({
|
||||
"collidesWith": "",
|
||||
"collisionMask": 0,
|
||||
"color": {
|
||||
"blue": 173,
|
||||
"green": 252,
|
||||
"red": 255
|
||||
},
|
||||
"dimensions": {
|
||||
"x": 100,
|
||||
"y": 100,
|
||||
"z": 100
|
||||
},
|
||||
"dynamic": false,
|
||||
"falloffRadius": 10,
|
||||
"intensity": 1,
|
||||
"isSpotlight": false,
|
||||
"localRotation": { w: 1, x: 0, y: 0, z: 0 },
|
||||
"name": CAMERA_NAME + "_Flash",
|
||||
"type": "Light",
|
||||
"parentID": camera
|
||||
}, "local");
|
||||
}
|
||||
// Take the snapshot!
|
||||
maybeTake360Snapshot();
|
||||
}
|
||||
}
|
||||
|
||||
function capture() {
|
||||
if (!isThrowMode) {
|
||||
if (useFlash) {
|
||||
flash = Entities.addEntity({
|
||||
"collidesWith": "",
|
||||
"collisionMask": 0,
|
||||
"color": {
|
||||
"blue": 173,
|
||||
"green": 252,
|
||||
"red": 255
|
||||
},
|
||||
"dimensions": {
|
||||
"x": 100,
|
||||
"y": 100,
|
||||
"z": 100
|
||||
},
|
||||
"dynamic": false,
|
||||
"falloffRadius": 10,
|
||||
"intensity": 1,
|
||||
"isSpotlight": false,
|
||||
"localRotation": { w: 1, x: 0, y: 0, z: 0 },
|
||||
"name": CAMERA_NAME + "_Flash",
|
||||
"type": "Light",
|
||||
"parentID": camera
|
||||
}, "avatar");
|
||||
}
|
||||
// Take the snapshot!
|
||||
maybeTake360Snapshot();
|
||||
}
|
||||
}
|
||||
|
||||
// Function Name: rpo360Off()
|
||||
var WAIT_AFTER_DOMAIN_SWITCH_BEFORE_CAMERA_DELETE_MS = 1 * 1000;
|
||||
function rpo360Off(isChangingDomains) {
|
||||
if (velocityLoopInterval) {
|
||||
Script.clearInterval(velocityLoopInterval);
|
||||
velocityLoopInterval = false;
|
||||
}
|
||||
|
||||
function deleteCamera() {
|
||||
if (flash) {
|
||||
Entities.deleteEntity(flash);
|
||||
flash = false;
|
||||
}
|
||||
if (camera) {
|
||||
Entities.deleteEntity(camera);
|
||||
camera = false;
|
||||
}
|
||||
//buttonActive(ui.isOpen);
|
||||
}
|
||||
|
||||
secondaryCameraConfig.attachedEntityId = false;
|
||||
if (camera) {
|
||||
// Workaround for Avatar Entities not immediately having properties after
|
||||
// the "Window.domainChanged()" signal is emitted.
|
||||
// May no longer be necessary; untested...
|
||||
if (isChangingDomains) {
|
||||
Script.setTimeout(function () {
|
||||
deleteCamera();
|
||||
rpo360On();
|
||||
}, WAIT_AFTER_DOMAIN_SWITCH_BEFORE_CAMERA_DELETE_MS);
|
||||
} else {
|
||||
deleteCamera();
|
||||
}
|
||||
}
|
||||
setTakePhotoControllerMappingStatus();
|
||||
}
|
||||
|
||||
var isCurrentlyTaking360Snapshot = false;
|
||||
var processing360Snapshot = false;
|
||||
function maybeTake360Snapshot() {
|
||||
// Don't take a snapshot if we're currently in the middle of taking one
|
||||
// or if the camera entity doesn't exist
|
||||
if (!isCurrentlyTaking360Snapshot && camera) {
|
||||
isCurrentlyTaking360Snapshot = true;
|
||||
var currentCameraPosition = Entities.getEntityProperties(camera, ["position"]).position;
|
||||
// Play a sound at the current camera position
|
||||
Audio.playSound(SOUND_SNAPSHOT, {
|
||||
"position": { "x": currentCameraPosition.x, "y": currentCameraPosition.y, "z": currentCameraPosition.z },
|
||||
"localOnly": isThrowMode,
|
||||
"volume": 0.8
|
||||
});
|
||||
Window.takeSecondaryCamera360Snapshot(currentCameraPosition);
|
||||
used360AppToTakeThisSnapshot = true;
|
||||
processing360Snapshot = true;
|
||||
|
||||
// Let the UI know we're processing a 360 snapshot now
|
||||
tablet.emitScriptEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "startedProcessing360Snapshot"
|
||||
}));
|
||||
}
|
||||
}
|
||||
|
||||
function on360SnapshotTaken(path) {
|
||||
isCurrentlyTaking360Snapshot = false;
|
||||
// Make the camera fall back to the ground with the same
|
||||
// physical properties as when it froze in the air
|
||||
if (isThrowMode) {
|
||||
Entities.editEntity(camera, {
|
||||
"velocity": snapshotVelocity,
|
||||
"angularVelocity": snapshotAngularVelocity,
|
||||
"gravity": cameraGravity,
|
||||
"grab": {
|
||||
"grabbable": true
|
||||
}
|
||||
});
|
||||
}
|
||||
// Delete the flash entity
|
||||
if (flash) {
|
||||
Entities.deleteEntity(flash);
|
||||
flash = false;
|
||||
}
|
||||
//console.log('360 Snapshot taken. Path: ' + path);
|
||||
|
||||
//update UI
|
||||
tablet.emitScriptEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "last360ThumbnailURL",
|
||||
"last360ThumbnailURL": path
|
||||
}));
|
||||
last360ThumbnailURL = path;
|
||||
Settings.setValue(SETTING_LAST_360_CAPTURE, last360ThumbnailURL);
|
||||
processing360Snapshot = false;
|
||||
tablet.emitScriptEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "finishedProcessing360Snapshot"
|
||||
}));
|
||||
}
|
||||
|
||||
|
||||
var last360ThumbnailURL = Settings.getValue(SETTING_LAST_360_CAPTURE, "");
|
||||
var used360AppToTakeThisSnapshot = false;
|
||||
|
||||
function onDomainChanged() {
|
||||
rpo360Off(true);
|
||||
}
|
||||
|
||||
// These functions will be called when the script is loaded.
|
||||
var SOUND_CAMERA_ON = SoundCache.getSound(Script.resolvePath("resources/sounds/cameraOn.wav"));
|
||||
var SOUND_SNAPSHOT = SoundCache.getSound(Script.resolvePath("resources/sounds/snap.wav"));
|
||||
|
||||
|
||||
var jsMainFileName = "cam360.js";
|
||||
var ROOT = Script.resolvePath('').split(jsMainFileName)[0];
|
||||
|
||||
var APP_NAME = "CAM360";
|
||||
var APP_URL = ROOT + "cam360.html";
|
||||
var APP_ICON_INACTIVE = ROOT + "resources/images/icons/cam360-i.svg";
|
||||
var APP_ICON_ACTIVE = ROOT + "resources/images/icons/cam360-a.svg";
|
||||
var appStatus = false;
|
||||
var channel = "cam360.ak.overte.com";
|
||||
|
||||
var timestamp = 0;
|
||||
var INTERCALL_DELAY = 200; //0.3 sec
|
||||
var DEG_TO_RAD = Math.PI/180;
|
||||
|
||||
var tablet = Tablet.getTablet("com.highfidelity.interface.tablet.system");
|
||||
|
||||
Window.domainChanged.connect(onDomainChanged);
|
||||
Window.snapshot360Taken.connect(on360SnapshotTaken);
|
||||
HMD.displayModeChanged.connect(onHMDChanged);
|
||||
|
||||
camera = false;
|
||||
|
||||
tablet.screenChanged.connect(onScreenChanged);
|
||||
|
||||
var button = tablet.addButton({
|
||||
text: APP_NAME,
|
||||
icon: APP_ICON_INACTIVE,
|
||||
activeIcon: APP_ICON_ACTIVE
|
||||
});
|
||||
|
||||
|
||||
function clicked(){
|
||||
if (appStatus === true) {
|
||||
tablet.webEventReceived.disconnect(onAppWebEventReceived);
|
||||
tablet.gotoHomeScreen();
|
||||
appStatus = false;
|
||||
}else{
|
||||
tablet.gotoWebScreen(APP_URL);
|
||||
tablet.webEventReceived.connect(onAppWebEventReceived);
|
||||
appStatus = true;
|
||||
}
|
||||
|
||||
button.editProperties({
|
||||
isActive: appStatus || camera
|
||||
});
|
||||
}
|
||||
|
||||
button.clicked.connect(clicked);
|
||||
|
||||
|
||||
function onAppWebEventReceived(message){
|
||||
var d = new Date();
|
||||
var n = d.getTime();
|
||||
var messageObj = JSON.parse(message);
|
||||
if (messageObj.channel === channel) {
|
||||
if (messageObj.method === "rpo360On" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
rpo360On();
|
||||
|
||||
} else if (messageObj.method === "rpo360Off" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
rpo360Off();
|
||||
|
||||
} else if (messageObj.method === "openSettings" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
if ((HMD.active && Settings.getValue("hmdTabletBecomesToolbar", false)) || (!HMD.active && Settings.getValue("desktopTabletBecomesToolbar", true))) {
|
||||
Desktop.show("hifi/dialogs/GeneralPreferencesDialog.qml", "GeneralPreferencesDialog");
|
||||
} else {
|
||||
tablet.pushOntoStack("hifi/tablet/TabletGeneralPreferences.qml");
|
||||
}
|
||||
|
||||
} else if (messageObj.method === "disableFlash" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
useFlash = false;
|
||||
|
||||
} else if (messageObj.method === "enableFlash" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
useFlash = true;
|
||||
|
||||
} else if (messageObj.method === "uiReady" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
tablet.emitScriptEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "initializeUI",
|
||||
"masterSwitchOn": !!camera,
|
||||
"last360ThumbnailURL": last360ThumbnailURL,
|
||||
"processing360Snapshot": processing360Snapshot,
|
||||
"useFlash": useFlash,
|
||||
"isThrowMode": isThrowMode
|
||||
}));
|
||||
|
||||
} else if (messageObj.method === "ThrowMode" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
isThrowMode = true;
|
||||
if (camera) {
|
||||
rpo360Off();
|
||||
rpo360On();
|
||||
}
|
||||
} else if (messageObj.method === "PositionMode" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
isThrowMode = false;
|
||||
if (camera) {
|
||||
rpo360Off();
|
||||
rpo360On();
|
||||
}
|
||||
} else if (messageObj.method === "Capture" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
if (camera) {
|
||||
capture();
|
||||
}
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
var udateSignateDisconnected = true;
|
||||
function onScreenChanged(type, url) {
|
||||
if (type === "Web" && url.indexOf(APP_URL) !== -1) {
|
||||
appStatus = true;
|
||||
Script.update.connect(myTimer);
|
||||
udateSignateDisconnected = false;
|
||||
} else {
|
||||
appStatus = false;
|
||||
if (!udateSignateDisconnected) {
|
||||
Script.update.disconnect(myTimer);
|
||||
udateSignateDisconnected = true;
|
||||
}
|
||||
}
|
||||
|
||||
button.editProperties({
|
||||
isActive: appStatus || camera
|
||||
});
|
||||
}
|
||||
|
||||
function myTimer(deltaTime) {
|
||||
var yaw = 0.0;
|
||||
var pitch = 0.0;
|
||||
var roll = 0.0;
|
||||
var euler;
|
||||
if (!HMD.active) {
|
||||
//Use cuser camera for destop
|
||||
euler = Quat.safeEulerAngles(Camera.orientation);
|
||||
yaw = -euler.y;
|
||||
pitch = -euler.x;
|
||||
roll = -euler.z;
|
||||
} else {
|
||||
//Use Tablet orientation for HMD
|
||||
var tabletRotation = Entities.getEntityProperties(HMD.tabletID, ["rotation"]).rotation;
|
||||
var noRoll = Quat.cancelOutRoll(tabletRotation); //Pushing the roll is getting quite complexe
|
||||
euler = Quat.safeEulerAngles(noRoll);
|
||||
yaw = euler.y - 180;
|
||||
if (yaw < -180) { yaw = yaw + 360;}
|
||||
yaw = -yaw;
|
||||
pitch = euler.x;
|
||||
roll = 0;
|
||||
}
|
||||
|
||||
tablet.emitScriptEvent(JSON.stringify({
|
||||
"channel": channel,
|
||||
"method": "yawPitchRoll",
|
||||
"yaw": yaw,
|
||||
"pitch": pitch,
|
||||
"roll": roll
|
||||
}));
|
||||
|
||||
}
|
||||
|
||||
function cleanup() {
|
||||
|
||||
if (appStatus) {
|
||||
tablet.gotoHomeScreen();
|
||||
tablet.webEventReceived.disconnect(onAppWebEventReceived);
|
||||
if (!udateSignateDisconnected) {
|
||||
Script.update.disconnect(myTimer);
|
||||
udateSignateDisconnected = true;
|
||||
}
|
||||
}
|
||||
|
||||
tablet.screenChanged.disconnect(onScreenChanged);
|
||||
tablet.removeButton(button);
|
||||
|
||||
rpo360Off();
|
||||
|
||||
if (takePhotoControllerMapping) {
|
||||
takePhotoControllerMapping.disable();
|
||||
}
|
||||
|
||||
Window.domainChanged.disconnect(onDomainChanged);
|
||||
Window.snapshot360Taken.disconnect(on360SnapshotTaken);
|
||||
HMD.displayModeChanged.disconnect(onHMDChanged);
|
||||
}
|
||||
|
||||
Script.scriptEnding.connect(cleanup);
|
||||
|
||||
//controller
|
||||
function setTakePhotoControllerMappingStatus() {
|
||||
if (!takePhotoControllerMapping) {
|
||||
return;
|
||||
}
|
||||
if (!isThrowMode) {
|
||||
takePhotoControllerMapping.enable();
|
||||
} else {
|
||||
takePhotoControllerMapping.disable();
|
||||
}
|
||||
}
|
||||
|
||||
var takePhotoControllerMapping;
|
||||
var takePhotoControllerMappingName = 'Overte-cam360-Mapping-Capture';
|
||||
function registerTakePhotoControllerMapping() {
|
||||
takePhotoControllerMapping = Controller.newMapping(takePhotoControllerMappingName);
|
||||
if (controllerType === "OculusTouch") {
|
||||
takePhotoControllerMapping.from(Controller.Standard.RS).to(function (value) {
|
||||
if (value === 1.0) {
|
||||
if (camera) {
|
||||
capture();
|
||||
}
|
||||
}
|
||||
return;
|
||||
});
|
||||
} else if (controllerType === "Vive") {
|
||||
takePhotoControllerMapping.from(Controller.Standard.RightPrimaryThumb).to(function (value) {
|
||||
if (value === 1.0) {
|
||||
if (camera) {
|
||||
capture();
|
||||
}
|
||||
}
|
||||
return;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
var controllerType = "Other";
|
||||
function registerButtonMappings() {
|
||||
var VRDevices = Controller.getDeviceNames().toString();
|
||||
if (VRDevices) {
|
||||
if (VRDevices.indexOf("Vive") !== -1) {
|
||||
controllerType = "Vive";
|
||||
} else if (VRDevices.indexOf("OculusTouch") !== -1) {
|
||||
controllerType = "OculusTouch";
|
||||
} else {
|
||||
return; // Neither Vive nor Touch detected
|
||||
}
|
||||
}
|
||||
|
||||
if (!takePhotoControllerMapping) {
|
||||
registerTakePhotoControllerMapping();
|
||||
}
|
||||
}
|
||||
|
||||
function onHMDChanged(isHMDMode) {
|
||||
registerButtonMappings();
|
||||
}
|
||||
|
||||
}());
|
50
applications/cam360/grabDetection.js
Normal file
50
applications/cam360/grabDetection.js
Normal file
|
@ -0,0 +1,50 @@
|
|||
"use strict";
|
||||
//
|
||||
// grabDetection.js
|
||||
//
|
||||
// Created by Alezia Kurdis on August 26th, 2022
|
||||
// Copyright 2022 Overte e.V.
|
||||
//
|
||||
// Distributed under the Apache License, Version 2.0
|
||||
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
||||
//
|
||||
(function(){
|
||||
|
||||
var _this;
|
||||
|
||||
DetectGrabbed = function() {
|
||||
_this = this;
|
||||
};
|
||||
|
||||
DetectGrabbed.prototype = {
|
||||
setRightHand: function () {
|
||||
//print("I am being held in a right hand... entity:" + this.entityID);
|
||||
},
|
||||
setLeftHand: function () {
|
||||
//print("I am being held in a left hand... entity:" + this.entityID);
|
||||
},
|
||||
startDistantGrab: function () {
|
||||
//print("I am being distance held... entity:" + this.entityID);
|
||||
},
|
||||
continueDistantGrab: function () {
|
||||
//print("I continue to be distance held... entity:" + this.entityID);
|
||||
},
|
||||
startNearGrab: function () {
|
||||
//print("I was just grabbed... entity:" + this.entityID);
|
||||
},
|
||||
continueNearGrab: function () {
|
||||
//print("I am still being grabbed... entity:" + this.entityID);
|
||||
},
|
||||
|
||||
releaseGrab: function () {
|
||||
//print("I was released... entity:" + this.entityID);
|
||||
Entities.editEntity(this.entityID, {"userData": "RELEASED"});
|
||||
},
|
||||
|
||||
preload: function(entityID) {
|
||||
this.entityID = entityID;
|
||||
},
|
||||
};
|
||||
|
||||
return new DetectGrabbed();
|
||||
})
|
BIN
applications/cam360/resources/fonts/FiraSans-SemiBold.ttf
Normal file
BIN
applications/cam360/resources/fonts/FiraSans-SemiBold.ttf
Normal file
Binary file not shown.
BIN
applications/cam360/resources/images/default.jpg
Normal file
BIN
applications/cam360/resources/images/default.jpg
Normal file
Binary file not shown.
After Width: | Height: | Size: 444 KiB |
58
applications/cam360/resources/images/icons/cam360-a.svg
Normal file
58
applications/cam360/resources/images/icons/cam360-a.svg
Normal file
|
@ -0,0 +1,58 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<svg
|
||||
version="1.0"
|
||||
width="66.666664"
|
||||
height="66.666664"
|
||||
viewBox="0 0 49.999998 49.999998"
|
||||
preserveAspectRatio="xMidYMid meet"
|
||||
id="svg986"
|
||||
sodipodi:docname="cam360-a.svg"
|
||||
inkscape:version="1.1.1 (3bf5ae0d25, 2021-09-20)"
|
||||
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg">
|
||||
<defs
|
||||
id="defs990" />
|
||||
<sodipodi:namedview
|
||||
id="namedview988"
|
||||
pagecolor="#ffffff"
|
||||
bordercolor="#666666"
|
||||
borderopacity="1.0"
|
||||
inkscape:pageshadow="2"
|
||||
inkscape:pageopacity="0.0"
|
||||
inkscape:pagecheckerboard="0"
|
||||
inkscape:document-units="px"
|
||||
showgrid="false"
|
||||
units="px"
|
||||
inkscape:zoom="1.9673913"
|
||||
inkscape:cx="183.74586"
|
||||
inkscape:cy="184"
|
||||
inkscape:window-width="1628"
|
||||
inkscape:window-height="932"
|
||||
inkscape:window-x="1709"
|
||||
inkscape:window-y="9"
|
||||
inkscape:window-maximized="0"
|
||||
inkscape:current-layer="svg986" />
|
||||
<g
|
||||
transform="matrix(0.02008251,0,0,-0.02008251,-2.5674559,53.27867)"
|
||||
fill="#000000"
|
||||
stroke="none"
|
||||
id="g984">
|
||||
<path
|
||||
d="m 1265,2392 -40,-7 -3,-52 -3,-53 h 160 161 v 49 c 0,28 -5,53 -10,56 -15,9 -219,15 -265,7 z"
|
||||
id="path974" />
|
||||
<path
|
||||
d="m 1109,2346 c -71,-30 -157,-85 -214,-137 l -44,-41 -46,27 c -25,15 -47,26 -49,24 -13,-15 -96,-151 -96,-158 0,-4 17,-20 39,-35 l 38,-27 -30,-87 c -29,-81 -31,-97 -31,-222 0,-120 3,-143 26,-210 l 27,-75 -35,-30 -35,-30 56,-77 c 31,-43 59,-78 63,-78 4,0 18,9 32,19 l 24,19 58,-54 c 75,-68 171,-125 273,-160 72,-25 94,-28 215,-28 121,0 143,3 215,28 105,36 200,93 281,168 l 65,61 30,-22 c 17,-13 32,-21 33,-19 18,21 108,170 105,172 -2,2 -19,11 -36,20 -18,9 -33,20 -33,24 0,4 9,36 20,71 45,144 36,320 -25,471 l -35,85 23,19 c 12,10 30,24 40,31 17,12 15,18 -33,86 -29,41 -56,77 -59,82 -4,4 -25,-6 -47,-23 -21,-16 -42,-30 -46,-30 -5,0 -35,20 -69,45 -57,42 -155,94 -206,110 -22,6 -23,4 -23,-59 v -66 h -200 -200 v 65 c 0,36 -3,65 -7,65 -5,-1 -33,-11 -64,-24 z m 386,-230 c 80,-21 146,-60 201,-117 162,-172 178,-377 44,-568 -46,-66 -80,-95 -150,-131 -154,-79 -335,-61 -476,47 -47,35 -114,136 -140,211 -111,314 198,645 521,558 z"
|
||||
id="path976" />
|
||||
<path
|
||||
d="m 1295,2017 c -97,-33 -172,-98 -214,-187 -37,-79 -37,-201 0,-280 33,-70 99,-136 169,-169 46,-22 69,-26 140,-26 71,0 94,4 140,26 70,33 136,99 169,169 37,79 37,201 0,280 -33,70 -99,136 -168,168 -66,30 -175,39 -236,19 z m 92,-117 c -56,-35 -119,-127 -138,-200 l -13,-50 h -43 -43 v 50 c 0,129 101,229 229,230 h 55 z"
|
||||
id="path978" />
|
||||
<path
|
||||
d="m 2180,1510 c 0,-5 4,-10 9,-10 4,0 47,-36 94,-81 186,-174 202,-328 50,-484 -108,-111 -414,-235 -640,-260 -110,-12 -103,-5 -103,-95 v -80 h 31 c 47,0 265,46 361,76 251,79 430,211 524,387 36,66 39,78 39,152 0,65 -5,89 -24,125 -51,97 -134,173 -264,245 -68,38 -77,40 -77,25 z"
|
||||
id="path980" />
|
||||
<path
|
||||
d="m 513,1481 c -117,-65 -196,-137 -249,-227 -26,-44 -29,-58 -29,-139 0,-82 3,-96 33,-152 C 375,756 635,598 983,528 l 107,-21 v -71 c 0,-88 9,-105 47,-90 42,16 292,193 303,215 7,12 7,25 1,37 -5,10 -78,64 -162,121 -176,120 -182,121 -187,22 -4,-73 0,-72 -147,-37 -196,48 -371,125 -480,214 -75,60 -125,149 -125,223 0,100 70,219 182,306 53,42 70,63 51,63 -5,0 -32,-13 -60,-29 z"
|
||||
id="path982" />
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 3.4 KiB |
64
applications/cam360/resources/images/icons/cam360-i.svg
Normal file
64
applications/cam360/resources/images/icons/cam360-i.svg
Normal file
|
@ -0,0 +1,64 @@
|
|||
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
|
||||
<svg
|
||||
version="1.0"
|
||||
width="66.666664"
|
||||
height="66.666664"
|
||||
viewBox="0 0 49.999998 49.999998"
|
||||
preserveAspectRatio="xMidYMid meet"
|
||||
id="svg986"
|
||||
sodipodi:docname="cam360-i.svg"
|
||||
inkscape:version="1.1.1 (3bf5ae0d25, 2021-09-20)"
|
||||
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
|
||||
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
xmlns:svg="http://www.w3.org/2000/svg">
|
||||
<defs
|
||||
id="defs990" />
|
||||
<sodipodi:namedview
|
||||
id="namedview988"
|
||||
pagecolor="#ffffff"
|
||||
bordercolor="#666666"
|
||||
borderopacity="1.0"
|
||||
inkscape:pageshadow="2"
|
||||
inkscape:pageopacity="0.0"
|
||||
inkscape:pagecheckerboard="0"
|
||||
inkscape:document-units="px"
|
||||
showgrid="false"
|
||||
units="px"
|
||||
inkscape:zoom="1.9673913"
|
||||
inkscape:cx="183.74586"
|
||||
inkscape:cy="184"
|
||||
inkscape:window-width="1628"
|
||||
inkscape:window-height="932"
|
||||
inkscape:window-x="1709"
|
||||
inkscape:window-y="9"
|
||||
inkscape:window-maximized="0"
|
||||
inkscape:current-layer="svg986" />
|
||||
<g
|
||||
transform="matrix(0.02008251,0,0,-0.02008251,-2.5674559,53.27867)"
|
||||
fill="#000000"
|
||||
stroke="none"
|
||||
id="g984"
|
||||
style="fill:#ffffff">
|
||||
<path
|
||||
d="m 1265,2392 -40,-7 -3,-52 -3,-53 h 160 161 v 49 c 0,28 -5,53 -10,56 -15,9 -219,15 -265,7 z"
|
||||
id="path974"
|
||||
style="fill:#ffffff" />
|
||||
<path
|
||||
d="m 1109,2346 c -71,-30 -157,-85 -214,-137 l -44,-41 -46,27 c -25,15 -47,26 -49,24 -13,-15 -96,-151 -96,-158 0,-4 17,-20 39,-35 l 38,-27 -30,-87 c -29,-81 -31,-97 -31,-222 0,-120 3,-143 26,-210 l 27,-75 -35,-30 -35,-30 56,-77 c 31,-43 59,-78 63,-78 4,0 18,9 32,19 l 24,19 58,-54 c 75,-68 171,-125 273,-160 72,-25 94,-28 215,-28 121,0 143,3 215,28 105,36 200,93 281,168 l 65,61 30,-22 c 17,-13 32,-21 33,-19 18,21 108,170 105,172 -2,2 -19,11 -36,20 -18,9 -33,20 -33,24 0,4 9,36 20,71 45,144 36,320 -25,471 l -35,85 23,19 c 12,10 30,24 40,31 17,12 15,18 -33,86 -29,41 -56,77 -59,82 -4,4 -25,-6 -47,-23 -21,-16 -42,-30 -46,-30 -5,0 -35,20 -69,45 -57,42 -155,94 -206,110 -22,6 -23,4 -23,-59 v -66 h -200 -200 v 65 c 0,36 -3,65 -7,65 -5,-1 -33,-11 -64,-24 z m 386,-230 c 80,-21 146,-60 201,-117 162,-172 178,-377 44,-568 -46,-66 -80,-95 -150,-131 -154,-79 -335,-61 -476,47 -47,35 -114,136 -140,211 -111,314 198,645 521,558 z"
|
||||
id="path976"
|
||||
style="fill:#ffffff" />
|
||||
<path
|
||||
d="m 1295,2017 c -97,-33 -172,-98 -214,-187 -37,-79 -37,-201 0,-280 33,-70 99,-136 169,-169 46,-22 69,-26 140,-26 71,0 94,4 140,26 70,33 136,99 169,169 37,79 37,201 0,280 -33,70 -99,136 -168,168 -66,30 -175,39 -236,19 z m 92,-117 c -56,-35 -119,-127 -138,-200 l -13,-50 h -43 -43 v 50 c 0,129 101,229 229,230 h 55 z"
|
||||
id="path978"
|
||||
style="fill:#ffffff" />
|
||||
<path
|
||||
d="m 2180,1510 c 0,-5 4,-10 9,-10 4,0 47,-36 94,-81 186,-174 202,-328 50,-484 -108,-111 -414,-235 -640,-260 -110,-12 -103,-5 -103,-95 v -80 h 31 c 47,0 265,46 361,76 251,79 430,211 524,387 36,66 39,78 39,152 0,65 -5,89 -24,125 -51,97 -134,173 -264,245 -68,38 -77,40 -77,25 z"
|
||||
id="path980"
|
||||
style="fill:#ffffff" />
|
||||
<path
|
||||
d="m 513,1481 c -117,-65 -196,-137 -249,-227 -26,-44 -29,-58 -29,-139 0,-82 3,-96 33,-152 C 375,756 635,598 983,528 l 107,-21 v -71 c 0,-88 9,-105 47,-90 42,16 292,193 303,215 7,12 7,25 1,37 -5,10 -78,64 -162,121 -176,120 -182,121 -187,22 -4,-73 0,-72 -147,-37 -196,48 -371,125 -480,214 -75,60 -125,149 -125,223 0,100 70,219 182,306 53,42 70,63 51,63 -5,0 -32,-13 -60,-29 z"
|
||||
id="path982"
|
||||
style="fill:#ffffff" />
|
||||
</g>
|
||||
</svg>
|
After Width: | Height: | Size: 3.5 KiB |
BIN
applications/cam360/resources/images/off.png
Normal file
BIN
applications/cam360/resources/images/off.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.3 KiB |
BIN
applications/cam360/resources/images/on.png
Normal file
BIN
applications/cam360/resources/images/on.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 2.5 KiB |
BIN
applications/cam360/resources/images/processing.gif
Normal file
BIN
applications/cam360/resources/images/processing.gif
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.1 KiB |
202
applications/cam360/resources/marzipano/LICENSE.txt
Normal file
202
applications/cam360/resources/marzipano/LICENSE.txt
Normal file
|
@ -0,0 +1,202 @@
|
|||
|
||||
Apache License
|
||||
Version 2.0, January 2004
|
||||
http://www.apache.org/licenses/
|
||||
|
||||
TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
|
||||
|
||||
1. Definitions.
|
||||
|
||||
"License" shall mean the terms and conditions for use, reproduction,
|
||||
and distribution as defined by Sections 1 through 9 of this document.
|
||||
|
||||
"Licensor" shall mean the copyright owner or entity authorized by
|
||||
the copyright owner that is granting the License.
|
||||
|
||||
"Legal Entity" shall mean the union of the acting entity and all
|
||||
other entities that control, are controlled by, or are under common
|
||||
control with that entity. For the purposes of this definition,
|
||||
"control" means (i) the power, direct or indirect, to cause the
|
||||
direction or management of such entity, whether by contract or
|
||||
otherwise, or (ii) ownership of fifty percent (50%) or more of the
|
||||
outstanding shares, or (iii) beneficial ownership of such entity.
|
||||
|
||||
"You" (or "Your") shall mean an individual or Legal Entity
|
||||
exercising permissions granted by this License.
|
||||
|
||||
"Source" form shall mean the preferred form for making modifications,
|
||||
including but not limited to software source code, documentation
|
||||
source, and configuration files.
|
||||
|
||||
"Object" form shall mean any form resulting from mechanical
|
||||
transformation or translation of a Source form, including but
|
||||
not limited to compiled object code, generated documentation,
|
||||
and conversions to other media types.
|
||||
|
||||
"Work" shall mean the work of authorship, whether in Source or
|
||||
Object form, made available under the License, as indicated by a
|
||||
copyright notice that is included in or attached to the work
|
||||
(an example is provided in the Appendix below).
|
||||
|
||||
"Derivative Works" shall mean any work, whether in Source or Object
|
||||
form, that is based on (or derived from) the Work and for which the
|
||||
editorial revisions, annotations, elaborations, or other modifications
|
||||
represent, as a whole, an original work of authorship. For the purposes
|
||||
of this License, Derivative Works shall not include works that remain
|
||||
separable from, or merely link (or bind by name) to the interfaces of,
|
||||
the Work and Derivative Works thereof.
|
||||
|
||||
"Contribution" shall mean any work of authorship, including
|
||||
the original version of the Work and any modifications or additions
|
||||
to that Work or Derivative Works thereof, that is intentionally
|
||||
submitted to Licensor for inclusion in the Work by the copyright owner
|
||||
or by an individual or Legal Entity authorized to submit on behalf of
|
||||
the copyright owner. For the purposes of this definition, "submitted"
|
||||
means any form of electronic, verbal, or written communication sent
|
||||
to the Licensor or its representatives, including but not limited to
|
||||
communication on electronic mailing lists, source code control systems,
|
||||
and issue tracking systems that are managed by, or on behalf of, the
|
||||
Licensor for the purpose of discussing and improving the Work, but
|
||||
excluding communication that is conspicuously marked or otherwise
|
||||
designated in writing by the copyright owner as "Not a Contribution."
|
||||
|
||||
"Contributor" shall mean Licensor and any individual or Legal Entity
|
||||
on behalf of whom a Contribution has been received by Licensor and
|
||||
subsequently incorporated within the Work.
|
||||
|
||||
2. Grant of Copyright License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
copyright license to reproduce, prepare Derivative Works of,
|
||||
publicly display, publicly perform, sublicense, and distribute the
|
||||
Work and such Derivative Works in Source or Object form.
|
||||
|
||||
3. Grant of Patent License. Subject to the terms and conditions of
|
||||
this License, each Contributor hereby grants to You a perpetual,
|
||||
worldwide, non-exclusive, no-charge, royalty-free, irrevocable
|
||||
(except as stated in this section) patent license to make, have made,
|
||||
use, offer to sell, sell, import, and otherwise transfer the Work,
|
||||
where such license applies only to those patent claims licensable
|
||||
by such Contributor that are necessarily infringed by their
|
||||
Contribution(s) alone or by combination of their Contribution(s)
|
||||
with the Work to which such Contribution(s) was submitted. If You
|
||||
institute patent litigation against any entity (including a
|
||||
cross-claim or counterclaim in a lawsuit) alleging that the Work
|
||||
or a Contribution incorporated within the Work constitutes direct
|
||||
or contributory patent infringement, then any patent licenses
|
||||
granted to You under this License for that Work shall terminate
|
||||
as of the date such litigation is filed.
|
||||
|
||||
4. Redistribution. You may reproduce and distribute copies of the
|
||||
Work or Derivative Works thereof in any medium, with or without
|
||||
modifications, and in Source or Object form, provided that You
|
||||
meet the following conditions:
|
||||
|
||||
(a) You must give any other recipients of the Work or
|
||||
Derivative Works a copy of this License; and
|
||||
|
||||
(b) You must cause any modified files to carry prominent notices
|
||||
stating that You changed the files; and
|
||||
|
||||
(c) You must retain, in the Source form of any Derivative Works
|
||||
that You distribute, all copyright, patent, trademark, and
|
||||
attribution notices from the Source form of the Work,
|
||||
excluding those notices that do not pertain to any part of
|
||||
the Derivative Works; and
|
||||
|
||||
(d) If the Work includes a "NOTICE" text file as part of its
|
||||
distribution, then any Derivative Works that You distribute must
|
||||
include a readable copy of the attribution notices contained
|
||||
within such NOTICE file, excluding those notices that do not
|
||||
pertain to any part of the Derivative Works, in at least one
|
||||
of the following places: within a NOTICE text file distributed
|
||||
as part of the Derivative Works; within the Source form or
|
||||
documentation, if provided along with the Derivative Works; or,
|
||||
within a display generated by the Derivative Works, if and
|
||||
wherever such third-party notices normally appear. The contents
|
||||
of the NOTICE file are for informational purposes only and
|
||||
do not modify the License. You may add Your own attribution
|
||||
notices within Derivative Works that You distribute, alongside
|
||||
or as an addendum to the NOTICE text from the Work, provided
|
||||
that such additional attribution notices cannot be construed
|
||||
as modifying the License.
|
||||
|
||||
You may add Your own copyright statement to Your modifications and
|
||||
may provide additional or different license terms and conditions
|
||||
for use, reproduction, or distribution of Your modifications, or
|
||||
for any such Derivative Works as a whole, provided Your use,
|
||||
reproduction, and distribution of the Work otherwise complies with
|
||||
the conditions stated in this License.
|
||||
|
||||
5. Submission of Contributions. Unless You explicitly state otherwise,
|
||||
any Contribution intentionally submitted for inclusion in the Work
|
||||
by You to the Licensor shall be under the terms and conditions of
|
||||
this License, without any additional terms or conditions.
|
||||
Notwithstanding the above, nothing herein shall supersede or modify
|
||||
the terms of any separate license agreement you may have executed
|
||||
with Licensor regarding such Contributions.
|
||||
|
||||
6. Trademarks. This License does not grant permission to use the trade
|
||||
names, trademarks, service marks, or product names of the Licensor,
|
||||
except as required for reasonable and customary use in describing the
|
||||
origin of the Work and reproducing the content of the NOTICE file.
|
||||
|
||||
7. Disclaimer of Warranty. Unless required by applicable law or
|
||||
agreed to in writing, Licensor provides the Work (and each
|
||||
Contributor provides its Contributions) on an "AS IS" BASIS,
|
||||
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
|
||||
implied, including, without limitation, any warranties or conditions
|
||||
of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
|
||||
PARTICULAR PURPOSE. You are solely responsible for determining the
|
||||
appropriateness of using or redistributing the Work and assume any
|
||||
risks associated with Your exercise of permissions under this License.
|
||||
|
||||
8. Limitation of Liability. In no event and under no legal theory,
|
||||
whether in tort (including negligence), contract, or otherwise,
|
||||
unless required by applicable law (such as deliberate and grossly
|
||||
negligent acts) or agreed to in writing, shall any Contributor be
|
||||
liable to You for damages, including any direct, indirect, special,
|
||||
incidental, or consequential damages of any character arising as a
|
||||
result of this License or out of the use or inability to use the
|
||||
Work (including but not limited to damages for loss of goodwill,
|
||||
work stoppage, computer failure or malfunction, or any and all
|
||||
other commercial damages or losses), even if such Contributor
|
||||
has been advised of the possibility of such damages.
|
||||
|
||||
9. Accepting Warranty or Additional Liability. While redistributing
|
||||
the Work or Derivative Works thereof, You may choose to offer,
|
||||
and charge a fee for, acceptance of support, warranty, indemnity,
|
||||
or other liability obligations and/or rights consistent with this
|
||||
License. However, in accepting such obligations, You may act only
|
||||
on Your own behalf and on Your sole responsibility, not on behalf
|
||||
of any other Contributor, and only if You agree to indemnify,
|
||||
defend, and hold each Contributor harmless for any liability
|
||||
incurred by, or claims asserted against, such Contributor by reason
|
||||
of your accepting any such warranty or additional liability.
|
||||
|
||||
END OF TERMS AND CONDITIONS
|
||||
|
||||
APPENDIX: How to apply the Apache License to your work.
|
||||
|
||||
To apply the Apache License to your work, attach the following
|
||||
boilerplate notice, with the fields enclosed by brackets "[]"
|
||||
replaced with your own identifying information. (Don't include
|
||||
the brackets!) The text should be enclosed in the appropriate
|
||||
comment syntax for the file format. We also recommend that a
|
||||
file or class name and description of purpose be included on the
|
||||
same "printed page" as the copyright notice for easier
|
||||
identification within third-party archives.
|
||||
|
||||
Copyright [yyyy] [name of copyright owner]
|
||||
|
||||
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.
|
16
applications/cam360/resources/marzipano/marzipano.js
Normal file
16
applications/cam360/resources/marzipano/marzipano.js
Normal file
File diff suppressed because one or more lines are too long
17
applications/cam360/resources/models/black.json
Normal file
17
applications/cam360/resources/models/black.json
Normal file
|
@ -0,0 +1,17 @@
|
|||
{
|
||||
"materialVersion": 1,
|
||||
"materials": [
|
||||
{
|
||||
"name": "ac:None1",
|
||||
"albedo": [
|
||||
0.10196078431372549,
|
||||
0.10196078431372549,
|
||||
0.10196078431372549
|
||||
],
|
||||
"metallic": 0.006,
|
||||
"roughness": 0.386,
|
||||
"cullFaceMode": "CULL_BACK",
|
||||
"defaultFallthrough": true
|
||||
}
|
||||
]
|
||||
}
|
BIN
applications/cam360/resources/models/cam360.fbx
Normal file
BIN
applications/cam360/resources/models/cam360.fbx
Normal file
Binary file not shown.
3
applications/cam360/resources/models/cam360black.fst
Normal file
3
applications/cam360/resources/models/cam360black.fst
Normal file
|
@ -0,0 +1,3 @@
|
|||
name = CAM360_BLACK
|
||||
filename = cam360.fbx
|
||||
materialMap = [{"mat::SHELL":"black.json#SHELL"}, {"mat::LIGHT":"greenLight.json#LIGHT"}, {"mat::GLASS":"glass.json#GLASS"}]
|
3
applications/cam360/resources/models/cam360white.fst
Normal file
3
applications/cam360/resources/models/cam360white.fst
Normal file
|
@ -0,0 +1,3 @@
|
|||
name = CAM360_WHITE
|
||||
filename = cam360.fbx
|
||||
materialMap = [{"mat::LIGHT":"redLight.json#LIGHT"}, {"mat::GLASS":"glass.json#GLASS"}]
|
16
applications/cam360/resources/models/glass.json
Normal file
16
applications/cam360/resources/models/glass.json
Normal file
|
@ -0,0 +1,16 @@
|
|||
{
|
||||
"materialVersion": 1,
|
||||
"materials": [
|
||||
{
|
||||
"albedo": [
|
||||
0.001,
|
||||
0.001,
|
||||
0.001
|
||||
],
|
||||
"metallic": 1,
|
||||
"roughness": 0.036,
|
||||
"opacity": 0.205,
|
||||
"cullFaceMode": "CULL_BACK"
|
||||
}
|
||||
]
|
||||
}
|
21
applications/cam360/resources/models/greenLight.json
Normal file
21
applications/cam360/resources/models/greenLight.json
Normal file
|
@ -0,0 +1,21 @@
|
|||
{
|
||||
"materialVersion": 1,
|
||||
"materials": [
|
||||
{
|
||||
"name": "Material.001",
|
||||
"albedo": [
|
||||
1,
|
||||
1,
|
||||
1
|
||||
],
|
||||
"metallic": 0.001,
|
||||
"roughness": 1,
|
||||
"emissive": [
|
||||
0.16235294117647056,
|
||||
2.3,
|
||||
0.4149019607843137
|
||||
],
|
||||
"cullFaceMode": "CULL_BACK"
|
||||
}
|
||||
]
|
||||
}
|
21
applications/cam360/resources/models/redLight.json
Normal file
21
applications/cam360/resources/models/redLight.json
Normal file
|
@ -0,0 +1,21 @@
|
|||
{
|
||||
"materialVersion": 1,
|
||||
"materials": [
|
||||
{
|
||||
"name": "Material.001",
|
||||
"albedo": [
|
||||
1,
|
||||
1,
|
||||
1
|
||||
],
|
||||
"metallic": 0.001,
|
||||
"roughness": 1,
|
||||
"emissive": [
|
||||
2.18,
|
||||
0.9061960784313726,
|
||||
0
|
||||
],
|
||||
"cullFaceMode": "CULL_BACK"
|
||||
}
|
||||
]
|
||||
}
|
BIN
applications/cam360/resources/sounds/cameraOn.wav
Normal file
BIN
applications/cam360/resources/sounds/cameraOn.wav
Normal file
Binary file not shown.
BIN
applications/cam360/resources/sounds/snap.wav
Normal file
BIN
applications/cam360/resources/sounds/snap.wav
Normal file
Binary file not shown.
|
@ -152,6 +152,15 @@ var metadata = { "applications":
|
|||
"jsfile": "home/app_home.js",
|
||||
"icon": "home/appicon_i.png",
|
||||
"caption": "HOME"
|
||||
},
|
||||
{
|
||||
"isActive": true,
|
||||
"directory": "cam360",
|
||||
"name": "Camera 360",
|
||||
"description": "This camera takes spherical snapshots. It has two capture modes:<br>- The 'Throw' mode: Rez the cam, grab it, then toss it high in the sky to capture an incredible view of of any event.<br>- The 'Position' mode: Position then click to capture. Ideal to generate an ambient light texture for any scene.<br>This includes a 360° media viewer.",
|
||||
"jsfile": "cam360/cam360.js",
|
||||
"icon": "cam360/resources/images/icons/cam360-i.svg",
|
||||
"caption": "CAM360"
|
||||
}
|
||||
]
|
||||
};
|
||||
|
|
Loading…
Reference in a new issue