| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990 | import page from '/lib/page.mjs';import { Header } from '/web/header.js';class IndexApp {    constructor() {        console.log("app constructor");        this.worlds = {};        this.language = _LangManager.language;        this.options = {            query: 'pathname=' + window.location.pathname.slice(1,                window.location.pathname.lastIndexOf("/")),            secure: window.location.protocol === "https:",            reconnection: false,            path: '',            transports: ['websocket']        }        //window.location.host        var socket = io.connect(window._app.reflectorHost, this.options);        const parse = (msg) => {            this.parseOnlineData(msg)        }        socket.on('getWebAppUpdate', msg => parse.call(this, msg));        socket.on("connect", function () {            let noty = new Noty({                text: 'Connected to Reflector!',                timeout: 2000,                theme: 'mint',                layout: 'bottomRight',                type: 'success'            });            noty.show();        })        socket.on('connect_error', function (err) {            console.log(err);            var errDiv = document.createElement("div");            errDiv.innerHTML = "<div class='vwf-err' style='z-index: 10; position: absolute; top: 80px; right: 50px'>Connection error to Reflector!" + err + "</div>";            document.querySelector('body').appendChild(errDiv);            let noty = new Noty({                text: 'Connection error to Reflector! ' + err,                theme: 'mint',                layout: 'bottomRight',                type: 'error'            });            noty.show();        });    }    initHTML() {        let self = this;        //first init from _app        document.querySelector('head').innerHTML += '<link rel="stylesheet" href="/web/index-app.css">';        let headerGUI = new Header();        headerGUI.init();        //add HTML        let entry = document.createElement("div");        entry.setAttribute("id", 'app');        document.body.appendChild(entry);        let divs = ['appGUI', 'userLobby', 'main', 'worldsGUI'];        divs.forEach(el => {            let appEl = document.createElement("div");            appEl.setAttribute("id", el);            entry.appendChild(appEl);        })        //init CELL        document.querySelector("#userLobby").$cell({            id: "userLobby",            $cell: true,            $type: "div",            $components: [],            $update: function () {                this.$components = self.initUserGUI()            }        });        document.querySelector("#worldsGUI").$cell({            id: 'worldsGUI',            $cell: true,            $type: "div",            $components: [],            _comps: [],            _refresh: async function (data, fn) {                _app.showProgressBar();                this._comps = await fn.call(self, data);                this.$update();                _app.hideProgressBar();            },            $update: async function () {                this.$components = this._comps            }        });    }    async generateFrontPage() {        let infoEl = document.createElement("div");        infoEl.setAttribute("id", "indexPage");        let lang = _LangManager.locale;        let infoElHTML = await _app.helpers.getHtmlText('/web/locale/' + lang + '/index.html');        infoEl.innerHTML = infoElHTML;        document.body.appendChild(infoEl);        document.querySelector('#ruLang').addEventListener('click', function (e) {            _LangManager.locale = 'ru';            window.location.reload(true);        });        document.querySelector('#enLang').addEventListener('click', function (e) {            _LangManager.locale = 'en';            window.location.reload(true);        });    }    initApp() {        // let appElHTML = await _app.helpers.getHtmlText('/web/app.html');        // appEl.innerHTML = appElHTML;        // document.body.appendChild(appEl);        this.initUser();        document.querySelector("#userLobby").$update();        //this.initWorldsListGUI();        //this.getAppDetailsFromDB();    }    async initWorldsProtosListForUser(userAlias) {        document.querySelector("#worldsGUI").$components = [];        await document.querySelector("#worldsGUI")._refresh(userAlias, this.getWorldsProtosListForUser);    }    async initWorldsStatesListForUser(userAlias) {        document.querySelector("#worldsGUI").$components = [];        await document.querySelector("#worldsGUI")._refresh(userAlias, this.getWorldsStatesListForUser);    }    async getWorldsStatesListForUser(userAlias) {        let self = this;        let worldsGUI = [];        let worlds = self.createWorldsGUI(userAlias, 'allStates' );        await _app.getAllStateWorldsInfoForUser(userAlias,            function (data) {            let doc = document.querySelector("#allStates_" + userAlias);            if (doc) {                Object.assign(doc._states, data);                doc.$update();            }        }            );            worldsGUI.push(worlds);        // Object.entries(data).forEach(el => {        //     let worlds = this.createWorldsGUI(userAlias, el[0]);        //     worlds._states = el[1];        //     worlds.$update();        //     worldsGUI.push(worlds);        // })        return [            {                $type: "div",                class: "mdc-layout-grid",                $components: [                    {                        $type: "div",                        class: "mdc-layout-grid__inner",                        $components: [                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [                                    {                                        $type: "h1",                                        class: "mdc-typography--headline4",                                        $text: 'States for ' + userAlias                                    }                                ]                            },                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [].concat(worldsGUI)                            }                        ]                    }                ]            }        ]    }    async getWorldsProtosListForUser(userAlias) {        let worldsGUI = [];        //let data = await _app.getAllProtoWorldsInfoForUser(userAlias);        let worlds = this.createWorldsGUI(userAlias);        await _app.getAllProtoWorldsInfoForUser(userAlias, function (data) {            let doc = document.querySelector("#allWorlds_" + userAlias);            if (doc) {                Object.assign(doc._states, data);                doc.$update();            }        })        //worlds._states = data;        //worlds.$update();        worldsGUI.push(worlds);        return [            {                $type: "div",                class: "mdc-layout-grid",                $components: [                    {                        $type: "div",                        class: "mdc-layout-grid__inner",                        $components: [                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [                                    {                                        $type: "h1",                                        class: "mdc-typography--headline4",                                        $text: 'Worlds for ' + userAlias                                    }                                ]                            },                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [].concat(worldsGUI)                            },                        ]                    }                ]            }        ]    }    initUser() {        _LCSDB.on('auth', (ack) => {            if (ack.pub) {                let alias = _LCSUSER.is.alias;                let userEl = document.querySelector('#userGUI');                userEl._status = 'Welcome ' + alias + '!';                //userEl.style.backgroundColor = '#e6e6e6';                   userEl.$update();                // document.querySelector('#worldGUI').$update();                // document.querySelector('#main').$update();                _LCSUSER.get('profile').once(function (data) { console.log(data) })                var el = document.getElementById("loginGUI");                if (el) {                    el.remove();                }                _LCSUSER.get('profile').not(function (key) {                    let profile = { 'alias': alias };                    _LCSUSER.get('profile').put(profile);                })                let actionsGUI = document.querySelector('#worldActionsGUI');                if (actionsGUI)                    actionsGUI._refresh();                new Noty({                    text: alias + ' is succesfully authenticated!',                    timeout: 2000,                    theme: 'mint',                    layout: 'bottomRight',                    type: 'success'                }).show();                //this.getAppDetailsFromUserDB();            }            console.log(_LCSUSER.is);        });    }    initUserGUI() {        let worldGUI =        {            $type: "div",            id: "worldGUI",            class: "mdc-layout-grid mdc-layout-grid--align-left",            _status: '',            $init: function () {                this._status = "init";            },            $update: function () {                let guiForAll = [                    window._app.widgets.buttonStroked(                        {                            "label": 'Default World Protos',                            "onclick": function (e) {                                e.preventDefault();                                page("/app/worlds/protos")                                //window.location.pathname = "/app/worlds/protos"                                //_app.indexApp.getAppDetailsFromDefaultDB('protos');                            }                        }),                    window._app.widgets.buttonStroked(                        {                            "label": 'Default World States',                            "onclick": function (e) {                                e.preventDefault();                                page("/app/worlds/states")                                // window.location.pathname = "/app/worlds/states"                                //_app.indexApp.getAppDetailsFromDefaultDB('states');                            }                        })                ];                var guiUser = [];                if (_LCSUSER.is) {                    guiUser = []                }                this.$components = [                    {                        $type: "h1",                        class: "mdc-typography--headline4",                        $text: "Worlds list"                    }                ].concat(guiForAll).concat(guiUser)            }        }        let userGUI =        {            $type: "div",            id: "userGUI",            // style:"background-color: #ffeb3b",            class: "mdc-layout-grid mdc-layout-grid--align-left",            _status: "",            $init: function () {                this._status = "Welcome!"            },            $update: function () {                var gui = {};                if (_LCSUSER.is) {                    gui = [                        window._app.widgets.buttonRaised(                            {                                "label": 'Sign OUT',                                "onclick": function (e) {                                    _LCSUSER.leave().then(ack => {                                        if (ack.ok == 0) {                                            window.sessionStorage.removeItem('alias');                                            window.sessionStorage.removeItem('tmp');                                            window.location.reload(true);                                        }                                    });                                }                            }),                        {                            $type: "p"                        },                        window._app.widgets.buttonStroked(                            {                                "label": 'PROFILE',                                "onclick": function (e) {                                    e.preventDefault();                                    //page("/profile")                                    window.location.pathname = "/profile"                                }                            }),                        {                            $type: "p"                        },                        window._app.widgets.buttonStroked(                            {                                "label": 'My World protos',                                "onclick": function (e) {                                    e.preventDefault();                                    let alias = _LCSUSER.is.alias;                                    //window.location.pathname = '/' + alias + '/worlds/protos'                                    page('/' + alias + '/worlds/protos');                                    //_app.indexApp.getWorldsProtosFromUserDB(alias);                                }                            }),                        window._app.widgets.buttonStroked(                            {                                "label": 'My World states',                                "onclick": function (e) {                                    e.preventDefault();                                    let alias = _LCSUSER.is.alias;                                    // window.location.pathname = '/' + alias + '/worlds/states'                                    page('/' + alias + '/worlds/states');                                    // page.redirect('/' + alias + '/worlds/states');                                    //_app.indexApp.getWorldsFromUserDB(alias);                                       }                            })                    ]                }                this.$components = [                    window._app.widgets.buttonRaised(                        {                            "label": 'Connection settings',                            "onclick": function (e) {                                e.preventDefault();                                window.location.pathname = '/settings';                            }                        }),                    {                        $type: "h1",                        class: "mdc-typography--headline3",                        $text: this._status                    }                ].concat(gui)            }        }        let loginGUI =        {            $type: "div",            id: "loginGUI",            //style:"background-color: #efefef",            class: "mdc-layout-grid mdc-layout-grid--align-left",            _alias: null,            _pass: null,            _passField: null,            _aliasField: null,            _initData: function () {                this._alias = '';                this._pass = '';                // if (window.sessionStorage.getItem('alias')) {                //     this._alias = window.sessionStorage.getItem('alias')                // }                // if (window.sessionStorage.getItem('tmp')) {                //     this._pass = window.sessionStorage.getItem('tmp')                // }            },            $init: function () {                this._initData();            },            $update: function () {                this.$components = [                    {                        $type: "div",                        class: "mdc-layout-grid__inner",                        $components: [                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [                                    {                                        $type: "span",                                        class: "mdc-typography--headline5",                                        $text: "Login: "                                    },                                    window._app.widgets.inputTextFieldOutlined({                                        "id": 'aliasInput',                                        "label": "Login",                                        "value": this._alias,                                        "type": "text",                                        "init": function () {                                            this._aliasField = new mdc.textField.MDCTextField(this);                                        }                                    }),                                ]                            },                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [                                    {                                        $type: "span",                                        class: "mdc-typography--headline5",                                        $text: "Password: "                                    },                                    window._app.widgets.inputTextFieldOutlined({                                        "id": 'passwordInput',                                        "label": "Password",                                        "value": this._pass,                                        "type": "password",                                        "init": function () {                                            this._passField = new mdc.textField.MDCTextField(this);                                        }                                    }),                                ]                            },                            {                                $type: "div",                                class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                $components: [                                    window._app.widgets.buttonRaised(                                        {                                            "label": 'Sign UP',                                            "onclick": function (e) {                                                e.preventDefault();                                                let alias = this._aliasField.value;                                                let pass = this._passField.value                                                if (pass.length < 7) {                                                    new Noty({                                                        text: "Your passphrase needs to be longer than 7 letters",                                                        timeout: 2000,                                                        theme: 'mint',                                                        layout: 'bottomRight',                                                        type: 'error'                                                    }).show();                                                } else {                                                    //                                                    _LCSUSER.create(alias, pass,                                                        (ack) => {                                                            if (!ack.wait) { }                                                            if (ack.err) {                                                                console.log(ack.err)                                                                return ack.err                                                            };                                                            if (ack.pub) {                                                                _LCSUSER.auth(alias, pass);                                                                _LCSDB.get('users').get(alias).put({                                                                    'alias': alias,                                                                    'pub': ack.pub                                                                });                                                            }                                                        });                                                }                                            }                                        }),                                    {                                        $type: "span",                                        $text: " "                                    },                                    window._app.widgets.buttonRaised(                                        {                                            "label": 'Sign IN',                                            "onclick": function (e) {                                                e.preventDefault();                                                _LCSUSER.auth(this._aliasField.value, this._passField.value, ack => {                                                    if (ack.err) {                                                        new Noty({                                                            text: ack.err,                                                            timeout: 2000,                                                            theme: 'mint',                                                            layout: 'bottomRight',                                                            type: 'error'                                                        }).show();                                                    }                                                });                                            }                                        })                                ]                            }                        ]                    }                ]            }        }        // document.querySelector("#userLobby").$cell({        //     id: "userLobby",        //     $cell: true,        //     $type: "div",        //     $components: [         //         userGUI, loginGUI, _app.widgets.divider, worldGUI]        // }        // );        return [userGUI, loginGUI, _app.widgets.divider, worldGUI]    }    refresh() {        // socket.emit('getWebAppUpdate', "");    }    parseOnlineData(data) {        let parcedData = _app.parseAppInstancesData(data);        //if (Object.entries(parcedData).length !== 0)        let onlineGUIs = document.querySelectorAll('.online');        onlineGUIs.forEach(function (item) {            item._refresh(parcedData)        });    }    createWorldCard(id, type) {        let self = this;        let onlineGUI = {            $cell: true,            id: "onlineGUI_" + id,            class: "online",            $type: "div",            _instances: {},            _worldListItem: function (m) {                return {                    $type: "li",                    class: "mdc-list-item",                    $components: [                        {                            $type: "span",                            class: "world-link mdc-list-item__text",                            $components: [                                {                                    $type: "span",                                    class: "mdc-list-item__primary-text",                                    $components: [                                        {                                            $type: "a",                                            $text: m[0],                                            target: "_blank",                                            href: window.location.protocol + "//" + window.location.host + "/" + m[1].user + m[0],                                            onclick: function (e) {                                                //self.refresh();                                            }                                        },                                    ]                                },                                {                                    $type: "span",                                    class: "mdc-list-item__secondary-text",                                    $text: self.language.t('users') + m[1].clients                                }                            ]                        }                    ]                }            },            $components: [],            _refresh: function (data) {                if (data) {                    if (Object.entries(data).length !== 0) {                        if (this._worldInfo) {                            let insts = Object.entries(data).filter(el => el[0] == this._worldInfo.worldName);                            if (insts.length !== 0)                                this._instances = insts[0][1];                        }                    } else {                        this._instances = {}                    }                }            },            $init: function () {                this._refresh();            },            $update: function () {                if (this._instances) {                    let cardListData = Object.entries(this._instances).filter(el => el[1].user == this._worldInfo.userAlias);                    this.$components = [                        {                            $type: "hr",                            class: "mdc-list-divider"                        }                    ].concat(cardListData.map(this._worldListItem))                }            }        }        return {            $cell: true,            id: 'worldCard_' + id,            $type: "div",            _worldInfo: {},            _refresh: function (data) {                this._worldInfo = data            },            // _getWorldInfo: async function () {            //     //get space for user            //     let info = await _app.getWorldInfo(user, space);            //     this._refresh(info);            // },            // _getStateInfo: async function () {            //     //get space for user            //     let info = await _app.getStateInfo(user, space, saveName);            //     this._refresh(info);            // },            $init: function () {                //get space for user                // if (!saveName) {                //     this._getWorldInfo();                // } else {                //     this._getStateInfo();                // }            },            $update: function () {                //console.log(this._worldInfo);                this.$components = [this._updateCard()]            },            $components: [],            _updateCard: function () {                let desc = this._worldInfo;                if (!desc || Object.keys(desc).length == 0) {                    return {                        $type: "h1",                        class: "mdc-typography--headline4",                        $text: "ERROR: NO WORLD!"                    }                }                let userGUI = [];                let online = [];                let cardInfo = {                    "title": ""                };                if (type == "full") {                } else {                    userGUI.push({                        $type: "a",                        class: "mdc-button mdc-button--compact mdc-card__action mdc-button--outlined",                        $text: "Details",                        onclick: function (e) {                            e.preventDefault();                            window.location.pathname = "/" + desc.userAlias + '/' + desc.worldName + '/about'                        }                    });                }                userGUI.push({                    $type: "a",                    class: "mdc-button mdc-button--raised mdc-card__action ",                    $text: self.language.t('start'),//"Start new",                    target: "_blank",                    href: "/" + desc.userAlias + '/' + desc.worldName,                    onclick: function (e) {                        //self.refresh();                    }                });                if (desc.type == 'saveState') {                    cardInfo.title = desc.worldName.split('/')[2];                }                if (desc.type == 'proto') {                    cardInfo.title = desc.worldName;                    // userGUI.push(                    //     {                    //         $type: "a",                    //         class: "mdc-button mdc-button--compact mdc-card__action",                    //         $text: "States",                    //         onclick: async function (e) {                    //             e.preventDefault();                    //             window.location.pathname = "/" + desc.userAlias + '/' + desc.worldName +'/about'                    //             //console.log('clone');                    //             // document.querySelector('#worldStatesGUI')._refresh({});                    //             // let data = await _app.getSaveStates(desc.userAlias, desc.worldName);                    //             // document.querySelector('#worldStatesGUI')._refresh(data);                    //         }                    //     }                    // )                }                online.push(onlineGUI);                return {                    $cell: true,                    $type: "div",                    class: "mdc-card world-card",                    $components: [                        {                            $type: "section",                            class: "mdc-card__media world-card__16-9-media",                            $init: function () {                                if (desc.info.imgUrl !== "") {                                    this.style.backgroundImage = 'linear-gradient(0deg, rgba(0, 0, 0, 0.3), rgba(0, 0, 0, 0.3) ), url(' + desc.info.imgUrl + ')';                                }                            }                        },                        {                            $type: "section",                            class: "mdc-card__primary",                            $components: [                                {                                    $type: "h1",                                    class: "mdc-card__title mdc-card__title--large",                                    $text: desc.info.title                                },                                {                                    $type: "h2",                                    class: "mdc-card__subtitle mdc-theme--text-secondary-on-background",                                    $text: desc.info.text                                },                                {                                    $type: "span",                                    class: "mdc-card__subtitle mdc-theme--text-secondary-on-background",                                    $text: 'id: '                                },                                {                                    $type: "input",                                    type: "text",                                    disabled: "",                                    style: "font-size:18px",                                    value: cardInfo.title                                },                                {                                    $type: "p",                                },                                {                                    $type: "span",                                    class: "mdc-card__subtitle mdc-theme--text-secondary-on-background",                                    $text: 'created: ' + (new Date(desc.created)).toUTCString()                                },                                {                                    $type: "p",                                }                                // ,{                                //     $type: "span",                                //     class: "mdc-card__subtitle mdc-theme--text-secondary-on-background",                                //     $text: 'modified: ' + (new Date(desc[5])).toUTCString()                                // }                            ]                        },                        {                            $type: "section",                            class: "mdc-card__actions",                            $components: [                            ].concat(userGUI)                        },                        {                            $type: "section",                            class: "mdc-card__actions",                            $components: [                                {                                    $type: 'div',                                    $text: 'online now: '                                }                            ].concat(online)                        }                    ]                }            }        }    }    createWorldsGUI(userAlias, worldName) {        let self = this;        let id = worldName ? worldName + '_' + userAlias : "allWorlds_" + userAlias        let headerText = worldName ? 'States for ' + worldName : 'All Worlds Protos'        let worldCards = {            $cell: true,            id: id,            $type: "div",            $components: [],            _states: {},            _refresh: function (data) {                this._states = data            },            $init: async function () {                //this._refresh();            },            _makeWorldCard: function (data) {                let cardID = data[1].userAlias + '_' + data[1].worldName + '_' + data[0];                let card = self.createWorldCard(cardID, 'min');                card._worldInfo = data[1];                card.$update();                return {                    $cell: true,                    $type: "div",                    class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-4",                    $components: [                        card                        //self.createWorldCard(data[1].userAlias, data[1].worldName, data[0])                        //this._worldCardDef(appInfo)                    ]                }                //console.log(data);            },            $update: function () {                this.$components = [                    {                        $type: "div",                        class: "mdc-layout-grid",                        $components: [                            {                                $type: "div",                                class: "mdc-layout-grid__inner",                                $components: [                                    {                                        $type: "div",                                        class: "mdc-layout-grid__cell mdc-layout-grid__cell--span-12",                                        $components: [                                            {                                                $type: "H3",                                                $text: headerText                                            }                                        ]                                    }                                ]                            },                            {                                $type: "div",                                class: "mdc-layout-grid__inner",                                $components: Object.entries(this._states)                                    .filter(el => Object.keys(el[1]).length !== 0)                                    .sort(function (el1, el2) {                                        return parseInt(el2[1].created) - parseInt(el1[1].created)                                    })                                    .map(this._makeWorldCard)                            }                        ]                    }                ]            }        }        return worldCards    }}export { IndexApp }    //export {getAppDetails, generateFrontPage, setLanguage, initLocale};
 |