commit 1ecdd095a2e265d6ed99770c9ba4009c7f7f7903 Author: Ruel Tmeizeh - RuhNet Date: Sun Mar 2 17:59:33 2025 -0500 Storage App diff --git a/app.js b/app.js new file mode 100644 index 0000000..95b8e9c --- /dev/null +++ b/app.js @@ -0,0 +1,615 @@ +define(function(require) { + var $ = require('jquery'), + monster = require('monster'), + toastr = require('toastr'), + storagesConfig = require('./storages'); + + var settings = { + debug: false + }; + + var log = function(msg){ + if(settings.debug) { + console.log(msg); + } + }; + + // Autoload submodules + // (Submodules should be described in /apps/storagemgmt/storages.js) + var storagesList = storagesConfig.storages; + var storagesPaths = []; + for (var i = 0, len = storagesList.length; i < len; i++) { + storagesPaths.push('./submodules/' + storagesList[i] + '/' + storagesList[i]) + } + require(storagesPaths); + + var storageManager = { + name: 'storagemgmt', + css: [ 'app' ], + requests: {}, + + subscribe: { + 'storagemgmt.fetchStorages': 'define_storage_nodes' // For all submodules + }, + + subModules: storagesList, + + storages: {}, + + i18n: { + 'en-US': { customCss: false } + }, + + load: function(callback) { + var self = this; + + self.initApp(function() { + callback && callback(self); + }); + }, + + initApp: function(callback) { + var self = this; + + monster.pub('auth.initApp', { + app: self, + callback: callback + }); + }, + + render: function(container) { + var self = this; + + monster.pub('storagemgmt.fetchStorages', { + storages: self.storages, + callback: function (args) { + self.extendI18nOfSubmodule(args); + } + }); + + monster.ui.generateAppLayout(self, { + menus: [ + { + tabs: [ + { + callback: self.storageManagerRender + } + ] + } + ] + }); + + $(document.body).addClass('storagemgmt-app'); // class for styles; + }, + + extendI18nOfSubmodule: function (args) { + var self = this; + if(args.i18n && args.submoduleName) { + var submoduleLanguages = args.i18n; + var curLanguage = self.i18n.hasOwnProperty(monster.config.whitelabel.language) ? monster.config.whitelabel.language : monster.defaultLanguage; + + if (submoduleLanguages.length && submoduleLanguages.length > 0) { + if(submoduleLanguages.indexOf(curLanguage) > -1) { + $.getJSON('/apps/storagemgmt/submodules/' + args.submoduleName + '/i18n/' + curLanguage + '.json').done(function (newDict) { + var dict = self.data.i18n[curLanguage]; + $.extend(true, dict, newDict); + }) + } + } + } else { + log('Extend i18n of submodule failed'); + } + }, + + storageManagerRender: function(pArgs) { + var self = this, + args = pArgs || {}, + $container = args.container || $('.app-content-wrapper'), + callback = args.callback; + + if(pArgs.hasOwnProperty('onSetDefault') && typeof(pArgs.onSetDefault) === 'function') { + self.storageManagerOnSetDefault = pArgs.onSetDefault; + } + + if(!monster.util.isAdmin()) { + log('Permission error. Use admin account for change storage settings'); + return; + } + + self.getStorage(function(data) { + var storagesData = self.storageManagerFormatData(data); + for (var i = 0, len = storagesData.length; i < len; i++) { + storagesData[i].logo = self.storages[storagesData[i].type].getLogo() + } + + log('Storages List:'); + log(storagesData); + + var template = $(self.getTemplate({ + name: 'layout', + data: { + storages: storagesData + } + })); + + self.storageManagerBind(template, args, storagesData); + + $container.empty() + .append(template); + + if(typeof(callback) === 'function') { + callback(data); + } + }); + }, + + _doStorageInitialRequest: function(callback) { + var self = this; + + self.callApi({ + resource: 'storage.add', + data: { + accountId : self.accountId, + data : { + 'attachments': {}, + 'plan': {} + }, + removeMetadataAPI: true, + generateError: settings.debug + }, + success: function(data) { + if(typeof(callback) === 'function') { + callback(data); + } + }, + error: function(error) { + var errorMessage = self.i18n.active().storagemgmt.universalErrorMessageTemplate.replace('%api%', 'Storage'); + monster.ui.alert(errorMessage); + log(error.status + ' - ' + error.error + ': ' + error.message + ' '); + } + }); + }, + + getStorage: function(callback) { + var self = this; + + self.callApi({ + resource: 'storage.get', + data: { + accountId: self.accountId, + removeMetadataAPI: true, + generateError: false + }, + success: function(data) { + log('Storage Data:'); + log(data.data); + callback(data.data); + }, + error: function(data, error, globalHandler) { + self._doStorageInitialRequest(function() { + self.getStorage(callback); + }); + } + }); + }, + + storageManagerUpdateStorage: function(storageData, callback) { + var self = this; + + if(!monster.util.isAdmin()) { + log('Permission error. Use admin account for change storage settings'); + return; + } + + self.callApi({ + resource: 'storage.update', + data: { + accountId: self.accountId, + removeMetadataAPI: true, // or generateError: false + data: storageData + }, + success: function(data, status) { + if(typeof(callback) === 'function') { + callback(data); + } + }, + error: function(data, error, globalHandler) { + if (error.status === 404) { + callback(undefined); + } else { + globalHandler(data); + } + } + }); + }, + + storageManagerPatchStorage: function(storageData, callback) { + var self = this; + + if(!monster.util.isAdmin()) { + log('Permission error. Use admin account for change storage settings'); + return; + } + + self.callApi({ + resource: 'storage.patch', + data: { + accountId: self.accountId, + removeMetadataAPI: true, // or generateError: false + data: storageData + }, + success: function(data, status) { + if(typeof(callback) === 'function') { + callback(data); + } + }, + error: function(data, error, globalHandler) { + if (error.status === 404) { + callback(undefined); + } else { + globalHandler(data); + } + } + }); + }, + + storageManagerFormatData: function(data) { + var activeStorageId = null; + try { + activeStorageId = data.plan.modb.types.call_recording.attachments.handler; + } catch(e) { + log('Active storage not found'); + } + var itemData; + var storagesList = []; + if(data && data.hasOwnProperty('attachments') && Object.keys(data.attachments).length > 0) { + var attachments = data.attachments; + for(var i in attachments) if(attachments.hasOwnProperty(i)) { + itemData = { + id: i, + type: attachments[i].handler, + name: attachments[i].name, + settings: attachments[i].settings, + isActive: false + }; + + if(activeStorageId && itemData.id === activeStorageId) { + itemData.isActive = true; + } + storagesList.push(itemData) + } + } + + return storagesList; + }, + + storageManagerBind: function(template, args, data) { + var self = this; + + template.on('click', '.js-edit-storage', function(e) { + e.preventDefault(); + + var $editStorageBtn = $(this); + self.getStorage(function(data) { + + var $storageItem = $editStorageBtn.closest('.js-storage-item'); + var storageType = $storageItem.data('type'); + var uuid = $storageItem.data('uuid'); + var $container = $storageItem + .find('.js-item-settings-wrapper') + .hide(); + + if(data.attachments.hasOwnProperty(uuid)) { + var storageData = data.attachments[uuid]; + } + + var template = self.getTemplate({ + name: 'item-settings', + data: { + formElements: self.storages[storageType].getFormElements(storageData) + } + }); + + $container.empty() + .append(template); + $container.slideDown(); + + self.storageManagerSettingsBind($container); + }) + }); + + template.on('click', '.js-remove-storage', function(e) { + e.preventDefault(); + var uuid = $(this).closest('.js-storage-item').data('uuid'); + monster.ui.confirm(self.i18n.active().storagemgmt.confirmDeleteText, function() { + self.storageManagerDeleteStorage(uuid, function() { + $('.js-storage-item[data-uuid="' + uuid + '"]').slideUp(400, function() { + $(this).remove(); + }); + self.storageManagerShowMessage(self.i18n.active().storagemgmt.successRemovingMessage); + }); + }, undefined, { + type: 'warning', + title: self.i18n.active().storagemgmt.confirmDeleteTitle, + confirmButtonText: self.i18n.active().storagemgmt.confirmDelete + }); + }); + + template.on('click', '.js-create-storage', function(e) { + e.preventDefault(); + var $newStorageItem = $('.js-storage-items .js-new-storage-item'); + if ($newStorageItem.length === 0) { + self.storageManagerShowNewItemPanel(); + } else { + $newStorageItem.addClass('flash-effect'); + (function($newStorageItem){ + var timeoutId = setTimeout(function() { + $newStorageItem.removeClass('flash-effect'); + }, 2000) + })($newStorageItem) + } + }); + + template.on('click', '.js-set-default-storage', function(e) { + e.preventDefault(); + var uuid = $(this).closest('.js-storage-item').data('uuid'); + var isAlreadyActive = $(this).closest('.js-storage-item').hasClass('active-storage'); + + if(isAlreadyActive) { + self.storageManagerShowMessage(self.i18n.active().storagemgmt.alreadyActiveMessage, 'warning') + } else { + self.storageManagerSetDefaultStorage(uuid); + } + }); + }, + + storageManagerShowNewItemPanel: function(){ + var self = this; + + var data = []; + + var keyword = ''; + var storagesList = Object.keys(self.storages); + for (var i = 0, len = storagesList.length; i < len; i++) { + keyword = storagesList[i]; + data.push({ + name: keyword, + type: keyword, + logo: self.storages[keyword].getLogo(), + tabId: keyword + '-new-item-content', + tabLink: '#' + keyword + '-new-item-content', + formElements: self.storages[keyword].getFormElements({}) + }) + } + + var template = $(self.getTemplate({ + name: 'new-item', + data: { + storages: data + } + })); + + self.storageManagerNewItemBind(template); + + $('.js-storage-items').append(template); + $('.js-new-storage-item').hide().slideDown(400, function(){}); + $('.js-new-storage-tabs').tabs(); + }, + + storageManagerNewItemBind: function(template) { + var self = this; + + template.on('click', '.js-save', function (e) { + e.preventDefault(); + + var $tab = $(this).closest('.js-tab-content-item'); + var $form = $tab.find('.js-storage-settings-form'); + var formData = monster.ui.getFormData($form[0]); + var isNeedSetDefault = $tab.find('input[name="set_default"]').is(':checked'); + var typeKeyword = $tab.data('type'); + var newUuid = self.storageManagerGenerateUUID(); + delete formData['set_default']; + var storageData = self.storageManagerMakeConfig(typeKeyword, formData, newUuid); + + self.storageManagerPatchStorage(storageData, function(){ + var renderArgs = { + callback: function () { + self.storageManagerShowMessage(self.i18n.active().storagemgmt.successSavingMessage, 'success'); + } + }; + + if(isNeedSetDefault) { + self.storageManagerSetDefaultStorage(newUuid, function () { + self.storageManagerRender(renderArgs); + }); + } else { + self.storageManagerRender(renderArgs); + } + }); + }); + + template.on('click', '.js-cancel', function (e) { + e.preventDefault(); + $('.js-new-storage-item').slideUp(400, function(){ + $('.js-new-storage-item').remove(); + }); + }); + }, + + storageManagerMakeConfig (storageKeyword, data, uuid) { + var self = this, + storageData = { + 'attachments': {} + }; + + if(typeof(uuid) === 'undefined') { + uuid = self.storageManagerGenerateUUID(); + } + + if(storageKeyword && self.storages.hasOwnProperty(storageKeyword)) { + storageData.attachments[uuid] = data; + return storageData; + } else { + monster.ui.alert('Please install storage correctly (' + storageKeyword + ')'); + } + }, + + storageManagerSetDefaultStorage: function(uuid, callback) { + var self = this; + + if(!monster.util.isAdmin()) { + log('Permission error. Use admin account for change storage settings'); + return; + } + + var newData = { + plan: { + modb: { + types: { + call_recording: { + attachments: { + handler: uuid + } + }, + mailbox_message: { + attachments: { + handler: uuid + } + } + } + }, + account: { + types: { + media: { + attachments: { + handler: uuid + } + } + } + }, + } + }; + + + self.storageManagerPatchStorage(newData, function(data) { + $('#storage_manager_wrapper').find('.js-storage-item') + .removeClass('active-storage'); + + $('.js-storage-item[data-uuid="' + uuid + '"]').addClass('active-storage'); + + self.storageManagerOnSetDefault(data); + callback && callback(data); + }); + }, + + storageManagerOnSetDefault: function(data) {}, + + storageManagerSettingsBind: function($settingsContainer) { + var self = this; + + $settingsContainer.find('.js-cancel').on('click', function(e) { + e.preventDefault(); + $settingsContainer.slideUp(400, function(){ + $settingsContainer.empty(); + }); + }); + + $settingsContainer.find('.js-save').on('click', function(e) { + e.preventDefault(); + var $storageItem = $(this).closest('.js-storage-item'); + var $form = $storageItem.find('.js-storage-settings-form'); + var formData = monster.ui.getFormData($form[0]); + var storageName = formData.name; + var typeKeyword = $storageItem.data('type'); + var uuid = $storageItem.data('uuid'); + var storageData = self.storageManagerMakeConfig(typeKeyword, formData, uuid); + + self.getStorage(function(existStorageData) { + self.storageManagerPatchStorage(storageData, function(data) { + // update item name + $('.js-storage-item[data-uuid="' + uuid + '"]').find('.js-storage-name').text(storageName); + self.storageManagerShowMessage(self.i18n.active().storagemgmt.successUpdate, 'success'); + }); + }); + + }); + }, + + storageManagerDeleteStorage: function(uuid, callback) { + var self = this; + + if(!monster.util.isAdmin()) { + log('Permission error. Use admin account for change storage settings'); + return; + } + + self.getStorage(function(storagesData) { + var resultData = {}; + if(storagesData.hasOwnProperty('attachments')) { + resultData.attachments = storagesData.attachments; + } + if(storagesData.hasOwnProperty('plan')) { + resultData.plan = storagesData.plan; + } + + if(resultData.attachments && resultData.attachments.hasOwnProperty(uuid)) { + delete resultData.attachments[uuid]; + } + + try { + if(resultData.plan.modb.types.call_recording.attachments.handler === uuid) { + resultData.plan = {}; + } + } catch (e) {} + + self.storageManagerUpdateStorage(resultData, function() { + if(typeof(callback) === 'function') { + callback(); + } + }); + }) + }, + + storageManagerGenerateUUID: function() { + return 'xxxxxxxxxxxx4xxxyxxxxxxxxxxxxxxx'.replace(/[xy]/g, function(c) { + var r = Math.random() * 16 | 0, v = c == 'x' ? r : (r & 0x3 | 0x8); + return v.toString(16); + }); + }, + + storageManagerShowMessage: function(msg, msgType) { + var msgTypeClass; + + if(typeof(msgType) === 'undefined') { + msgType = 'info'; + } + + switch(msgType) { + case 'warning': + msgTypeClass = 'storage-msg-warning'; + break; + case 'success': + msgTypeClass = 'storage-msg-success'; + break; + default: // 'info' + msgTypeClass = 'storage-msg-info'; + } + + var $msg = $('
' + msg + '
') + .appendTo($('.js-storage-msg-box')).hide().fadeIn(); + + $msg.animate({ + backgroundColor: '#ffffff' + }, 1000 + ); + + window.setTimeout(function(){ + $msg.fadeOut(400, function() { + $msg.remove(); + }) + }, 4000); + } + }; + + return storageManager; +}); diff --git a/i18n/en-US.json b/i18n/en-US.json new file mode 100644 index 0000000..5701b8c --- /dev/null +++ b/i18n/en-US.json @@ -0,0 +1,33 @@ +{ + "storagemgmt": { + "title": "Storage Engine Management", + "description": "The Storage Engine Management app allows you manage your storage", + "universalErrorMessageTemplate": "Please contact your system administrator to configure %api% API, required for this application to work", + "welcome": "Welcome {{variable}} to the Storage Engine Management App", + "settingsBtnText": "Settings", + "settingsTitle": "Storage settings", + "newStorageBtnText": "New storage", + "itemSettings": { + "nameLabel": "Storage name", + "bucketLabel": "Bucket name", + "keyLabel": "Key", + "secretLabel": "Secret", + "saveBtnText": "Save Changes", + "cancelBtnText": "Cancel", + "successUpdate": "Your Storage Settings were successfully updated!", + "setDefaultLabel": "Set default" + }, + "newItem": { + "saveNewItemText": "Create", + "cancelBtnText": "Cancel" + }, + "confirmDelete": "Yes, Remove Storage", + "confirmDeleteText": "This will remove this Storage from this account.", + "confirmDeleteTitle": "Warning! Are you sure?", + "successRemovingMessage": "You successfully removed the storage.", + "successSavingMessage": "You successfully saved the storage.", + "successUpdate": "Your Storage Settings were successfully updated!", + "alreadyActiveMessage": "This storage is already active", + "storageError": "Stores do not work correctly. Contact your administrator to configure it." + } +} diff --git a/metadata/app.json b/metadata/app.json new file mode 100644 index 0000000..dac454f --- /dev/null +++ b/metadata/app.json @@ -0,0 +1,26 @@ +{ + "name": "storage", + "i18n": { + "en-US": { + "label": "Storage", + "description": "The storage management app allows you to configure storage" + } + }, + "tags": [ + "reseller" + ], + "icon": "storage.png", + "api_url": "", + "author": "(C) 2019 CONVERBA LIMITED/2025 RuhNet", + "version": "1.0", + "license": "MPL", + "price": 0, + "screenshots": [ + "storage1.png" + ], + "urls": { + "documentation": "{documentation_url}", + "howto": "{howto_video_url}" + }, + "pvt_type": "app" +} diff --git a/metadata/icon/storage.png b/metadata/icon/storage.png new file mode 100644 index 0000000..546222a Binary files /dev/null and b/metadata/icon/storage.png differ diff --git a/metadata/screenshots/storage1.png b/metadata/screenshots/storage1.png new file mode 100644 index 0000000..67984d1 Binary files /dev/null and b/metadata/screenshots/storage1.png differ diff --git a/storages.js b/storages.js new file mode 100644 index 0000000..3339764 --- /dev/null +++ b/storages.js @@ -0,0 +1,8 @@ +define(function(require) { + return { + "storages": [ + "s3", + "mts" + ] + }; +}); diff --git a/style/app.css b/style/app.css new file mode 100644 index 0000000..2ed82e9 --- /dev/null +++ b/style/app.css @@ -0,0 +1,295 @@ +.storagemgmt-app .storages-settings { + display: block; + background: white; + padding: 13px 23px 2px; + margin-bottom: 24px; + border-radius: 8px; + position: relative; } + .storagemgmt-app .storages-settings .close-btn { + width: 31px; + display: inline-block; + text-align: center; + position: absolute; + top: 10px; + right: 10px; } + .storagemgmt-app .storages-settings .close-btn i.fa { + font-size: 20px; } + .storagemgmt-app .storages-settings .close-btn:hover i.fa { + color: #22A5FF; } + +.storage-manager-wrapper { + padding: 0 0 15px 0; + margin-bottom: 15px; } + .storage-manager-wrapper .buttons-wrapper .dynamic-link { + font-size: 16px; + vertical-align: top; + margin-left: 18px; + line-height: 30px; } + .storage-manager-wrapper .btn { + vertical-align: top; } + .storage-manager-wrapper a.btn .fa { + margin-right: 5px; + vertical-align: -1px; } + .storage-manager-wrapper a.btn:hover .fa { + color: #22A5FF; } + .storage-manager-wrapper .storage-items-list { + margin-bottom: 10px; } + .storage-manager-wrapper h3 { + color: #555; + font-weight: normal; + line-height: 1; + font-size: 22px; + margin: 5px 0 13px; } + .storage-manager-wrapper .empty-storage { + border: 1px dashed #999; + height: 320px; + text-align: center; } + .storage-manager-wrapper .empty-storage button { + margin-top: 25px; } + .storage-manager-wrapper .empty-storage i.main-icon { + margin-top: 60px; + font-size: 72px; } + .storage-manager-wrapper .empty-storage .main-title { + font-size: 18px; + margin-top: 40px; } + .storage-manager-wrapper .empty-storage .sub-title { + color: #909099; + margin-top: 5px; } + .storage-manager-wrapper .storage-choice .logo-header { + display: inline-block; + vertical-align: top; + width: 118px; + height: 40px; + padding-top: 8px; } + .storage-manager-wrapper .storage-choice .logo-header img, + .storage-manager-wrapper .storage-choice .logo-header svg { + display: block; + max-height: 40px; } + .storage-manager-wrapper .storage-choice .item-settings-wrapper { + clear: both; } + .storage-manager-wrapper .storage-provider-wrapper { + opacity: 0.6; + max-width: 596px; + width: 49%; + display: inline-block; + vertical-align: top; + margin: 6px 12px 6px 0; } + .storage-manager-wrapper .storage-provider-wrapper.active-storage, .storage-manager-wrapper .storage-provider-wrapper.new-storage-item, .storage-manager-wrapper .storage-provider-wrapper:hover { + opacity: 1; } + .storage-manager-wrapper .storage-provider-wrapper button { + margin-right: 10px; + font-size: 14px; } + .storage-manager-wrapper .storage-provider-wrapper.active-storage .storage-choice .check-icon-wrapper { + visibility: visible; } + .storage-manager-wrapper .storage-provider-wrapper.active-storage .storage-choice a .check-icon { + color: #22A5FF; + opacity: 1; } + .storage-manager-wrapper .storage-provider-wrapper:hover .storage-choice .check-icon-wrapper { + visibility: visible; + opacity: 0.4; + /*webkit-animation: blinker 2s linear infinite; + animation: blinker 2s linear infinite; */ } + .storage-manager-wrapper .storage-provider-wrapper.active-storage:hover .storage-choice .check-icon-wrapper { + opacity: 1; } + .storage-manager-wrapper .storage-provider-wrapper .storage-choice a:hover .check-icon { + color: #22A5FF; } + .storage-manager-wrapper .storage-provider-wrapper .storage-item-settings > .row-fluid { + border-top: 1px solid #D0D0D9; + padding-top: 18px; } + .storage-manager-wrapper .storage-choice { + background: #FFF; + border: 1px solid #D0D0D9; + border-radius: 2px; + line-height: 46px; + padding: 7px 25px 7px; } + .storage-manager-wrapper .storage-choice::after { + content: ''; + display: block; + clear: both; } + .storage-manager-wrapper .storage-choice .check-icon-wrapper { + width: 22px; + display: inline-block; + vertical-align: 5px; + margin-right: 15px; + visibility: hidden; } + .storage-manager-wrapper .storage-choice .check-icon { + font-size: 25px; } + .storage-manager-wrapper .storage-choice .right-section { + float: right; + line-height: 25px; + padding-top: 13px; + z-index: 10; + position: relative; } + .storage-manager-wrapper .storage-choice .right-section a { + margin-left: 11px; + vertical-align: top; + display: inline-block; } + .storage-manager-wrapper .storage-choice .right-section a.remove-storage-link { + position: relative; + top: -2px; } + .storage-manager-wrapper .storage-choice .right-section a:hover .fa { + color: #22A5FF; } + .storage-manager-wrapper .storage-choice .right-section a:hover .fa { + color: #22A5FF; } + .storage-manager-wrapper .storage-choice .storage-manager-wrapper .active-storage .select-storage-link:hover .fa { + cursor: default; + /*color: #555;*/ } + .storage-manager-wrapper .storage-choice .right-section a.remove-storage-link:hover .fa { + color: #ff0009; } + .storage-manager-wrapper .storage-choice a .fa { + font-size: 25px; + vertical-align: middle; } + .storage-manager-wrapper .storage-choice .edit-storage-link:hover { + color: #22A5FF; } + .storage-manager-wrapper .storage-choice .storage-name { + font-weight: 600; + text-align: left; + display: inline-block; + vertical-align: top; + margin-top: 0; + width: 160px; + white-space: nowrap; + text-overflow: ellipsis; + overflow: hidden; } + .storage-manager-wrapper .storage-choice .folder-icon { + color: #D0D0D9; + font-size: 20px; + margin-right: 10px; + vertical-align: middle; } + .storage-manager-wrapper .storage-choice .edit-path { + color: #505059; + cursor: pointer; + font-size: 20px; + margin-left: 10px; + vertical-align: middle; } + .storage-manager-wrapper .storage-choice #path { + margin: 0 5px; } + .storage-manager-wrapper .storage-choice .custom-path { + display: inline-block; } + +.storage-item-settings .storage-item-logo { + text-align: center; } + .storage-item-settings .storage-item-logo img, .storage-item-settings .storage-item-logo svg { + max-height: 40px; } + +.storage-item-settings .form-horizontal { + margin: 0 auto; + display: block; + max-width: 410px; } + .storage-item-settings .form-horizontal .control-label { + width: 110px; + display: inline-block; + vertical-align: top; + float: none; + padding-right: 14px; } + .storage-item-settings .form-horizontal .controls { + margin-left: 0; + display: inline-block; + vertical-align: top; } + +.storage-item-settings input[type="text"], +.storage-item-settings input[type="password"] { + border-right: 1px solid #F5F5F5; + border-bottom: 1px solid #F5F5F5; + font-size: 14px; } + +.storage-item-settings .buttons-wrapper { + text-align: center; } + +.storage-item-settings .ui-tabs { + border: none; + background: none; + padding: 0; } + .storage-item-settings .ui-tabs .ui-tabs-nav { + background: none; + border: none; + padding: 0; } + .storage-item-settings .ui-tabs .ui-tabs-nav li { + border: 0; + border-radius: 0; + background: #9B9B9B; + margin: 0; + width: 50%; } + .storage-item-settings .ui-tabs .ui-tabs-nav li.ui-tabs-active, .storage-item-settings .ui-tabs .ui-tabs-nav li.ui-tabs-active.ui-state-hover { + border: none; + border-radius: 0; + margin: 0; + background: white; } + .storage-item-settings .ui-tabs .ui-tabs-nav li.ui-state-hover { + background: #CDCDCD; + border: none; } + .storage-item-settings .ui-tabs .ui-tabs-nav li a { + width: 100%; + box-sizing: border-box; } + .storage-item-settings .ui-tabs .ui-tabs-panel { + padding-bottom: 0; } + +.new-storage-item.flash-effect > .storage-choice { + animation-name: flash_border; + animation-duration: 2s; + animation-timing-function: linear; + animation-iteration-count: 2; } + +.new-storage-item .storage-choice { + padding: 0; } + +.new-storage-item .storage-item-settings { + padding: 0; } + +.new-storage-item .logo-header { + display: block; + margin: 0 auto; } + +.new-storage-item .tab-content { + background: none; + border: none; } + +.icon-loader { + width: 24px; + height: 24px; + display: inline-block; + vertical-align: top; + background: url(""); } + +.storage-message { + padding: 10px; + border: none; + max-width: 596px; + width: 49%; + min-width: 250px; + margin-bottom: 10px; + font-size: 16px; + line-height: 1.2; + background: white; + color: black; + box-sizing: border-box; } + +.storage-msg-warning { + background-color: #ff4200; } + +.storage-msg-info { + background-color: #22A5FF; } + +.storage-msg-success { + background-color: #00cc2e; } + +.dynamic-link:hover { + border-bottom: 1px dashed; } + +@media (max-width: 1000px) { + .storage-manager-wrapper .storage-provider-wrapper { + width: 100%; } } + +/* ---------------------------- + Animations ---------------*/ +@keyframes blinker { + 50% { + opacity: 0; } } + +@keyframes flash_border { + 0% { + border-color: #22A5FF; } + 50% { + border-color: #D0D0D9; } + 100% { + border-color: #22A5FF; } } diff --git a/style/app.scss b/style/app.scss new file mode 100644 index 0000000..b002de5 --- /dev/null +++ b/style/app.scss @@ -0,0 +1,442 @@ +.storagemgmt-app { + .storages-settings { + display: block; + background: white; + padding: 13px 23px 2px; + margin-bottom: 24px; + border-radius: 8px; + position: relative; + + .close-btn { + width: 31px; + display: inline-block; + text-align: center; + position: absolute; + top: 10px; + right: 10px; + } + + .close-btn { + i.fa { + font-size: 20px + } + + &:hover i.fa { + color: #22A5FF; + } + + } + } +} + + +.storage-manager-wrapper { + padding: 0 0 15px 0; + margin-bottom: 15px; + + .buttons-wrapper .dynamic-link { + font-size: 16px; + vertical-align: top; + margin-left: 18px; + line-height: 30px; + } + + .btn { + vertical-align: top; + } + + a.btn .fa { + margin-right: 5px; + vertical-align: -1px; + } + + a.btn:hover .fa { + color: #22A5FF; + } + + .storage-items-list { + margin-bottom: 10px; + } + + h3 { + color: #555; + font-weight: normal; + line-height: 1; + font-size: 22px; + margin: 5px 0 13px; + } + + .empty-storage { + border: 1px dashed #999; + height: 320px; + text-align: center; + } + + .empty-storage button { + margin-top: 25px; + } + + .empty-storage i.main-icon { + margin-top: 60px; + font-size: 72px; + } + + .empty-storage .main-title { + font-size: 18px; + margin-top: 40px; + } + + .empty-storage .sub-title { + color: #909099; + margin-top: 5px; + } + + .storage-choice .logo-header { + display: inline-block; + vertical-align: top; + width: 118px; + height: 40px; + padding-top: 8px; + + img, + svg { + display: block; + max-height: 40px; + } + } + + .storage-choice .item-settings-wrapper { + clear: both; + } + + .storage-provider-wrapper { + opacity: 0.6; + max-width: 596px; + width: 49%; + display: inline-block; + vertical-align: top; + margin: 6px 12px 6px 0; + + &.active-storage, + &.new-storage-item, + &:hover { + opacity: 1; + } + + button { + margin-right: 10px; + font-size: 14px; + } + + &.active-storage .storage-choice { + .check-icon-wrapper { + visibility: visible; + } + + a .check-icon { + color: #22A5FF; + opacity: 1; + } + } + + &:hover .storage-choice .check-icon-wrapper { + visibility: visible; + opacity: 0.4; + /*webkit-animation: blinker 2s linear infinite; + animation: blinker 2s linear infinite; */ + } + + &.active-storage:hover .storage-choice .check-icon-wrapper { + opacity: 1; + } + + .storage-choice a:hover .check-icon { + color: #22A5FF; + } + + .storage-item-settings >.row-fluid { + border-top: 1px solid #D0D0D9; + padding-top: 18px; + } + } + + .storage-choice { + background: #FFF; + border: 1px solid #D0D0D9; + border-radius: 2px; + line-height: 46px; + padding: 7px 25px 7px; + + &::after { + content: ''; + display: block; + clear: both; + } + + .check-icon-wrapper { + width: 22px; + display: inline-block; + vertical-align: 5px; + margin-right: 15px; + visibility: hidden; + } + + .check-icon { + font-size: 25px; + } + + + .right-section { + float: right; + line-height: 25px; + padding-top: 13px; + z-index: 10; + position: relative; + } + + .right-section a { + margin-left: 11px; + vertical-align: top; + display: inline-block; + } + + .right-section a.remove-storage-link { + position: relative; + top: -2px; + } + + .right-section a:hover .fa { + color: #22A5FF; + } + + .right-section a:hover .fa { + color: #22A5FF; + } + + .storage-manager-wrapper .active-storage .select-storage-link:hover .fa { + cursor: default; + /*color: #555;*/ + } + + .right-section a.remove-storage-link:hover .fa { + color: #ff0009; + } + + a .fa { + font-size: 25px; + vertical-align: middle; + } + + .edit-storage-link:hover { + color: #22A5FF; + } + + .storage-name { + font-weight: 600; + text-align: left; + display: inline-block; + vertical-align: top; + margin-top: 0; + width: 160px; + white-space: nowrap; + text-overflow: ellipsis; + overflow: hidden; + } + + .folder-icon { + color: #D0D0D9; + font-size: 20px; + margin-right: 10px; + vertical-align: middle; + } + + .edit-path { + color: #505059; + cursor: pointer; + font-size: 20px; + margin-left: 10px; + vertical-align: middle; + } + + #path { + margin: 0 5px; + } + + .custom-path { + display: inline-block; + } + } +} + +.storage-item-settings { + .storage-item-logo { + text-align: center; + + img, svg { + max-height: 40px; + } + } + + .form-horizontal { + margin: 0 auto; + display: block; + max-width: 410px; + + .control-label { + width: 110px; + display: inline-block; + vertical-align: top; + float: none; + padding-right: 14px; + } + + .controls { + margin-left: 0; + display: inline-block; + vertical-align: top; + } + } + + input[type="text"], + input[type="password"] { + border-right: 1px solid #F5F5F5; + border-bottom: 1px solid #F5F5F5; + font-size: 14px; + } + + .buttons-wrapper { + text-align: center; + } + + .ui-tabs { + border: none; + background: none; + padding: 0; + + .ui-tabs-nav { + background: none; + border: none; + padding: 0; + + li { + border: 0; + border-radius: 0; + background: #9B9B9B; + margin: 0; + width: 50%; + + &.ui-tabs-active, + &.ui-tabs-active.ui-state-hover { + border: none; + border-radius: 0; + margin: 0; + background: white; + } + + &.ui-state-hover { + background: #CDCDCD; + border: none; + } + + a { + width: 100%; + box-sizing: border-box; + } + } + } + + .ui-tabs-panel { + padding-bottom: 0; + } + } +} + +.new-storage-item { + &.flash-effect > .storage-choice { + animation-name: flash_border; + animation-duration: 2s; + animation-timing-function: linear; + animation-iteration-count: 2; + } + + .storage-choice { + padding: 0; + } + + .storage-item-settings { + padding: 0; + } + + .logo-header { + display: block; + margin: 0 auto; + } + + .tab-content { + background: none; + border: none; + } +} + +.icon-loader { + width: 24px; + height: 24px; + display: inline-block; + vertical-align: top; + background: url(''); +} + +.storage-message { + padding: 10px; + border: none; + max-width: 596px; + width: 49%; + min-width: 250px; + margin-bottom: 10px; + font-size: 16px; + line-height: 1.2; + background: white; + color: black; + box-sizing: border-box; +} + +.storage-msg-warning { + background-color: #ff4200; +} + +.storage-msg-info { + background-color: #22A5FF; +} + +.storage-msg-success { + background-color: #00cc2e; +} + +.dynamic-link:hover { + border-bottom: 1px dashed; +} + +@media (max-width: 1000px) { + .storage-manager-wrapper .storage-provider-wrapper { + width: 100%; + } +} + +/* ---------------------------- + Animations ---------------*/ + +@keyframes blinker { + 50% { opacity: 0; } +} + +@keyframes flash_border { + 0% { + border-color: #22A5FF; + } + 50% { + border-color: #D0D0D9; + } + 100% { + border-color: #22A5FF; + } +} diff --git a/style/static/images/mts.svg b/style/static/images/mts.svg new file mode 100644 index 0000000..3a0b7f3 --- /dev/null +++ b/style/static/images/mts.svg @@ -0,0 +1,23 @@ + + + + + + + + + + + + + + + + diff --git a/submodules/mts/i18n/en-US.json b/submodules/mts/i18n/en-US.json new file mode 100644 index 0000000..658c9a4 --- /dev/null +++ b/submodules/mts/i18n/en-US.json @@ -0,0 +1,12 @@ +{ + "storagemgmt": { + "submodules": { + "mts": { + "nameLabel": "Name", + "bucketLabel": "Bucket", + "keyLabel": "Key", + "secretLabel": "Secret" + } + } + } +} diff --git a/submodules/mts/img/logo.png b/submodules/mts/img/logo.png new file mode 100644 index 0000000..6b2f2ea Binary files /dev/null and b/submodules/mts/img/logo.png differ diff --git a/submodules/mts/mts.js b/submodules/mts/mts.js new file mode 100644 index 0000000..2c99be7 --- /dev/null +++ b/submodules/mts/mts.js @@ -0,0 +1,48 @@ +define(function(require){ + var $ = require('jquery'); + + const CONFIG = { + submoduleName: 'mts', + i18n: [ 'en-US' ] + }; + + var app = { + requests: {}, + + subscribe: { + 'storagemgmt.fetchStorages': 'defineStorageMTS' + }, + + defineStorageMTS: function(args) { + var self = this, + storage_nodes = args.storages; + + var methods = { + getLogo: function () { + return self.getTemplate({ + name: 'logo', + submodule: CONFIG.submoduleName, + data: {} + }); + }, + + getFormElements: function (storageData) { + return self.getTemplate({ + name: 'formElements', + submodule: CONFIG.submoduleName, + data: storageData + }); + } + }; + + $.extend(true, storage_nodes, { + 'mts': methods + } + ); + + args.callback && args.callback(CONFIG) + } + }; + + return app; +}); diff --git a/submodules/mts/views/formElements.html b/submodules/mts/views/formElements.html new file mode 100644 index 0000000..ccdabd1 --- /dev/null +++ b/submodules/mts/views/formElements.html @@ -0,0 +1,26 @@ + + + + + + diff --git a/submodules/mts/views/logo.html b/submodules/mts/views/logo.html new file mode 100644 index 0000000..3e196af --- /dev/null +++ b/submodules/mts/views/logo.html @@ -0,0 +1 @@ +mts diff --git a/submodules/s3/i18n/en-US.json b/submodules/s3/i18n/en-US.json new file mode 100644 index 0000000..0c0bad4 --- /dev/null +++ b/submodules/s3/i18n/en-US.json @@ -0,0 +1,12 @@ +{ + "storagemgmt": { + "submodules": { + "s3": { + "nameLabel": "Name", + "bucketLabel": "Bucket", + "keyLabel": "Key", + "secretLabel": "Secret" + } + } + } +} diff --git a/submodules/s3/img/logo.png b/submodules/s3/img/logo.png new file mode 100644 index 0000000..2e25ac5 Binary files /dev/null and b/submodules/s3/img/logo.png differ diff --git a/submodules/s3/s3.js b/submodules/s3/s3.js new file mode 100644 index 0000000..ee15e6d --- /dev/null +++ b/submodules/s3/s3.js @@ -0,0 +1,48 @@ +define(function(require){ + var $ = require('jquery'); + + const CONFIG = { + submoduleName: 's3', + i18n: [ 'en-US' ] + }; + + var app = { + requests: {}, + + subscribe: { + 'storagemgmt.fetchStorages': 'defineStorageS3' + }, + + defineStorageS3: function(args) { + var self = this, + storage_nodes = args.storages; + + var methods = { + getLogo: function () { + return self.getTemplate({ + name: 'logo', + submodule: CONFIG.submoduleName, + data: {} + }); + }, + + getFormElements: function (storageData) { + return self.getTemplate({ + name: 'formElements', + submodule: CONFIG.submoduleName, + data: storageData + }); + } + }; + + $.extend(true, storage_nodes, { + 's3': methods + } + ); + + args.callback && args.callback(CONFIG) + } + }; + + return app; +}); diff --git a/submodules/s3/views/formElements.html b/submodules/s3/views/formElements.html new file mode 100644 index 0000000..b56aef8 --- /dev/null +++ b/submodules/s3/views/formElements.html @@ -0,0 +1,25 @@ + + + + + diff --git a/submodules/s3/views/logo.html b/submodules/s3/views/logo.html new file mode 100644 index 0000000..e3dc5f7 --- /dev/null +++ b/submodules/s3/views/logo.html @@ -0,0 +1 @@ +s3 diff --git a/views/field-path.html b/views/field-path.html new file mode 100644 index 0000000..08becf5 --- /dev/null +++ b/views/field-path.html @@ -0,0 +1,5 @@ +
+ + + {{ i18n.cancel }} +
diff --git a/views/item-settings.html b/views/item-settings.html new file mode 100644 index 0000000..feccf10 --- /dev/null +++ b/views/item-settings.html @@ -0,0 +1,17 @@ +
+
+
+
+ {{{ formElements }}} +
+ + +
+
+
+
+
diff --git a/views/layout.html b/views/layout.html new file mode 100644 index 0000000..9043438 --- /dev/null +++ b/views/layout.html @@ -0,0 +1,32 @@ +
+

{{ @root.i18n.storagemgmt.settingsTitle }}

+
+ {{#compare storages.length ">" 0}} + {{#each storages}} +
+
+ + + +
+ {{{ logo }}} +
+ {{ name }} +
+ + +
+
+
+
+ {{/each}} + {{/compare}} +
+
+ +
diff --git a/views/new-item.html b/views/new-item.html new file mode 100644 index 0000000..06e65e2 --- /dev/null +++ b/views/new-item.html @@ -0,0 +1,44 @@ +
+
+
+
+ {{#if storages.length}} + +
+ {{#each storages}} +
+
+ {{{ formElements }}} + +
+ + +
+
+
+ {{/each}} +
+ {{/if}} +
+
+
+
+ +