mirror of
https://github.com/overte-org/community-apps.git
synced 2025-04-05 21:22:00 +02:00
Merge pull request #88 from AleziaKurdis/flyAvatar2.0
FlyAvatar 2.0 (flying avatar defined per bookmark)
This commit is contained in:
commit
f3dbaf65bd
2 changed files with 292 additions and 34 deletions
|
@ -25,10 +25,11 @@
|
|||
var timestamp = 0;
|
||||
var INTERCALL_DELAY = 200; //0.3 sec
|
||||
var FLY_AVATAR_SETTING_KEY = "overte.application.more.flyAvatar.avatarUrl";
|
||||
var FLY_AVATAR_SETTING_KEY_2 = "overte.application.more.flyAvatar.avatarUrl.2";
|
||||
var FLY_AVATAR_SWITCH_SETTING_KEY = "overte.application.more.flyAvatar.switch";
|
||||
var FLY_AVATAR_ORIGINAL_AVATAR_SETTING_KEY = "overte.application.more.flyAvatar.originalAvatarUrl";
|
||||
var flyAvatarSwitch = true;
|
||||
var flyAvatarUrl = "";
|
||||
var flyAvatarUrl = []; //must be an array of {"avatarUrl": "", "flyAvatarUrl": ""}
|
||||
var originalAvatarUrl = "";
|
||||
var isFlying = false;
|
||||
var UPDATE_TIMER_INTERVAL = 500; // 5 sec
|
||||
|
@ -54,7 +55,6 @@
|
|||
colorCaption = ICON_CAPTION_COLOR;
|
||||
appStatus = false;
|
||||
}else{
|
||||
//Launching the Application UI.
|
||||
tablet.gotoWebScreen(APP_URL);
|
||||
tablet.webEventReceived.connect(onAppWebEventReceived);
|
||||
colorCaption = "#000000";
|
||||
|
@ -69,23 +69,18 @@
|
|||
|
||||
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") {
|
||||
if (instruction.action === "REQUEST_INITIAL_DATA") {
|
||||
sendCurrentFlyAvatarUrlToUI();
|
||||
} else if (instruction.action === "UPDATE_URL") {
|
||||
flyAvatarUrl = instruction.url;
|
||||
flyAvatarSwitch = instruction.mainSwitch;
|
||||
Settings.setValue( FLY_AVATAR_SETTING_KEY, flyAvatarUrl);
|
||||
Settings.setValue( FLY_AVATAR_SETTING_KEY_2, flyAvatarUrl);
|
||||
Settings.setValue( FLY_AVATAR_SWITCH_SETTING_KEY, flyAvatarSwitch);
|
||||
updateAvatar();
|
||||
if (flyAvatarSwitch) {
|
||||
|
@ -94,7 +89,7 @@
|
|||
inactiveIcon = APP_ICON_INACTIVE_OFF;
|
||||
}
|
||||
button.editProperties({icon: inactiveIcon});
|
||||
} 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)
|
||||
} else if (instruction.action === "SELF_UNINSTALL" && (n - timestamp) > INTERCALL_DELAY) {
|
||||
d = new Date();
|
||||
timestamp = d.getTime();
|
||||
ScriptDiscoveryService.stopScript(Script.resolvePath(''), false);
|
||||
|
@ -102,19 +97,51 @@
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
function getFlyAvatarOfAvatar(avatarUrl) {
|
||||
let i;
|
||||
let url = "";
|
||||
if (flyAvatarUrl.length > 0) {
|
||||
for (i = 0; i < flyAvatarUrl.length; i++) {
|
||||
if (flyAvatarUrl[i].avatarUrl === avatarUrl) {
|
||||
url = flyAvatarUrl[i].flyAvatarUrl;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return url;
|
||||
}
|
||||
|
||||
function getIndexOfFlyAvatar(urlOfFlyAvatar) {
|
||||
let i;
|
||||
let index = -1;
|
||||
if (flyAvatarUrl.length > 0) {
|
||||
for (i = 0; i < flyAvatarUrl.length; i++) {
|
||||
if (flyAvatarUrl[i].flyAvatarUrl === urlOfFlyAvatar) {
|
||||
index = i;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return index;
|
||||
}
|
||||
|
||||
function updateAvatar() {
|
||||
if (MyAvatar.isFlying() && flyAvatarSwitch) {
|
||||
MyAvatar.useFullAvatarURL(flyAvatarUrl);
|
||||
let replacement = getFlyAvatarOfAvatar(originalAvatarUrl);
|
||||
if (replacement !== "") {
|
||||
MyAvatar.useFullAvatarURL(replacement);
|
||||
}
|
||||
} else {
|
||||
if (MyAvatar.skeletonModelURL === flyAvatarUrl) {
|
||||
MyAvatar.useFullAvatarURL(originalAvatarUrl);
|
||||
let index = getIndexOfFlyAvatar(MyAvatar.skeletonModelURL);
|
||||
if (index !== -1) {
|
||||
MyAvatar.useFullAvatarURL(flyAvatarUrl[index].avatarUrl);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
MyAvatar.skeletonModelURLChanged.connect(function () {
|
||||
if (!MyAvatar.isFlying() && MyAvatar.skeletonModelURL !== flyAvatarUrl) {
|
||||
if (!MyAvatar.isFlying() && getIndexOfFlyAvatar(MyAvatar.skeletonModelURL) === -1) {
|
||||
originalAvatarUrl = MyAvatar.skeletonModelURL;
|
||||
Settings.setValue( FLY_AVATAR_ORIGINAL_AVATAR_SETTING_KEY, originalAvatarUrl);
|
||||
}
|
||||
|
@ -139,6 +166,7 @@
|
|||
"channel": channel,
|
||||
"action": "FLY-AVATAR-URL",
|
||||
"url": flyAvatarUrl,
|
||||
"bookmarks": AvatarBookmarks.getBookmarks(),
|
||||
"mainSwitch": flyAvatarSwitch
|
||||
};
|
||||
tablet.emitScriptEvent(JSON.stringify(message));
|
||||
|
@ -159,6 +187,20 @@
|
|||
});
|
||||
}
|
||||
|
||||
function isFlyAvatar(url) {
|
||||
let i;
|
||||
let isFlyAv = false;
|
||||
if (flyAvatarUrl.length > 0) {
|
||||
for (i = 0; i < flyAvatarUrl.length; i++) {
|
||||
if (flyAvatarUrl[i].flyAvatarUrl === url) {
|
||||
isFlyAv = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return isFlyAv;
|
||||
}
|
||||
|
||||
function cleanup() {
|
||||
|
||||
if (appStatus) {
|
||||
|
@ -173,9 +215,12 @@
|
|||
|
||||
Script.scriptEnding.connect(cleanup);
|
||||
originalAvatarUrl = MyAvatar.skeletonModelURL;
|
||||
flyAvatarUrl = Settings.getValue( FLY_AVATAR_SETTING_KEY, "" );
|
||||
if (Settings.getValue( FLY_AVATAR_SETTING_KEY, "" ) !== "") { //Clear Old Settings if present
|
||||
Settings.setValue( FLY_AVATAR_SETTING_KEY, "");
|
||||
}
|
||||
flyAvatarUrl = Settings.getValue( FLY_AVATAR_SETTING_KEY_2, "" );
|
||||
flyAvatarSwitch = Settings.getValue( FLY_AVATAR_SWITCH_SETTING_KEY, true );
|
||||
if (originalAvatarUrl === flyAvatarUrl) {
|
||||
if (isFlyAvatar(originalAvatarUrl)) {
|
||||
var lastRecordedOriginalAvatar = Settings.getValue( FLY_AVATAR_ORIGINAL_AVATAR_SETTING_KEY, "" );
|
||||
if (lastRecordedOriginalAvatar !== "") {
|
||||
originalAvatarUrl = lastRecordedOriginalAvatar;
|
||||
|
|
|
@ -20,7 +20,8 @@
|
|||
var thisPageName = "flyAvatar.html";
|
||||
var currentPath = window.location.protocol + "//" + window.location.host + window.location.pathname;
|
||||
var ROOTPATH = currentPath.replace(thisPageName, "");
|
||||
var flyAvatarUrl = "";
|
||||
var flyAvatarUrl = [];
|
||||
var bookmarks = [];
|
||||
var flyAvatarSwitch = true;
|
||||
|
||||
EventBridge.scriptEventReceived.connect(function(message){
|
||||
|
@ -28,9 +29,11 @@
|
|||
if (messageObj.channel === channel) {
|
||||
if (messageObj.action === "FLY-AVATAR-URL") {
|
||||
flyAvatarUrl = messageObj.url;
|
||||
bookmarks = messageObj.bookmarks;
|
||||
flyAvatarSwitch = messageObj.mainSwitch;
|
||||
document.getElementById("mainSwitch").checked = flyAvatarSwitch;
|
||||
document.getElementById("avatarUrl").value = flyAvatarUrl;
|
||||
genBookmarkMenu();
|
||||
genAvatarList();
|
||||
}
|
||||
}
|
||||
});
|
||||
|
@ -86,7 +89,12 @@
|
|||
}
|
||||
|
||||
#avatarUrl {
|
||||
width: 98%;
|
||||
width: 75%;
|
||||
font-size: 11px;
|
||||
font-family: FiraSans-SemiBold;
|
||||
background-color: #222222;
|
||||
color: #bbbbbb;
|
||||
border: 1px solid #bbbbbb;
|
||||
}
|
||||
|
||||
#avatarUrl:focus {
|
||||
|
@ -94,29 +102,142 @@
|
|||
}
|
||||
#formContainer {
|
||||
width: 100%;
|
||||
height: 590px;
|
||||
}
|
||||
#bookmarkPanel {
|
||||
height: 690px;
|
||||
width: 100%;
|
||||
position: fixed;
|
||||
z-index: 1;
|
||||
top: 0px;
|
||||
left: 0px;
|
||||
background-color: #cccccc;
|
||||
overflow-x: hidden;
|
||||
overflow-y: auto;
|
||||
transition: 0.5s;
|
||||
display: none;
|
||||
}
|
||||
button.menuItem {
|
||||
border: 0px;
|
||||
border-bottom: 1px solid #222222;
|
||||
text-align: left;
|
||||
color: #000000;
|
||||
width: 100%;
|
||||
background-color: #bbbbbb;
|
||||
font-size: 12px;
|
||||
font-family: FiraSans-SemiBold;
|
||||
}
|
||||
button.menuItem:hover {
|
||||
width: 100%;
|
||||
background-color: #333333;
|
||||
color: #bbbbbb;
|
||||
}
|
||||
#closeMenuBtn {
|
||||
text-align: right;
|
||||
width:100%;
|
||||
color: #000000;
|
||||
font-size: 18px;
|
||||
font-family: FiraSans-SemiBold;
|
||||
border-bottom: 1px solid #222222;
|
||||
}
|
||||
input:focus {
|
||||
outline: none;
|
||||
}
|
||||
button:focus {
|
||||
outline: none;
|
||||
}
|
||||
#bookmarks {
|
||||
font-family: FiraSans-SemiBold;
|
||||
background-color: #222222;
|
||||
font-size: 11px;
|
||||
color: #cccccc;
|
||||
border-radius: 5px;
|
||||
border: 0px solid #000000;
|
||||
transition-duration: 0.2s;
|
||||
padding: 3px 10px 3px 10px;
|
||||
margin-top: 3px;
|
||||
margin-bottom: 3px;
|
||||
}
|
||||
#bookmarks:hover {
|
||||
background-color: #111111;
|
||||
color: #ffffff;
|
||||
}
|
||||
#avatarList {
|
||||
height: 470px;
|
||||
width: 100%;
|
||||
background-color: #222222;
|
||||
overflow-x: hidden;
|
||||
overflow-y: auto;
|
||||
color: #bbbbbb;
|
||||
font-size: 8px;
|
||||
font-family: FiraSans-Regular;
|
||||
margin-bottom: 5px;
|
||||
}
|
||||
#addContainer {
|
||||
border: 1px solid #bbbbbb;
|
||||
width: 97%;
|
||||
padding: 5px;
|
||||
}
|
||||
font.formTitle {
|
||||
font-size: 14px;
|
||||
font-family: FiraSans-SemiBold;
|
||||
}
|
||||
#addBtn {
|
||||
font-family: FiraSans-SemiBold;
|
||||
background-color: #0066ff;
|
||||
background-image: linear-gradient(to bottom, #0066ff, #0020a1);
|
||||
font-size: 11px;
|
||||
color: #cccccc;
|
||||
border-radius: 5px;
|
||||
border: 0px solid #000000;
|
||||
transition-duration: 0.2s;
|
||||
padding: 3px 17px 3px 17px;
|
||||
margin-top: 5px;
|
||||
}
|
||||
#addBtn:hover {
|
||||
background-color: #26a1ff;
|
||||
background-image: linear-gradient(to bottom, #26a1ff, #0060d6);
|
||||
color: #ffffff;
|
||||
}
|
||||
span.removeBtn {
|
||||
font-family: FiraSans-SemiBold;
|
||||
font-size: 16px;
|
||||
color: #c27a7a;
|
||||
}
|
||||
table.grid {
|
||||
width: 100%;
|
||||
border-collapse: collapse;
|
||||
table-layout: fixed;
|
||||
}
|
||||
td {
|
||||
border: 1px solid #888888;
|
||||
color: #bbbbbb;
|
||||
font-size: 10px;
|
||||
font-family: FiraSans-Regular;
|
||||
word-wrap: break-word;
|
||||
padding: 3px;
|
||||
}
|
||||
tr {
|
||||
vertical-align: top;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<div id="formContainer">
|
||||
<h1>FLY AVATAR</h1>
|
||||
<div id="formContainer"><br><br>
|
||||
<input type="checkbox" id="mainSwitch" name="mainSwitch" value="true" oninput = "updateAvatarUrl();"><label for="mainSwitch"> Replace avatar when flying.</label>
|
||||
<br><br>
|
||||
Avatar Url to use while flying:<br>
|
||||
<input type = "text" id="avatarUrl" oninput = "updateAvatarUrl();"><br>
|
||||
</div><hr>
|
||||
<br><br><div id = "addContainer">
|
||||
<font class="formTitle">Link Avatar Url to use while flying to a specific avatar:</font><br>
|
||||
Avatar: <button id="bookmarks" onClick="openAvatarBookmarkMenu();">-Select-</button><div id="bookmarkPanel"></div><br>
|
||||
Flying Avatar Url: <input type = "text" id="avatarUrl"><br>
|
||||
<div style="text-align:right; width:97%;"><button id="addBtn" onclick = "updateAvatarUrl();">+ Add</button></div>
|
||||
</div></div><br>
|
||||
<div id="avatarList"></div>
|
||||
|
||||
<div style="text-align: right; width:100%;">
|
||||
<button id="uninstall" onClick = "uninstall();">Uninstall this app</button>
|
||||
</div>
|
||||
<script>
|
||||
//UI functions here
|
||||
|
||||
|
||||
//UI Action function here
|
||||
function uninstall() { //Example of a action called to the application (.js) (you can add the property you need to this, but minimally the channel and the action.
|
||||
function uninstall() {
|
||||
var message = {
|
||||
"channel": channel,
|
||||
"action": "SELF_UNINSTALL"
|
||||
|
@ -124,8 +245,69 @@
|
|||
EventBridge.emitWebEvent(JSON.stringify(message));
|
||||
}
|
||||
|
||||
function updateAvatarUrl() {
|
||||
flyAvatarUrl = document.getElementById("avatarUrl").value;
|
||||
function genAvatarList() {
|
||||
let i;
|
||||
let avatarlist = "";
|
||||
if (flyAvatarUrl.length > 0) {
|
||||
avatarlist = "<table class='grid'>";
|
||||
avatarlist = avatarlist + "<tr><td style='background-color: #000000; color:#ffffff; width: 25%;'>AVATAR</td>";
|
||||
avatarlist = avatarlist + "<td style='background-color: #000000; color:#ffffff; width: 70%;'>FLYING AVATAR URL</td>";
|
||||
avatarlist = avatarlist + "<td style='background-color: #000000; color:#ffffff; text-align: center; width: 5%;'> </td></tr>";
|
||||
for (i = 0; i < flyAvatarUrl.length; i++) {
|
||||
avatarlist = avatarlist + "<tr><td style='width: 25%;'>" + getBookmarkName(flyAvatarUrl[i].avatarUrl) + "</td>";
|
||||
avatarlist = avatarlist + "<td style='width: 70%;'>" + flyAvatarUrl[i].flyAvatarUrl + "</td>";
|
||||
avatarlist = avatarlist + "<td style='text-align: center; width: 5%;'>";
|
||||
avatarlist = avatarlist + "<span class='removeBtn' onClick='removeEntry(" + i + ")'>⮾</span></td></tr>";
|
||||
}
|
||||
avatarlist = avatarlist + "</table>";
|
||||
}
|
||||
document.getElementById("avatarList").innerHTML = avatarlist;
|
||||
}
|
||||
|
||||
function openAvatarBookmarkMenu() {
|
||||
document.getElementById("bookmarkPanel").style.display = "block";
|
||||
}
|
||||
|
||||
function closeAvatarBookmarkMenu() {
|
||||
document.getElementById("bookmarkPanel").style.display = "none";
|
||||
}
|
||||
|
||||
function genBookmarkMenu() {
|
||||
let opt = "<div id='closeMenuBtn' onClick = 'closeAvatarBookmarkMenu();'>🗙 </div>";
|
||||
let key;
|
||||
let panel = document.getElementById('bookmarkPanel');
|
||||
for (key in bookmarks) {
|
||||
opt = opt + "<button class = 'menuItem' onclick = 'selectBookMark(" + '"' + key + '"' + ");'>" + key + "</button><br>";
|
||||
}
|
||||
panel.innerHTML = opt;
|
||||
}
|
||||
|
||||
function getBookmarkName(url) {
|
||||
let key;
|
||||
let name = "Unknown";
|
||||
for (key in bookmarks) {
|
||||
if (bookmarks[key].avatarUrl === url) {
|
||||
name = key;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return name;
|
||||
}
|
||||
|
||||
function getBookmarkUrlFromName(name) {
|
||||
let key;
|
||||
let url = "";
|
||||
for (key in bookmarks) {
|
||||
if (key === name) {
|
||||
url = bookmarks[key].avatarUrl;
|
||||
break;
|
||||
}
|
||||
}
|
||||
return url;
|
||||
}
|
||||
|
||||
function removeEntry(index) {
|
||||
flyAvatarUrl.splice(index,1);
|
||||
flyAvatarSwitch = document.getElementById("mainSwitch").checked;
|
||||
var message = {
|
||||
"channel": channel,
|
||||
|
@ -134,6 +316,37 @@
|
|||
"mainSwitch": flyAvatarSwitch
|
||||
};
|
||||
EventBridge.emitWebEvent(JSON.stringify(message));
|
||||
genAvatarList();
|
||||
}
|
||||
|
||||
function selectBookMark(name) {
|
||||
document.getElementById('bookmarks').innerText = name;
|
||||
closeAvatarBookmarkMenu();
|
||||
}
|
||||
|
||||
function updateAvatarUrl() {
|
||||
let avUrl = getBookmarkUrlFromName(document.getElementById('bookmarks').innerText);
|
||||
let flyAv = document.getElementById("avatarUrl").value;
|
||||
|
||||
if (avUrl !== "" && flyAv !== "") {
|
||||
|
||||
let link = {
|
||||
"avatarUrl": avUrl,
|
||||
"flyAvatarUrl": flyAv
|
||||
};
|
||||
flyAvatarUrl.push(link);
|
||||
flyAvatarSwitch = document.getElementById("mainSwitch").checked;
|
||||
var message = {
|
||||
"channel": channel,
|
||||
"action": "UPDATE_URL",
|
||||
"url": flyAvatarUrl,
|
||||
"mainSwitch": flyAvatarSwitch
|
||||
};
|
||||
EventBridge.emitWebEvent(JSON.stringify(message));
|
||||
document.getElementById('bookmarks').innerText = "-Select-";
|
||||
document.getElementById("avatarUrl").value = "";
|
||||
genAvatarList();
|
||||
}
|
||||
}
|
||||
|
||||
function requestInitialData() {
|
||||
|
|
Loading…
Reference in a new issue