Overte-community-apps-Alezi.../applications/flyAvatar/app-flyAvatar.js
Alezia Kurdis e5f91da546
Add the "Fly Avatar" app
Fly Avatar
This application replaces your avatar for a specific one when you are flying. It reverts automatically the original avatar as soon as you land.
2023-12-17 22:29:13 -05:00

163 lines
5.7 KiB
JavaScript

//
// app-flyAvatar.js
//
// Created by Alezia Kurdis, December 16th 2023.
// Copyright 2023 Overte e.V.
//
// This automatically replace your avatar by a specific one as soon as you are flying.
//
// 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 jsMainFileName = "app-flyAvatar.js";
var ROOT = Script.resolvePath('').split(jsMainFileName)[0];
var APP_NAME = "FLY-AV";
var APP_URL = ROOT + "flyAvatar.html";
var APP_ICON_INACTIVE = ROOT + "icon_inactive.png";
var APP_ICON_ACTIVE = ROOT + "icon_active.png"; // BLACK on
var ICON_CAPTION_COLOR = "#FFFFFF";
var appStatus = false;
var channel = "overte.application.more.flyAvatar";
var timestamp = 0;
var INTERCALL_DELAY = 200; //0.3 sec
var FLY_AVATAR_SETTING_KEY = "overte.application.more.flyAvatar.avatarUrl";
var flyAvatarUrl = "";
var originalAvatarUrl = "";
var isFlying = false;
var UPDATE_TIMER_INTERVAL = 500; // 5 sec
var processTimer = 0;
var tablet = Tablet.getTablet("com.highfidelity.interface.tablet.system");
tablet.screenChanged.connect(onScreenChanged);
var button = tablet.addButton({
text: APP_NAME,
icon: APP_ICON_INACTIVE,
activeIcon: APP_ICON_ACTIVE,
captionColor: ICON_CAPTION_COLOR
});
function clicked(){
var colorCaption;
if (appStatus === true) {
tablet.webEventReceived.disconnect(onAppWebEventReceived);
tablet.gotoHomeScreen();
colorCaption = ICON_CAPTION_COLOR;
appStatus = false;
}else{
//Launching the Application UI.
tablet.gotoWebScreen(APP_URL); // <== Data can be transmitted at opening of teh UI by using GET method, through paramater in the URL. + "?parameter=value"
tablet.webEventReceived.connect(onAppWebEventReceived);
colorCaption = "#000000";
appStatus = true;
}
button.editProperties({
isActive: appStatus,
captionColor: colorCaption
});
}
button.clicked.connect(clicked);
//This recieved the message from the UI(html) for a specific actions
function onAppWebEventReceived(message) {
if (typeof message === "string") {
var d = new Date();
var n = d.getTime();
var instruction = JSON.parse(message);
if (instruction.channel === channel) {
if (instruction.action === "HUMAN_CALLED_ACTION_NAME" && (n - timestamp) > INTERCALL_DELAY) { //<== Use this for action trigger by a human (button or any ui control). The delay prevent multiple call to destabilize everything.
d = new Date();
timestamp = d.getTime();
//Call a function to do something here
} else if (instruction.action === "REQUEST_INITIAL_DATA") {
sendCurrentFlyAvatarUrlToUI();
} else if (instruction.action === "UPDATE_URL") {
flyAvatarUrl = instruction.url;
Settings.setValue( FLY_AVATAR_SETTING_KEY, flyAvatarUrl);
updateAvatar();
} else if (instruction.action === "SELF_UNINSTALL" && (n - timestamp) > INTERCALL_DELAY) { //<== This is a good practice to add a "Uninstall this app" button for rarely used app. (toolbar has a limit in size)
d = new Date();
timestamp = d.getTime();
ScriptDiscoveryService.stopScript(Script.resolvePath(''), false);
}
}
}
}
function updateAvatar() {
if (MyAvatar.isFlying()) {
MyAvatar.useFullAvatarURL(flyAvatarUrl);
} else {
if (MyAvatar.skeletonModelURL === flyAvatarUrl) {
MyAvatar.useFullAvatarURL(originalAvatarUrl);
}
}
}
MyAvatar.skeletonModelURLChanged.connect(function () {
if (!MyAvatar.isFlying() && MyAvatar.skeletonModelURL !== flyAvatarUrl) {
originalAvatarUrl = MyAvatar.skeletonModelURL;
}
});
function myTimer(deltaTime) {
var today = new Date();
if ((today.getTime() - processTimer) > UPDATE_TIMER_INTERVAL ) {
if (isFlying !== MyAvatar.isFlying()) {
updateAvatar();
isFlying = MyAvatar.isFlying();
}
today = new Date();
processTimer = today.getTime();
}
}
function sendCurrentFlyAvatarUrlToUI() {
var message = {
"channel": channel,
"action": "FLY-AVATAR-URL",
"url": flyAvatarUrl
};
tablet.emitScriptEvent(JSON.stringify(message));
}
function onScreenChanged(type, url) {
if (type === "Web" && url.indexOf(APP_URL) !== -1) {
colorCaption = "#000000";
appStatus = true;
} else {
colorCaption = ICON_CAPTION_COLOR;
appStatus = false;
}
button.editProperties({
isActive: appStatus,
captionColor: colorCaption
});
}
function cleanup() {
if (appStatus) {
tablet.gotoHomeScreen();
tablet.webEventReceived.disconnect(onAppWebEventReceived);
}
tablet.screenChanged.disconnect(onScreenChanged);
tablet.removeButton(button);
Script.update.disconnect(myTimer);
}
Script.scriptEnding.connect(cleanup);
originalAvatarUrl = MyAvatar.skeletonModelURL;
flyAvatarUrl = Settings.getValue( FLY_AVATAR_SETTING_KEY, "" );
Script.update.connect(myTimer);
}());