| @ -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 = $('<div class="storage-message ' + msgTypeClass + '">' + msg + '</div>') | |||||
| .appendTo($('.js-storage-msg-box')).hide().fadeIn(); | |||||
| $msg.animate({ | |||||
| backgroundColor: '#ffffff' | |||||
| }, 1000 | |||||
| ); | |||||
| window.setTimeout(function(){ | |||||
| $msg.fadeOut(400, function() { | |||||
| $msg.remove(); | |||||
| }) | |||||
| }, 4000); | |||||
| } | |||||
| }; | |||||
| return storageManager; | |||||
| }); | |||||
| @ -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." | |||||
| } | |||||
| } | |||||
| @ -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" | |||||
| } | |||||
| @ -0,0 +1,8 @@ | |||||
| define(function(require) { | |||||
| return { | |||||
| "storages": [ | |||||
| "s3", | |||||
| "mts" | |||||
| ] | |||||
| }; | |||||
| }); | |||||
| @ -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; } } | |||||
| @ -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; | |||||
| } | |||||
| } | |||||
| @ -0,0 +1,23 @@ | |||||
| <?xml version="1.0" encoding="utf-8"?> | |||||
| <!-- Generator: Adobe Illustrator 22.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0) --> | |||||
| <svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px" | |||||
| viewBox="0 0 98 35" style="enable-background:new 0 0 98 35;" xml:space="preserve"> | |||||
| <style type="text/css"> | |||||
| .st0{fill:#E30613;} | |||||
| .st1{fill:#FFFFFF;} | |||||
| </style> | |||||
| <path class="st0" d="M97.6,21.4c0,0-2.1,0.3-4.6,0.3c-3,0-3.6-1-3.6-3.3v-2.9c0-2.4,0.5-3.4,3.6-3.4c2.2,0,4.3,0.3,4.3,0.3L97.7,8 | |||||
| c0,0-2.7-0.4-5.1-0.4c-6.3,0-8.5,2.1-8.5,7.1v4.4c0,5.5,2.8,7.1,8.5,7.1c2.3,0,5.4-0.4,5.4-0.4L97.6,21.4z"/> | |||||
| <polygon class="st0" points="66.3,12.5 71.5,12.5 71.5,25.9 76.9,25.9 76.9,12.5 82.1,12.5 82.4,8 66,8 "/> | |||||
| <polygon class="st0" points="55.3,24.1 58.2,17.3 58.5,17.3 59.1,25.9 64.3,25.9 63.1,8 58.1,8 53.8,18.2 53.5,18.2 48.9,8 44,8 | |||||
| 42.8,25.9 47.9,25.9 48.4,17.3 48.7,17.3 51.9,24.1 "/> | |||||
| <g id="XMLID_1_"> | |||||
| <g> | |||||
| <path class="st0" d="M35,3.5v28c0,1.9-1.6,3.5-3.5,3.5h-28C1.6,35,0,33.4,0,31.5v-28C0,1.6,1.6,0,3.5,0h28C33.4,0,35,1.6,35,3.5z | |||||
| M17.5,30.4C24,30.4,27,25.2,27,20c0-7.3-5-15.5-9.5-15.5C13.1,4.5,8,12.8,8,20.1C8,25.2,10.9,30.4,17.5,30.4z"/> | |||||
| <path class="st1" d="M27,20c0,5.2-3,10.4-9.5,10.4c-6.6,0-9.5-5.2-9.5-10.3c0-7.3,5.1-15.6,9.5-15.6C22,4.5,27,12.7,27,20z"/> | |||||
| </g> | |||||
| <g> | |||||
| </g> | |||||
| </g> | |||||
| </svg> | |||||
| @ -0,0 +1,12 @@ | |||||
| { | |||||
| "storagemgmt": { | |||||
| "submodules": { | |||||
| "mts": { | |||||
| "nameLabel": "Name", | |||||
| "bucketLabel": "Bucket", | |||||
| "keyLabel": "Key", | |||||
| "secretLabel": "Secret" | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| @ -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; | |||||
| }); | |||||
| @ -0,0 +1,26 @@ | |||||
| <input type="hidden" name="handler" value="s3"/> | |||||
| <input type="hidden" name="settings.host" value="s3.cloud.mts.ru"/> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.nameLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{name}}" name="name"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.bucketLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.bucket}}" name="settings.bucket"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.keyLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.key}}" name="settings.key"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.secretLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.secret}}" name="settings.secret"> | |||||
| </div> | |||||
| </label> | |||||
| @ -0,0 +1 @@ | |||||
| <img src="/apps/storagemgmt/submodules/mts/img/logo.png" alt="mts" /> | |||||
| @ -0,0 +1,12 @@ | |||||
| { | |||||
| "storagemgmt": { | |||||
| "submodules": { | |||||
| "s3": { | |||||
| "nameLabel": "Name", | |||||
| "bucketLabel": "Bucket", | |||||
| "keyLabel": "Key", | |||||
| "secretLabel": "Secret" | |||||
| } | |||||
| } | |||||
| } | |||||
| } | |||||
| @ -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; | |||||
| }); | |||||
| @ -0,0 +1,25 @@ | |||||
| <input type="hidden" name="handler" value="s3"/> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.nameLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{name}}" name="name"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.bucketLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.bucket}}" name="settings.bucket"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.keyLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.key}}" name="settings.key"> | |||||
| </div> | |||||
| </label> | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.submodules.s3.secretLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="text" value="{{settings.secret}}" name="settings.secret"> | |||||
| </div> | |||||
| </label> | |||||
| @ -0,0 +1 @@ | |||||
| <img src="/apps/storagemgmt/submodules/s3/img/logo.png" alt="s3" /> | |||||
| @ -0,0 +1,5 @@ | |||||
| <div class="path-wrapper"> | |||||
| <input class="input-small" type="text" id="path" placeholder="{{ i18n.storagemgmt.pathPlaceholder }}" value="{{path}}"> | |||||
| <button class="save-path monster-button-fit monster-button-success">{{ i18n.save }}</button> | |||||
| <a href="javascript:void(0);" class="monster-link cancel-link">{{ i18n.cancel }}</a> | |||||
| </div> | |||||
| @ -0,0 +1,17 @@ | |||||
| <div class="storage-item-settings"> | |||||
| <div class="{{ type }} row-fluid"> | |||||
| <div class="span12"> | |||||
| <form class="form-horizontal js-storage-settings-form"> | |||||
| {{{ formElements }}} | |||||
| <div class="buttons-wrapper clearfix"> | |||||
| <button type="submit" class="btn btn-primary js-save" > | |||||
| {{ @root.i18n.storagemgmt.itemSettings.saveBtnText }} | |||||
| </button> | |||||
| <button type="button" class="btn js-cancel"> | |||||
| {{ @root.i18n.storagemgmt.itemSettings.cancelBtnText }} | |||||
| </button> | |||||
| </div> | |||||
| </form> | |||||
| </div> | |||||
| </div> | |||||
| </div> | |||||
| @ -0,0 +1,32 @@ | |||||
| <div id="storage_manager_wrapper" class="storage-manager-wrapper"> | |||||
| <h3>{{ @root.i18n.storagemgmt.settingsTitle }}</h3> | |||||
| <div class="js-storage-items storage-items-list"> | |||||
| {{#compare storages.length ">" 0}} | |||||
| {{#each storages}} | |||||
| <div class="js-storage-item storage-provider-wrapper {{#if isActive }}active-storage{{/if}}" data-type="{{type}}" data-uuid="{{id}}"> | |||||
| <div class="storage-choice"> | |||||
| <a href="" class="js-set-default-storage select-storage-link check-icon-wrapper"> | |||||
| <i class="check-icon fa fa-check-circle"></i> | |||||
| </a> | |||||
| <div class="logo-header"> | |||||
| {{{ logo }}} | |||||
| </div> | |||||
| <span class="js-storage-name storage-name">{{ name }}</span> | |||||
| <div class="right-section"> | |||||
| <a class="js-edit-storage edit-storage-link" href="#"><i class="fa fa-pencil-square-o"></i></a> | |||||
| <a class="js-remove-storage remove-storage-link" href="#"><i class="fa fa-trash-o"></i></a> | |||||
| </div> | |||||
| <div class="js-item-settings-wrapper item-settings-wrapper"></div> | |||||
| </div> | |||||
| </div> | |||||
| {{/each}} | |||||
| {{/compare}} | |||||
| </div> | |||||
| <div class="js-storage-msg-box"></div> | |||||
| <div class="buttons-wrapper"> | |||||
| <a href="" class="js-create-storage btn"> | |||||
| <i class="fa fa-plus"></i> | |||||
| {{ @root.i18n.storagemgmt.newStorageBtnText }} | |||||
| </a> | |||||
| </div> | |||||
| </div> | |||||
| @ -0,0 +1,44 @@ | |||||
| <div class="js-new-storage-item new-storage-item storage-provider-wrapper"> | |||||
| <div class="storage-choice"> | |||||
| <div class="storage-item-settings"> | |||||
| <div class="js-new-storage-tabs"> | |||||
| {{#if storages.length}} | |||||
| <ul> | |||||
| {{#each storages}} | |||||
| <li> | |||||
| <a href="#{{ type }}-new-item-content" class="storage-item-logo"> | |||||
| {{{ logo }}} | |||||
| </a> | |||||
| </li> | |||||
| {{/each}} | |||||
| </ul> | |||||
| <div class="tab-content"> | |||||
| {{#each storages}} | |||||
| <div class="js-tab-content-item {{ type }} tab-content-item" data-type="{{ type }}" id="{{ type }}-new-item-content"> | |||||
| <form class="form-horizontal js-storage-settings-form"> | |||||
| {{{ formElements }}} | |||||
| <label class="control-group"> | |||||
| <span class="control-label">{{ @root.i18n.storagemgmt.itemSettings.setDefaultLabel }}</span> | |||||
| <div class="controls"> | |||||
| <input type="checkbox" checked="checked" value="" name="set_default" /> | |||||
| </div> | |||||
| </label> | |||||
| <div class="buttons-wrapper clearfix"> | |||||
| <button type="submit" class="btn btn-primary js-save" > | |||||
| {{ @root.i18n.storagemgmt.newItem.saveNewItemText }} | |||||
| </button> | |||||
| <button type="button" class="btn js-cancel"> | |||||
| {{ @root.i18n.storagemgmt.newItem.cancelBtnText }} | |||||
| </button> | |||||
| </div> | |||||
| </form> | |||||
| </div> | |||||
| {{/each}} | |||||
| </div> | |||||
| {{/if}} | |||||
| </div> | |||||
| </div> | |||||
| </div> | |||||
| </div> | |||||