mirror of
https://thingvellir.net/git/overte
synced 2025-03-27 23:52:03 +01:00
Merge pull request #1296 from AleziaKurdis/placeApp_rezTP
Places app: Portal Spawner and Copy URL
This commit is contained in:
commit
ccdac96b9d
9 changed files with 377 additions and 17 deletions
|
@ -2,9 +2,9 @@
|
||||||
// EntityTypes.h
|
// EntityTypes.h
|
||||||
// libraries/entities/src
|
// libraries/entities/src
|
||||||
//
|
//
|
||||||
// Created by Brad Hefta-Gaub on 12/4/13.
|
// Created by Brad Hefta-Gaub on December 4th, 2013.
|
||||||
// Copyright 2013 High Fidelity, Inc.
|
// Copyright 2013 High Fidelity, Inc.
|
||||||
// Copyright 2023 Overte e.V.
|
// Copyright 2023-2025 Overte e.V.
|
||||||
//
|
//
|
||||||
// Distributed under the Apache License, 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
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
||||||
|
@ -91,7 +91,7 @@ public:
|
||||||
* <tr><td><code>"Material"</code></td><td>Modifies the existing materials on entities and avatars.</td>
|
* <tr><td><code>"Material"</code></td><td>Modifies the existing materials on entities and avatars.</td>
|
||||||
* <td>{@link Entities.EntityProperties-Material|EntityProperties-Material}</td></tr>
|
* <td>{@link Entities.EntityProperties-Material|EntityProperties-Material}</td></tr>
|
||||||
* <tr><td><code>"Sound"</code></td><td>Plays a sound.</td>
|
* <tr><td><code>"Sound"</code></td><td>Plays a sound.</td>
|
||||||
* <td>{@link Entities.EntityProperties-Material|EntityProperties-Sound}</td></tr>
|
* <td>{@link Entities.EntityProperties-Sound|EntityProperties-Sound}</td></tr>
|
||||||
* </tbody>
|
* </tbody>
|
||||||
* </table>
|
* </table>
|
||||||
* @typedef {string} Entities.EntityType
|
* @typedef {string} Entities.EntityType
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
//
|
//
|
||||||
// Created by Sam Gondelman on 12/7/18.
|
// Created by Sam Gondelman on December 7th, 2018.
|
||||||
// Copyright 2018 High Fidelity, Inc.
|
// Copyright 2018 High Fidelity, Inc.
|
||||||
|
// Copyright 2025 Overte e.V.
|
||||||
//
|
//
|
||||||
// Distributed under the Apache License, 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
|
// See the accompanying file LICENSE or http://www.apache.org/licenses/LICENSE-2.0.html
|
||||||
|
@ -26,7 +27,7 @@ public:
|
||||||
* <tr><td>PICK_DOMAIN_ENTITIES</td><td><code>1</code></td><td>Include domain entities when intersecting.</td></tr>
|
* <tr><td>PICK_DOMAIN_ENTITIES</td><td><code>1</code></td><td>Include domain entities when intersecting.</td></tr>
|
||||||
* <tr><td>PICK_AVATAR_ENTITIES</td><td><code>2</code></td><td>Include avatar entities when intersecting.</td></tr>
|
* <tr><td>PICK_AVATAR_ENTITIES</td><td><code>2</code></td><td>Include avatar entities when intersecting.</td></tr>
|
||||||
* <tr><td>PICK_LOCAL_ENTITIES</td><td><code>4</code></td><td>Include local entities when intersecting.</td></tr>
|
* <tr><td>PICK_LOCAL_ENTITIES</td><td><code>4</code></td><td>Include local entities when intersecting.</td></tr>
|
||||||
* <tr><td>PICK_AVATATRS</td><td><code>8</code></td><td>Include avatars when intersecting.</td></tr>
|
* <tr><td>PICK_AVATARS</td><td><code>8</code></td><td>Include avatars when intersecting.</td></tr>
|
||||||
* <tr><td>PICK_HUD</td><td><code>16</code></td><td>Include the HUD surface when intersecting in HMD mode.</td></tr>
|
* <tr><td>PICK_HUD</td><td><code>16</code></td><td>Include the HUD surface when intersecting in HMD mode.</td></tr>
|
||||||
* <tr><td>PICK_INCLUDE_VISIBLE</td><td><code>32</code></td><td>Include visible objects when intersecting.</td></tr>
|
* <tr><td>PICK_INCLUDE_VISIBLE</td><td><code>32</code></td><td>Include visible objects when intersecting.</td></tr>
|
||||||
* <tr><td>PICK_INCLUDE_INVISIBLE</td><td><code>64</code></td><td>Include invisible objects when intersecting.</td></tr>
|
* <tr><td>PICK_INCLUDE_INVISIBLE</td><td><code>64</code></td><td>Include invisible objects when intersecting.</td></tr>
|
||||||
|
|
BIN
scripts/system/places/icons/portalFX.png
Normal file
BIN
scripts/system/places/icons/portalFX.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 78 KiB |
|
@ -3,7 +3,7 @@
|
||||||
// places.css
|
// places.css
|
||||||
//
|
//
|
||||||
// Created by Alezia Kurdis, January 1st, 2022.
|
// Created by Alezia Kurdis, January 1st, 2022.
|
||||||
// Copyright 2022 Overte e.V.
|
// Copyright 2022-2025 Overte e.V.
|
||||||
//
|
//
|
||||||
// css for the ui of the Places application.
|
// css for the ui of the Places application.
|
||||||
//
|
//
|
||||||
|
@ -750,19 +750,20 @@ font.domain-nbrUser_small {
|
||||||
color: #cccccc;
|
color: #cccccc;
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
text-align: justify;
|
text-align: justify;
|
||||||
text-justify: inter-word;
|
text-justify: inter-word;
|
||||||
}
|
}
|
||||||
|
|
||||||
#placeDetail-visitBtn {
|
#placeDetail-visitBtn {
|
||||||
background: #0000ff;
|
background: #0000ff;
|
||||||
background-image: linear-gradient(to bottom, #0000ff, #000020);
|
background-image: linear-gradient(to bottom, #0000ff, #000020);
|
||||||
border: 0px;
|
border: 0px;
|
||||||
border-radius: 10px;
|
border-radius: 6px;
|
||||||
font-weight: 800;
|
font-weight: 700;
|
||||||
color: #ffffff;
|
color: #ffffff;
|
||||||
font-size: 20px;
|
font-size: 14px;
|
||||||
padding: 3px 22px 3px 22px;
|
padding: 2px 22px 2px 22px;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
|
width: 90%;
|
||||||
}
|
}
|
||||||
|
|
||||||
#placeDetail-visitBtn:hover {
|
#placeDetail-visitBtn:hover {
|
||||||
|
@ -774,7 +775,57 @@ font.domain-nbrUser_small {
|
||||||
#placeDetail-visitBtn-container {
|
#placeDetail-visitBtn-container {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
margin-bottom: 40px;
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-rezPortalBtn {
|
||||||
|
background: #0000ff;
|
||||||
|
background-image: linear-gradient(to bottom, #0000ff, #000020);
|
||||||
|
border: 0px;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #ffffff;
|
||||||
|
font-size: 14px;
|
||||||
|
padding: 2px 22px 2px 22px;
|
||||||
|
text-decoration: none;
|
||||||
|
width: 90%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-rezPortalBtn:hover {
|
||||||
|
background: #057eff;
|
||||||
|
background-image: linear-gradient(to bottom, #057eff, #00090f);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-rezPortalBtn-container {
|
||||||
|
width: 100%;
|
||||||
|
text-align: left;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-copyPlaceURLBtn {
|
||||||
|
background: #0000ff;
|
||||||
|
background-image: linear-gradient(to bottom, #0000ff, #000020);
|
||||||
|
border: 0px;
|
||||||
|
border-radius: 6px;
|
||||||
|
font-weight: 700;
|
||||||
|
color: #ffffff;
|
||||||
|
font-size: 14px;
|
||||||
|
padding: 2px 22px 2px 22px;
|
||||||
|
text-decoration: none;
|
||||||
|
width: 90%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-copyPlaceURLBtn:hover {
|
||||||
|
background: #057eff;
|
||||||
|
background-image: linear-gradient(to bottom, #057eff, #00090f);
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#placeDetail-copyPlaceURLBtn-container {
|
||||||
|
width: 100%;
|
||||||
|
text-align: left;
|
||||||
|
margin-bottom: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
#placeDetail-placedata {
|
#placeDetail-placedata {
|
||||||
|
@ -804,7 +855,7 @@ font.domain-nbrUser_small {
|
||||||
|
|
||||||
#placeDetail-users {
|
#placeDetail-users {
|
||||||
font-size: 30px;
|
font-size: 30px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
||||||
|
|
||||||
#placeDetail-capacity {
|
#placeDetail-capacity {
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
// places.html
|
// places.html
|
||||||
//
|
//
|
||||||
// Created by Alezia Kurdis, January 1st, 2022.
|
// Created by Alezia Kurdis, January 1st, 2022.
|
||||||
// Copyright 2022 Overte e.V.
|
// Copyright 2022-2025 Overte e.V.
|
||||||
//
|
//
|
||||||
// html for the ui of the Places application.
|
// html for the ui of the Places application.
|
||||||
//
|
//
|
||||||
|
@ -118,6 +118,8 @@
|
||||||
<td id="" width='29%'>
|
<td id="" width='29%'>
|
||||||
|
|
||||||
<div id="placeDetail-visitBtn-container"></div>
|
<div id="placeDetail-visitBtn-container"></div>
|
||||||
|
<div id="placeDetail-rezPortalBtn-container"></div>
|
||||||
|
<div id="placeDetail-copyPlaceURLBtn-container"></div>
|
||||||
<div id="placeDetail-maturity"></div>
|
<div id="placeDetail-maturity"></div>
|
||||||
<div id="placeDetail-placedata">
|
<div id="placeDetail-placedata">
|
||||||
DOMAIN:
|
DOMAIN:
|
||||||
|
@ -502,6 +504,28 @@
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function rezPortal(name, address, placeID) {
|
||||||
|
var portalOrder = {
|
||||||
|
"channel": channel,
|
||||||
|
"action": "REQUEST_PORTAL",
|
||||||
|
"name": name,
|
||||||
|
"address": address,
|
||||||
|
"placeID": placeID
|
||||||
|
};
|
||||||
|
EventBridge.emitWebEvent(JSON.stringify(portalOrder));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
function copyPlaceURL(address) {
|
||||||
|
var portalOrder = {
|
||||||
|
"channel": channel,
|
||||||
|
"action": "COPY_URL",
|
||||||
|
"address": address
|
||||||
|
};
|
||||||
|
EventBridge.emitWebEvent(JSON.stringify(portalOrder));
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
function goHome() {
|
function goHome() {
|
||||||
var message = {
|
var message = {
|
||||||
"channel": channel,
|
"channel": channel,
|
||||||
|
@ -766,6 +790,8 @@
|
||||||
placeUrl = "hifi://" + placeDetail.address;
|
placeUrl = "hifi://" + placeDetail.address;
|
||||||
}
|
}
|
||||||
document.getElementById("placeDetail-visitBtn-container").innerHTML = "<button id='placeDetail-visitBtn' onclick='teleport(" + '"' + placeDetail.id + '"' + ", " + '"' + placeUrl + '"' + "); return false;'>Visit</button>";
|
document.getElementById("placeDetail-visitBtn-container").innerHTML = "<button id='placeDetail-visitBtn' onclick='teleport(" + '"' + placeDetail.id + '"' + ", " + '"' + placeUrl + '"' + "); return false;'>Visit</button>";
|
||||||
|
document.getElementById("placeDetail-rezPortalBtn-container").innerHTML = "<button id='placeDetail-rezPortalBtn' onclick='rezPortal(" + '"' + placeDetail.name + '"' + ", " + '"' + placeUrl + '"' + ", " + '"' + placeDetail.id + '"' + "); return false;'>Portal</button>";
|
||||||
|
document.getElementById("placeDetail-copyPlaceURLBtn-container").innerHTML = "<button id='placeDetail-copyPlaceURLBtn' onclick='copyPlaceURL(" + '"' + placeUrl + '"' + "); return false;'>Copy URL</button>";
|
||||||
document.getElementById("placeDetail-maturity").innerHTML = placeDetail.maturity.toUpperCase();
|
document.getElementById("placeDetail-maturity").innerHTML = placeDetail.maturity.toUpperCase();
|
||||||
document.getElementById("placeDetail-maturity").className = placeDetail.maturity + "FilterOn placeMaturity";
|
document.getElementById("placeDetail-maturity").className = placeDetail.maturity + "FilterOn placeMaturity";
|
||||||
document.getElementById("placeDetail-domain").innerHTML = placeDetail.domain.toUpperCase();
|
document.getElementById("placeDetail-domain").innerHTML = placeDetail.domain.toUpperCase();
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
// places.js
|
// places.js
|
||||||
//
|
//
|
||||||
// Created by Alezia Kurdis, January 1st, 2022.
|
// Created by Alezia Kurdis, January 1st, 2022.
|
||||||
// Copyright 2022-2023 Overte e.V.
|
// Copyright 2022-2025 Overte e.V.
|
||||||
//
|
//
|
||||||
// Generate an explore app based on the differents source of placename data.
|
// Generate an explore app based on the differents source of placename data.
|
||||||
//
|
//
|
||||||
|
@ -36,6 +36,12 @@
|
||||||
var APP_ICON_ACTIVE = ROOT + "icons/appicon_a.png";
|
var APP_ICON_ACTIVE = ROOT + "icons/appicon_a.png";
|
||||||
var appStatus = false;
|
var appStatus = false;
|
||||||
var channel = "com.overte.places";
|
var channel = "com.overte.places";
|
||||||
|
|
||||||
|
var portalChannelName = "com.overte.places.portalRezzer";
|
||||||
|
var MAX_DISTANCE_TO_CONSIDER_PORTAL = 100.0; //in meters
|
||||||
|
var PORTAL_DURATION_MILLISEC = 45000; //45 sec
|
||||||
|
var rezzerPortalCount = 0;
|
||||||
|
var MAX_REZZED_PORTAL = 15;
|
||||||
|
|
||||||
var tablet = Tablet.getTablet("com.highfidelity.interface.tablet.system");
|
var tablet = Tablet.getTablet("com.highfidelity.interface.tablet.system");
|
||||||
|
|
||||||
|
@ -92,6 +98,24 @@
|
||||||
Window.location = messageObj.address;
|
Window.location = messageObj.address;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
} else if (messageObj.action === "REQUEST_PORTAL" && (n - timestamp) > INTERCALL_DELAY) {
|
||||||
|
d = new Date();
|
||||||
|
timestamp = d.getTime();
|
||||||
|
var portalPosition = Vec3.sum(MyAvatar.feetPosition, Vec3.multiplyQbyV(MyAvatar.orientation, {"x": 0.0, "y": 0.0, "z": -2.0}));
|
||||||
|
var requestToSend = {
|
||||||
|
"action": "REZ_PORTAL",
|
||||||
|
"position": portalPosition,
|
||||||
|
"url": messageObj.address,
|
||||||
|
"name": messageObj.name,
|
||||||
|
"placeID": messageObj.placeID
|
||||||
|
};
|
||||||
|
Messages.sendMessage(portalChannelName, JSON.stringify(requestToSend), false);
|
||||||
|
|
||||||
|
} else if (messageObj.action === "COPY_URL" && (n - timestamp) > INTERCALL_DELAY) {
|
||||||
|
d = new Date();
|
||||||
|
timestamp = d.getTime();
|
||||||
|
Window.copyToClipboard(messageObj.address);
|
||||||
|
Window.displayAnnouncement("Place URL copied.");
|
||||||
} else if (messageObj.action === "GO_HOME" && (n - timestamp) > INTERCALL_DELAY) {
|
} else if (messageObj.action === "GO_HOME" && (n - timestamp) > INTERCALL_DELAY) {
|
||||||
d = new Date();
|
d = new Date();
|
||||||
timestamp = d.getTime();
|
timestamp = d.getTime();
|
||||||
|
@ -284,8 +308,8 @@
|
||||||
region = "local";
|
region = "local";
|
||||||
order = "A";
|
order = "A";
|
||||||
fetch = true;
|
fetch = true;
|
||||||
pinned = false;
|
pinned = false;
|
||||||
currentFound = true;
|
currentFound = true;
|
||||||
} else {
|
} else {
|
||||||
region = "federation";
|
region = "federation";
|
||||||
order = "F";
|
order = "F";
|
||||||
|
@ -555,6 +579,57 @@
|
||||||
}
|
}
|
||||||
//####### END of seed random library ################
|
//####### END of seed random library ################
|
||||||
|
|
||||||
|
function onMessageReceived(paramChannel, paramMessage, paramSender, paramLocalOnly) {
|
||||||
|
if (paramChannel === portalChannelName) {
|
||||||
|
var instruction = JSON.parse(paramMessage);
|
||||||
|
if (instruction.action === "REZ_PORTAL") {
|
||||||
|
generatePortal(instruction.position, instruction.url, instruction.name, instruction.placeID);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function generatePortal(position, url, name, placeID) {
|
||||||
|
if (rezzerPortalCount <= MAX_REZZED_PORTAL) {
|
||||||
|
var TOLERANCE_FACTOR = 1.1;
|
||||||
|
if (Vec3.distance(MyAvatar.position, position) < MAX_DISTANCE_TO_CONSIDER_PORTAL) {
|
||||||
|
var height = MyAvatar.userHeight * MyAvatar.scale * TOLERANCE_FACTOR;
|
||||||
|
|
||||||
|
var portalPosition = Vec3.sum(position, {"x": 0.0, "y": height/2, "z": 0.0});
|
||||||
|
var dimensions = {"x": height * 0.618, "y": height, "z": height * 0.618};
|
||||||
|
var userdata = {
|
||||||
|
"url": url,
|
||||||
|
"name": name,
|
||||||
|
"placeID": placeID
|
||||||
|
};
|
||||||
|
|
||||||
|
var portalID = Entities.addEntity({
|
||||||
|
"position": portalPosition,
|
||||||
|
"dimensions": dimensions,
|
||||||
|
"type": "Shape",
|
||||||
|
"shape": "Sphere",
|
||||||
|
"name": "Portal to " + name,
|
||||||
|
"canCastShadow": false,
|
||||||
|
"collisionless": true,
|
||||||
|
"userData": JSON.stringify(userdata),
|
||||||
|
"script": ROOT + "portal.js",
|
||||||
|
"visible": "false",
|
||||||
|
"grab": {
|
||||||
|
"grabbable": false
|
||||||
|
}
|
||||||
|
}, "local");
|
||||||
|
rezzerPortalCount = rezzerPortalCount + 1;
|
||||||
|
|
||||||
|
Script.setTimeout(function () {
|
||||||
|
Entities.deleteEntity(portalID);
|
||||||
|
rezzerPortalCount = rezzerPortalCount - 1;
|
||||||
|
if (rezzerPortalCount < 0) {
|
||||||
|
rezzerPortalCount = 0;
|
||||||
|
}
|
||||||
|
}, PORTAL_DURATION_MILLISEC);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
function cleanup() {
|
function cleanup() {
|
||||||
|
|
||||||
if (appStatus) {
|
if (appStatus) {
|
||||||
|
@ -562,9 +637,15 @@
|
||||||
tablet.webEventReceived.disconnect(onAppWebEventReceived);
|
tablet.webEventReceived.disconnect(onAppWebEventReceived);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Messages.messageReceived.disconnect(onMessageReceived);
|
||||||
|
Messages.unsubscribe(portalChannelName);
|
||||||
|
|
||||||
tablet.screenChanged.disconnect(onScreenChanged);
|
tablet.screenChanged.disconnect(onScreenChanged);
|
||||||
tablet.removeButton(button);
|
tablet.removeButton(button);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Messages.subscribe(portalChannelName);
|
||||||
|
Messages.messageReceived.connect(onMessageReceived);
|
||||||
|
|
||||||
Script.scriptEnding.connect(cleanup);
|
Script.scriptEnding.connect(cleanup);
|
||||||
}());
|
}());
|
||||||
|
|
201
scripts/system/places/portal.js
Normal file
201
scripts/system/places/portal.js
Normal file
|
@ -0,0 +1,201 @@
|
||||||
|
//
|
||||||
|
// portal.js
|
||||||
|
//
|
||||||
|
// Created by Alezia Kurdis, January 14th, 2025.
|
||||||
|
// Copyright 2025, Overte e.V.
|
||||||
|
//
|
||||||
|
// 3D portal for Places app. portal spawner.
|
||||||
|
//
|
||||||
|
// 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 ROOT = Script.resolvePath('').split("portal.js")[0];
|
||||||
|
var portalURL = "";
|
||||||
|
var portalName = "";
|
||||||
|
var TP_SOUND = SoundCache.getSound(ROOT + "sounds/teleportSound.mp3");
|
||||||
|
|
||||||
|
this.preload = function(entityID) {
|
||||||
|
|
||||||
|
var properties = Entities.getEntityProperties(entityID, ["userData", "dimensions"]);
|
||||||
|
var userDataObj = JSON.parse(properties.userData);
|
||||||
|
portalURL = userDataObj.url;
|
||||||
|
portalName = userDataObj.name;
|
||||||
|
var portalColor = getColorFromPlaceID(userDataObj.placeID);
|
||||||
|
|
||||||
|
var textLocalPosition = {"x": 0.0, "y": (properties.dimensions.y / 2) * 1.2, "z": 0.0};
|
||||||
|
var scale = textLocalPosition.y/1.2;
|
||||||
|
var textID = Entities.addEntity({
|
||||||
|
"type": "Text",
|
||||||
|
"parentID": entityID,
|
||||||
|
"localPosition": textLocalPosition,
|
||||||
|
"dimensions": {
|
||||||
|
"x": 1 * scale,
|
||||||
|
"y": 0.15 * scale,
|
||||||
|
"z": 0.01
|
||||||
|
},
|
||||||
|
"name": portalName,
|
||||||
|
"text": portalName,
|
||||||
|
"textColor": portalColor.light,
|
||||||
|
"lineHeight": 0.10 * scale,
|
||||||
|
"backgroundAlpha": 0.0,
|
||||||
|
"unlit": true,
|
||||||
|
"alignment": "center",
|
||||||
|
"verticalAlignment": "center",
|
||||||
|
"canCastShadow": false,
|
||||||
|
"billboardMode": "yaw",
|
||||||
|
"grab": {
|
||||||
|
"grabbable": false
|
||||||
|
}
|
||||||
|
},"local");
|
||||||
|
|
||||||
|
var fxID = Entities.addEntity({
|
||||||
|
"type": "ParticleEffect",
|
||||||
|
"parentID": entityID,
|
||||||
|
"localPosition": {
|
||||||
|
"x": 0.0,
|
||||||
|
"y": 0.0,
|
||||||
|
"z": 0.0
|
||||||
|
},
|
||||||
|
"name": "PORTAL_FX",
|
||||||
|
"dimensions": {
|
||||||
|
"x": 5.2 * scale,
|
||||||
|
"y": 5.2 * scale,
|
||||||
|
"z": 5.2 * scale
|
||||||
|
},
|
||||||
|
"grab": {
|
||||||
|
"grabbable": false
|
||||||
|
},
|
||||||
|
"shapeType": "ellipsoid",
|
||||||
|
"color": portalColor.light,
|
||||||
|
"alpha": 0.1,
|
||||||
|
"textures": ROOT + "icons/portalFX.png",
|
||||||
|
"maxParticles": 600,
|
||||||
|
"lifespan": 0.6,
|
||||||
|
"emitRate": 1000,
|
||||||
|
"emitSpeed": -1 * scale,
|
||||||
|
"speedSpread": 0 * scale,
|
||||||
|
"emitOrientation": {
|
||||||
|
"x": 0,
|
||||||
|
"y": 0,
|
||||||
|
"z": 0,
|
||||||
|
"w": 1
|
||||||
|
},
|
||||||
|
"emitDimensions": {
|
||||||
|
"x": 1.28 * scale,
|
||||||
|
"y": 2 * scale,
|
||||||
|
"z": 1.28 * scale
|
||||||
|
},
|
||||||
|
"polarFinish": 3.1415927410125732,
|
||||||
|
"emitAcceleration": {
|
||||||
|
"x": 0,
|
||||||
|
"y": 0,
|
||||||
|
"z": 0
|
||||||
|
},
|
||||||
|
"particleRadius": 0.4000000059604645 * scale,
|
||||||
|
"radiusSpread": 0.30000001192092896 * scale,
|
||||||
|
"radiusStart": 1 * scale,
|
||||||
|
"radiusFinish": 0 * scale,
|
||||||
|
"colorStart": portalColor.saturated,
|
||||||
|
"colorFinish": {
|
||||||
|
"red": 255,
|
||||||
|
"green": 255,
|
||||||
|
"blue": 255
|
||||||
|
},
|
||||||
|
"alphaSpread": 0.019999999552965164,
|
||||||
|
"alphaStart": 0,
|
||||||
|
"alphaFinish": 0.20000000298023224,
|
||||||
|
"emitterShouldTrail": true,
|
||||||
|
"particleSpin": 1.5700000524520874,
|
||||||
|
"spinSpread": 2.9700000286102295,
|
||||||
|
"spinStart": 0,
|
||||||
|
"spinFinish": 0
|
||||||
|
},"local");
|
||||||
|
|
||||||
|
var loopSoundID = Entities.addEntity({
|
||||||
|
"type": "Sound",
|
||||||
|
"parentID": entityID,
|
||||||
|
"localPosition": {"x": 0.0, "y": 0.0, "z": 0.0},
|
||||||
|
"name": "PORTAL SOUND",
|
||||||
|
"soundURL": ROOT + "sounds/portalSound.mp3",
|
||||||
|
"volume": 0.15,
|
||||||
|
"loop": true,
|
||||||
|
"positional": true,
|
||||||
|
"localOnly": true
|
||||||
|
},"local");
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
this.enterEntity = function(entityID) {
|
||||||
|
var injectorOptions = {
|
||||||
|
"position": MyAvatar.position,
|
||||||
|
"volume": 0.3,
|
||||||
|
"loop": false,
|
||||||
|
"localOnly": true
|
||||||
|
};
|
||||||
|
var injector = Audio.playSound(TP_SOUND, injectorOptions);
|
||||||
|
|
||||||
|
var timer = Script.setTimeout(function () {
|
||||||
|
Window.location = portalURL;
|
||||||
|
Entities.deleteEntity(entityID);
|
||||||
|
}, 1000);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
function getColorFromPlaceID(placeID) {
|
||||||
|
var idIntegerConstant = getStringScore(placeID);
|
||||||
|
var hue = (idIntegerConstant%360)/360;
|
||||||
|
var color = hslToRgb(hue, 1, 0.5);
|
||||||
|
var colorLight = hslToRgb(hue, 1, 0.75);
|
||||||
|
return {
|
||||||
|
"saturated": {"red": color[0], "green": color[1], "blue": color[2]},
|
||||||
|
"light": {"red": colorLight[0], "green": colorLight[1], "blue": colorLight[2]},
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function getStringScore(str) {
|
||||||
|
var score = 0;
|
||||||
|
for (var j = 0; j < str.length; j++){
|
||||||
|
score += str.charCodeAt(j);
|
||||||
|
}
|
||||||
|
return score;
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Converts an HSL color value to RGB. Conversion formula
|
||||||
|
* adapted from http://en.wikipedia.org/wiki/HSL_color_space.
|
||||||
|
* Assumes h, s, and l are contained in the set [0, 1] and
|
||||||
|
* returns r, g, and b in the set [0, 255].
|
||||||
|
*
|
||||||
|
* @param {number} h The hue
|
||||||
|
* @param {number} s The saturation
|
||||||
|
* @param {number} l The lightness
|
||||||
|
* @return {Array} The RGB representation
|
||||||
|
*/
|
||||||
|
function hslToRgb(h, s, l){
|
||||||
|
var r, g, b;
|
||||||
|
|
||||||
|
if(s == 0){
|
||||||
|
r = g = b = l; // achromatic
|
||||||
|
}else{
|
||||||
|
var hue2rgb = function hue2rgb(p, q, t){
|
||||||
|
if(t < 0) t += 1;
|
||||||
|
if(t > 1) t -= 1;
|
||||||
|
if(t < 1/6) return p + (q - p) * 6 * t;
|
||||||
|
if(t < 1/2) return q;
|
||||||
|
if(t < 2/3) return p + (q - p) * (2/3 - t) * 6;
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
var q = l < 0.5 ? l * (1 + s) : l + s - l * s;
|
||||||
|
var p = 2 * l - q;
|
||||||
|
r = hue2rgb(p, q, h + 1/3);
|
||||||
|
g = hue2rgb(p, q, h);
|
||||||
|
b = hue2rgb(p, q, h - 1/3);
|
||||||
|
}
|
||||||
|
|
||||||
|
return [Math.round(r * 255), Math.round(g * 255), Math.round(b * 255)];
|
||||||
|
}
|
||||||
|
|
||||||
|
})
|
BIN
scripts/system/places/sounds/portalSound.mp3
Normal file
BIN
scripts/system/places/sounds/portalSound.mp3
Normal file
Binary file not shown.
BIN
scripts/system/places/sounds/teleportSound.mp3
Normal file
BIN
scripts/system/places/sounds/teleportSound.mp3
Normal file
Binary file not shown.
Loading…
Reference in a new issue