diff --git a/content-scripts.css b/content-scripts.css index 6246b33d7..565570135 100644 --- a/content-scripts.css +++ b/content-scripts.css @@ -1,4 +1,4 @@ - + /*----------------------------------------------------------------------------- >>> APPEARANCE ------------------------------------------------------------------------------- @@ -25,6 +25,7 @@ 6.1 Live chat 6.2 Playlist 6.3 Related videos + 6.4 Sidebar and thumbnail position 7.0 Footer -----------------------------------------------------------------------------*/ @@ -193,23 +194,17 @@ html[it-player-hide-cards=true] .ytp-ce-element, html[it-player-hide-cards=true] .ytp-ce-video, html[it-player-hide-cards=true] .ytp-cards-button, html[it-player-hide-cards=true] .ytp-cards-teaser, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:not(:hover) .ytp-ce-element, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:not(:hover) .ytp-ce-video, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:not(:hover) .ytp-cards-button, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:not(:hover) .ytp-cards-teaser +html[it-player-show-cards-on-mouse-hover=true] .ytp-cards-button { display: none !important; } -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:hover .ytp-ce-element, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:hover .ytp-ce-video, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:hover .ytp-cards-button, -html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:hover .ytp-cards-teaser +html[it-player-show-cards-on-mouse-hover=true] .html5-video-player:hover .iv-drawer { display: block !important; + transform: none; } - /*----------------------------------------------------------------------------- 2.4 Color -----------------------------------------------------------------------------*/ @@ -757,6 +752,19 @@ html[it-related-videos=collapsed].related-videos-collapsed #improvedtube-collaps box-shadow: inset 0 1px 0 #ddd; } +/*----------------------------------------------------------------------------- +6.4 Sidebar and thumbnail position +-----------------------------------------------------------------------------*/ + +html[it-sidebar-left='true'] #columns>#primary, +html[it-sidebar-left='true'] #head>#upnext /* also moving autoplay button to the outside as the glow bothers next to the video */ +{ + order:7; +} +html[it-thumbnails-right='true'] #dismissable>ytd-thumbnail +{ + order:7; +} /*----------------------------------------------------------------------------- 7.0 Footer @@ -767,7 +775,7 @@ html[it-hide-footer=true] #footer-container.yt-base-gutter { display: none !important; } - + /*----------------------------------------------------------------------------- >>> APPEARANCE: PLAYER SIZE ------------------------------------------------------------------------------- @@ -1285,7 +1293,7 @@ html[it-page-type="video"][it-header-position='hover_on_video_page'][it-player-s { margin-top: 14px !important; } - + .improvedtube-add-to-blacklist { font-size: 32px; @@ -1322,7 +1330,7 @@ html[it-page-type="video"][it-header-position='hover_on_video_page'][it-player-s { transform: scale(1.25); } - + /*----------------------------------------------------------------------------- >>> CHANNEL ------------------------------------------------------------------------------- @@ -1345,7 +1353,7 @@ html[it-channel-hide-featured-content=true] #secondary ytd-browse-secondary-cont width: fit-content; height: fit-content; } - + /*----------------------------------------------------------------------------- >>> GENERAL ------------------------------------------------------------------------------- @@ -1625,7 +1633,7 @@ html[it-remove-related-search-results=true] li > div.search-refinements { display: none; }*/ - + .it-rate-notify { position: fixed; @@ -1773,7 +1781,7 @@ html:not([it-page-type='video']) .it-rate-notify.it-rate-notify--show transform: scale(1); } } - + /*----------------------------------------------------------------------------- >>> PLAYER ------------------------------------------------------------------------------- @@ -2124,7 +2132,7 @@ html[it-player-crop-chapter-titles='false'] .ytp-chapter-title { width: auto !important; } - + html[it-youtube-version=new] #it-playlist-reverse { display: flex; @@ -2173,7 +2181,7 @@ html[it-youtube-version=old] #it-playlist-reverse svg { fill: #fff; } - + /*----------------------------------------------------------------------------- >>> SETTINGS ------------------------------------------------------------------------------- @@ -2344,7 +2352,7 @@ html[it-improvedtube-youtube-icon='draggable'] .it-btn { opacity: .75 !important; } - + #it-status { font-size: 5rem; @@ -2356,7 +2364,7 @@ html[it-improvedtube-youtube-icon='draggable'] .it-btn color: #0ff; text-shadow: 1px 1px 2px #000; } - + /*----------------------------------------------------------------------------- >>> THEMES ------------------------------------------------------------------------------- @@ -2444,7 +2452,7 @@ html[it-bluelight][it-youtube-version=old] body { visibility: visible !important; } - + [it-black-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #000!important; @@ -3077,7 +3085,7 @@ html[it-black-theme=true]:not([it-schedule=system_peference_dark]):not([it-sched --yt-chat-bubble-self-background-color: #000; --yt-blue-suggestive: transparent; } -} +} [it-dawn-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #d44d5c!important; @@ -3714,7 +3722,7 @@ html[it-dawn-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedu { -webkit-filter: grayscale(1) brightness(4); filter: grayscale(1) brightness(4); - } + } [it-youtube-version=old][it-default-dark-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-header-primary: rgb(34,38,42)!important; @@ -4277,7 +4285,7 @@ html[it-dawn-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedu background: rgb(19,19,19)!important; } -} +} [it-desert-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #faf9f9!important; @@ -4900,7 +4908,7 @@ html[it-desert-theme=true]:not([it-schedule=system_peference_dark]):not([it-sche --yt-chat-bubble-self-background-color: #faf9f9; --yt-blue-suggestive: transparent; } -} +} html[it-theme=true] #author-name.style-scope.yt-live-chat-message-input-renderer, html[it-theme=true] #content #message.yt-live-chat-text-message-renderer, html[it-theme=true] #delete-account-confirm, @@ -10137,7 +10145,7 @@ html[it-theme=true] .yt-uix-shelfslider-prev-arrow.yt-uix-tooltip.yt-sprite -webkit-filter: brightness(3)!important ; filter: brightness(3)!important; } - + [it-night-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #003459!important; @@ -10772,7 +10780,7 @@ html[it-night-theme=true]:not([it-schedule=system_peference_dark]):not([it-sched --yt-chat-bubble-self-background-color: #003459; --yt-blue-suggestive: transparent; } -} +} [it-plain-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #c7efcf!important; @@ -11395,7 +11403,7 @@ html[it-plain-theme=true]:not([it-schedule=system_peference_dark]):not([it-sched --yt-chat-bubble-self-background-color: #c7efcf; --yt-blue-suggestive: transparent; } -} +} [it-sunset-theme=true]:not([it-schedule=system_peference_dark]):not([it-schedule=system_peference_light]) { --yt-swatch-primary: #2f3364!important; diff --git a/content-scripts.js b/content-scripts.js index ccc91cc50..9cc4669cc 100644 --- a/content-scripts.js +++ b/content-scripts.js @@ -1,4 +1,4 @@ - + /*----------------------------------------------------------------------------- >>> CORE ------------------------------------------------------------------------------- @@ -184,7 +184,7 @@ ImprovedTube.init = function() { function withoutInjection(object) { youtubeHomePage__documentStart(object.youtube_home_page); } - + /*----------------------------------------------------------------------------- >>> EVENTS ------------------------------------------------------------------------------- @@ -263,6 +263,24 @@ ImprovedTube.events = function() { window.addEventListener('resize', function() { ImprovedTube.fitToWindow(); ImprovedTube.improvedtube_youtube_icon_resize(); + + setTimeout(function() { + if (document.querySelector('.html5-video-player video') && document.querySelector('.it-rotate-styles')) { + var video = document.querySelector('.html5-video-player video'), + transform = '', + rotate = (document.querySelector('.it-rotate-styles') && document.querySelector('.it-rotate-styles').textContent.match(/rotate\([0-9.]+deg\)/g) || [''])[0]; + + rotate = Number((rotate.match(/[0-9.]+/g) || [])[0]) || 0; + + transform += 'rotate(' + rotate + 'deg)'; + + if (rotate == 90 || rotate == 270) { + transform += ' scale(' + video.offsetHeight / video.offsetWidth + ')'; + } + + document.querySelector('.it-rotate-styles').textContent = '.html5-video-player:not(it-mini-player) video {transform:' + transform + '}'; + } + }, 500); }); window.addEventListener('scroll', function() { @@ -331,7 +349,7 @@ ImprovedTube.events = function() { } }, true); }; - + chrome.storage.local.get(function(items) { document.addEventListener('ImprovedTubeAnalyzer', function() { if (items.analyzer_activation !== false) { @@ -344,7 +362,7 @@ chrome.storage.local.get(function(items) { } }); }); - + /*----------------------------------------------------------------------------- >>> APPEARANCE ------------------------------------------------------------------------------- @@ -797,7 +815,7 @@ ImprovedTube.related_videos = function() { }, 260); } }; - + // TODO: HIGH CPU USAGE document.addEventListener('ImprovedTubeBlacklist', function(event) { @@ -1013,7 +1031,7 @@ ImprovedTube.blacklist = function() { } } }; - + /*----------------------------------------------------------------------------- >>> CHANNEL ------------------------------------------------------------------------------- @@ -1068,7 +1086,7 @@ ImprovedTube.channel_default_tab = function() { } } }; - + /*----------------------------------------------------------------------------- >>> GENERAL ------------------------------------------------------------------------------- @@ -1355,7 +1373,7 @@ ImprovedTube.mark_watched_videos = function() { } } }; - + /*----------------------------------------------------------------------------- >>> PLAYER ------------------------------------------------------------------------------- @@ -2127,7 +2145,7 @@ ImprovedTube.player_rotate_button = function() { onclick: function() { var video = document.querySelector('.html5-video-player video'), transform = '', - rotate = (video.style.transform.match(/rotate\([0-9.]+deg\)/g) || [''])[0]; + rotate = (document.querySelector('.it-rotate-styles') && document.querySelector('.it-rotate-styles').textContent.match(/rotate\([0-9.]+deg\)/g) || [''])[0]; rotate = Number((rotate.match(/[0-9.]+/g) || [])[0]) || 0; @@ -2143,12 +2161,23 @@ ImprovedTube.player_rotate_button = function() { transform += ' scale(' + video.offsetHeight / video.offsetWidth + ')'; } - video.style.transform = transform; + //video.style.transform = transform; + + if (!document.querySelector('.it-rotate-styles')) { + var styles = document.createElement('style'); + + styles.className = 'it-rotate-styles'; + + document.body.appendChild(styles); + } + + document.querySelector('.it-rotate-styles').textContent = '.html5-video-player:not(it-mini-player) video {transform:' + transform + '}'; }, title: 'Rotate' }); } else if (document.querySelector('.it-rotate-button')) { document.querySelector('.it-rotate-button').remove(); + document.querySelector('.it-rotate-styles').remove(); } }; @@ -2214,7 +2243,7 @@ ImprovedTube.player_loudness_normalization = function() { } } }; - + /*----------------------------------------------------------------------------- >>> PLAYLIST ------------------------------------------------------------------------------- @@ -2460,7 +2489,7 @@ ImprovedTube.playlist_up_next_autoplay = function(player) { player.querySelector('video').removeEventListener('timeupdate', ImprovedTube.playlist_up_next_autoplay_f, true); player.querySelector('video').addEventListener('timeupdate', ImprovedTube.playlist_up_next_autoplay_f, true); }; - + /*----------------------------------------------------------------------------- >>> SETTINGS ------------------------------------------------------------------------------- @@ -2694,7 +2723,7 @@ ImprovedTube.youtube_language = function() { location.reload(); }, 100); }; - + /*----------------------------------------------------------------------------- >>> SHORTCUTS ------------------------------------------------------------------------------- @@ -3087,7 +3116,7 @@ ImprovedTube.shortcuts = function() { capture: true }); }; - + /*----------------------------------------------------------------------------- >>> THEMES ------------------------------------------------------------------------------- @@ -3325,7 +3354,7 @@ ImprovedTube.themeEditor = function() { document.documentElement.appendChild(style); } - + /*----------------------------------------------------------------------------- >>> VOLUME MIXER ------------------------------------------------------------------------------- @@ -3336,7 +3365,7 @@ ImprovedTube.themeEditor = function() { 1.0 Inject -----------------------------------------------------------------------------*/ -ImprovedTube.volumeMixer = function() {}; +ImprovedTube.volumeMixer = function() {}; /*----------------------------------------------------------------------------- >>> FUNCTIONS ------------------------------------------------------------------------------- @@ -3438,7 +3467,7 @@ ImprovedTube.pageType = function() { chrome.runtime.sendMessage({ enabled: true -}); +}); /*----------------------------------------------------------------------------- >>> INJECTION ------------------------------------------------------------------------------- @@ -3581,7 +3610,7 @@ chrome.runtime.onMessage.addListener(function(request, sender, sendResponse) { }, 250); } }); - + /*----------------------------------------------------------------------------- >>> MIGRATION ------------------------------------------------------------------------------- @@ -4118,7 +4147,7 @@ chrome.storage.local.get(function(object) { location.reload(); } -}); +}); /*----------------------------------------------------------------------------- >>> MUTATIONS ------------------------------------------------------------------------------- diff --git a/manifest.json b/manifest.json index d8f2113f2..0121921fa 100644 --- a/manifest.json +++ b/manifest.json @@ -2,7 +2,7 @@ "manifest_version": 2, "short_name" : "ImprovedTube", "name": "Improve YouTube! (Open-Source for YouTube)", - "version": "3.80", + "version": "3.100", "icons": { "16": "assets/icons/16.png", "32": "assets/icons/32.png", diff --git a/popup.js b/popup.js index f9c2c2dc5..5ad669c1d 100644 --- a/popup.js +++ b/popup.js @@ -1,4 +1,4 @@ - + var Menu = { header: { type: 'header', @@ -95,7 +95,7 @@ var Menu = { } } }; - + Menu.main = { type: 'main', appearanceId: 'home', @@ -122,7 +122,7 @@ Menu.main = { found_a_bug: { type: 'button', class: 'satus-button--found-a-bug', - label: 'foundABug', + label: 'found a bug?', title: '/ImprovedTube/ImprovedTube', onclick: function(){ window.open('https://github.com/ImprovedTube/ImprovedTube/issues/new', '_blank'); @@ -154,7 +154,7 @@ Menu.main = { } } }; - + Menu.header.section_end.button_vert.onClickRender.active_features = { type: 'folder', before: '', @@ -214,7 +214,7 @@ Menu.header.section_end.button_vert.onClickRender.active_features = { }); } } -}; +}; Menu.header.section_end.button_vert.onClickRender.mixer = { type: 'folder', before: '', @@ -329,7 +329,7 @@ Menu.header.section_end.button_vert.onClickRender.mixer = { } } }; - + Menu.header.section_end.button_vert.onClickRender.settings = { type: 'folder', before: '', @@ -1201,7 +1201,7 @@ Menu.header.section_end.button_vert.onClickRender.settings = { } } }; - + Menu.main.section.general = { type: 'folder', before: '', @@ -1295,7 +1295,7 @@ Menu.main.section.general = { } } }; - + Menu.main.section.appearance = { type: 'folder', before: '', @@ -1355,11 +1355,11 @@ Menu.main.section.appearance = { section: { type: 'section', - player_hide_annotations: { + /*player_hide_annotations: { type: 'switch', label: 'hideAnnotations', tags: 'hide,remove,elements' - }, + },*/ player_hide_cards: { type: 'switch', label: 'hideCards', @@ -1575,7 +1575,30 @@ Menu.main.section.appearance = { section: { type: 'section', - + sidebar_left: { + type: 'switch', + label: 'Sidebar on the Left' + }, + thumbnails_right: { + type: 'switch', + label: 'Thumbnails on the Right' + }, + related_videos: { + type: 'select', + label: 'relatedVideos', + options: [{ + label: 'normal', + value: 'normal' + }, { + label: 'collapsed', + value: 'collapsed' + }, { + label: 'hidden', + value: 'hidden' + }], + tags: 'right' + }, + livechat: { type: 'select', label: 'liveChat', @@ -1606,21 +1629,6 @@ Menu.main.section.appearance = { hide_playlist: { type: 'switch', label: 'hidePlaylist' - }, - related_videos: { - type: 'select', - label: 'relatedVideos', - options: [{ - label: 'normal', - value: 'normal' - }, { - label: 'collapsed', - value: 'collapsed' - }, { - label: 'hidden', - value: 'hidden' - }], - tags: 'right' } } }, @@ -1665,7 +1673,7 @@ Menu.main.section.appearance = { } } }; - + function themePopupChange() { if (Satus.storage.get('red_popup_theme') === true) { document.documentElement.setAttribute('popup-theme', 'red'); @@ -2080,7 +2088,7 @@ Menu.main.section.themes = { onchange: themeChange } }; - + Menu.main.section.player = { type: 'folder', before: '', @@ -2358,7 +2366,7 @@ Menu.main.section.player = { } } }; - + Menu.main.section.playlist = { type: 'folder', before: '', @@ -2397,7 +2405,7 @@ Menu.main.section.playlist = { label: 'shuffle' } } -}; +}; Menu.main.section.channel = { type: 'folder', before: '', @@ -2432,7 +2440,7 @@ Menu.main.section.channel = { label: 'hideFeaturedContent' } } -}; +}; Menu.main.section.shortcuts = { type: 'folder', before: '', @@ -2657,7 +2665,7 @@ Menu.main.section.shortcuts = { } } }; - + Menu.main.section.blacklist = { type: 'folder', before: '', @@ -2819,7 +2827,7 @@ Menu.main.section.blacklist = { } } }; - + Menu.main.section.analyzer = { type: 'folder', before: '', @@ -2966,7 +2974,7 @@ Menu.main.section.analyzer = { } } }; - + chrome.storage.local.get(function(items) { for (var key in items) { document.documentElement.setAttribute('it-' + key.replace(/_/g, '-'), items[key]); diff --git a/satus.css b/satus.css index 7f032cc4c..bf6648b33 100644 --- a/satus.css +++ b/satus.css @@ -1,291 +1,187 @@ -/*-------------------------------------------------------------- ->>> BUTTON ---------------------------------------------------------------*/ - -.satus-button -{ - font-size: inherit; - - position: relative; - - display: flex; - - margin: 0; - - cursor: pointer; - - color: inherit; - border: none; - outline: none; - background-color: var(--satus-theme-button); - - -webkit-tap-highlight-color: transparent; - align-items: center; - -webkit-appearance: none; -} - -.satus-button::before -{ - position: absolute; - top: 0; - left: 0; - - width: 100%; - height: 100%; - - content: ''; - - opacity: 0; - background-color: var(--satus-theme-ripple); -} - -.satus-button:focus::before, -.satus-button:hover::before -{ - opacity: 1; -} - -.satus-button__icon -{ - width: 24px; - height: 24px; -} - -.satus-button svg -{ - width: 100%; - - color: inherit; - - fill: var(--satus-theme-primary); -} - -.satus-button__icon + .satus-button__label -{ - margin-left: 8px; -} - -/*-------------------------------------------------------------- ->>> COLOR PICKER ---------------------------------------------------------------*/ - -.satus-color-picker -{ - font-size: inherit; - - position: relative; - - display: flex; - - box-sizing: border-box; - margin: 0; - - cursor: pointer; - - color: inherit; - border: none; - outline: none; - background-color: var(--satus-theme-button); - - justify-content: space-between; - -webkit-tap-highlight-color: transparent; - align-items: center; - -webkit-appearance: none; -} +/*----------------------------------------------------------------------------- +>>> ANIMATIONS +-----------------------------------------------------------------------------*/ -.satus-color-picker__value +.satus-animation--fade-in { - width: 24px; - height: 24px; - - border-radius: 50%; + animation: fadeIn 250ms; } -.satus-dialog--color-picker .satus-scrollbar__content +.satus-animation--fade-out { - display: flex; - flex-direction: column; - - align-items: center; + animation: fadeOut 250ms; } -.satus-dialog--color-picker .satus-button +.satus-animation--fade-in-left { - box-sizing: unset; - width: 24px; - height: 24px; - margin: 0 0 8px; - padding: 8px; - - border-radius: 50%; -} - -.satus-dialog--color-picker .satus-button:hover -{ - background-color: var(--satus-theme-ripple); + animation: fadeInLeft 250ms; } -.satus-dialog--color-picker .satus-button::before +.satus-animation--fade-in-right { - content: unset; + animation: fadeInRight 250ms; } -.satus-dialog--color-picker .satus-button svg +.satus-animation--fade-out-left { - fill: transparent; - stroke: var(--satus-theme-primary); + animation: fadeOutLeft 250ms; } -.satus-dialog--color-picker canvas +.satus-animation--fade-out-right { - width: 200px; - height: 200px; - margin: 0 0 16px; + animation: fadeOutRight 250ms; } - -/*-------------------------------------------------------------- ->>> DIALOG ---------------------------------------------------------------*/ -.satus-dialog +@keyframes fadeIn { - position: absolute; - z-index: 100; - top: 0; - left: 0; - - display: flex; - - width: 100%; - height: 100vh; + from + { + opacity: 0; + } + to + { + transform: translateX(0%); - justify-content: center; - align-items: center; + opacity: 1; + } } -.satus-dialog__scrim +@keyframes fadeOut { - position: absolute; - top: 0; - left: 0; - - width: 100%; - height: 100%; - - animation: fadeIn 150ms linear forwards; - - opacity: 0; - background: rgba(25,25,25,.2); + from + { + opacity: 1; + } + to + { + transform: translateX(0%); - fill: var(--satus-theme-dialog-text); - backdrop-filter: blur(8px); + opacity: 0; + } } -.satus-dialog__surface +@keyframes fadeInLeft { - font-size: 14px; - - display: flex; - flex-direction: column; - - width: 95%; - min-width: 240px; - max-width: 560px; - max-height: 80%; - padding: 8px 0; - - transform: scale(.8); - animation: zoomIn 150ms linear forwards; - animation-delay: 20ms; - - opacity: 0; - color: var(--satus-theme-dialog-text); - border-radius: 6px; - background-color: var(--satus-theme-dialog); - box-shadow: inset 0 -1px 1px 1px rgba(0,0,0,.1), 0 2px 6px rgba(0, 0, 0, .15); -} + from + { + transform: translateX(-10%); -.satus-dialog--closing .satus-dialog__scrim -{ - animation: fadeOut 70ms linear forwards; -} + opacity: 0; + } + to + { + transform: translateX(0%); -.satus-dialog--closing .satus-dialog__surface -{ - animation: zoomOut 70ms linear forwards; + opacity: 1; + } } -@keyframes fadeIn +@keyframes fadeInRight { from { + transform: translateX(10%); + opacity: 0; } to { + transform: translateX(0%); + opacity: 1; } } -@keyframes fadeOut +@keyframes fadeOutLeft { from { + transform: translateX(0%); + opacity: 1; } to { + transform: translateX(-10%); + opacity: 0; } } -@keyframes zoomIn +@keyframes fadeOutRight { from { - transform: scale(.8); + transform: translateX(0%); - opacity: 0; + opacity: 1; } to { - transform: scale(1); + transform: translateX(10%); - opacity: 1; + opacity: 0; } } + +/*----------------------------------------------------------------------------- +>>> THEMES +------------------------------------------------------------------------------- +1.0 Default +-----------------------------------------------------------------------------*/ -@keyframes zoomOut +html { - from - { - transform: scale(1); + --satus-theme-primary: #f6b465; - opacity: 1; - } - to - { - transform: scale(.8); + --satus-theme-dialog: #f7f7f6; + --satus-theme-dialog-text: #777; + + --satus-theme-header: #fff; + --satus-theme-header-text: #777; + + --satus-theme-main: #f7f7f6; + --satus-theme-main-text: #777; + + --satus-theme-section: #fff; + + --satus-theme-button: transparent; + + --satus-theme-scrollbar: rgba(0, 0, 0, .2); + --satus-theme-scrollbar-focus: rgba(0, 0, 0, .4); + + --satus-theme-tooltip: rgba(0, 0, 0, .4); - opacity: 0; - } + --satus-theme-ripple: rgba(0, 0, 0, .04); + + + + + --satus-theme-tabs-background: #efefef; + --satus-theme-tabs-border: #dfdfdf; } /*-------------------------------------------------------------- ->>> FOLDER +>>> TABLE --------------------------------------------------------------*/ -.satus-folder +.satus-table { - text-align: left + font-size: .875rem; + + position: relative; + + overflow: hidden; + display: flex; + flex-direction: column; + box-sizing: border-box; + + border: 1px solid #ececec; + border-radius: 6px; } - -/*-------------------------------------------------------------- ->>> HEADER ---------------------------------------------------------------*/ -.satus-header +.satus-table__head { position: relative; z-index: 1; @@ -294,246 +190,287 @@ box-sizing: border-box; width: 100%; - min-height: 58px; + height: 44px; padding: 0 8px; - color: var(--satus-theme-header-text); - background-color: var(--satus-theme-header); - box-shadow: 0 0 3px rgba(0,0,0,.1); - - fill: var(--satus-theme-header-text); + background: #fff; + box-shadow: 0 2px 4px rgba(0, 0, 0, .1); } -.satus-header .satus-text--title +.satus-table__head > div { - font-size: 15px; - font-weight: 400; + font-weight: 600; + + display: inline-flex; + + box-sizing: border-box; + padding: 0 8px 0 0; + + opacity: .7; + + flex-shrink: 0; + align-items: center; } - -/*-------------------------------------------------------------- ->>> LIST ---------------------------------------------------------------*/ -.satus-list--compact, -.satus-list--compact li +.satus-table__head > div:last-child { - margin: 0; padding: 0; +} - list-style: none; +.satus-table__head > div:hover +{ + cursor: pointer; + + opacity: 1; } - -/*----------------------------------------------------------------------------- ->>> MAIN -------------------------------------------------------------------------------- -1.0 General -2.0 Animations - 2.1 Opening - 2.2 Closing ------------------------------------------------------------------------------*/ -/*----------------------------------------------------------------------------- -1.0 General ------------------------------------------------------------------------------*/ +.satus-table__head > div[data-sorting=false]:hover +{ + cursor: default; +} -.satus-main +.satus-table__head > div > span { position: relative; +} - overflow: hidden; +.satus-table__head > div[data-sorting=asc] > span::after +{ + position: absolute; + top: calc(50% - 2px); + right: -14px; - width: 100%; - height: 100%; + width: 0; + height: 0; - color: var(--satus-theme-main-text); - background-color: var(--satus-theme-main); + content: ''; - fill: var(--satus-theme-main-text); + border-right: 4px solid transparent; + border-bottom: 5px solid currentColor; + border-left: 4px solid transparent; } -.satus-main__container +.satus-table__head > div[data-sorting=desc] > span::after { position: absolute; - top: 0; - left: 0; - - overflow: auto; - - box-sizing: border-box; - width: 100%; - height: 100%; + top: calc(50% - 2px); + right: -13px; - transition: 250ms; + width: 0; + height: 0; - background-color: var(--satus-theme-main); + content: ''; - will-change: transform; + border-top: 5px solid currentColor; + border-right: 4px solid transparent; + border-left: 4px solid transparent; } +.satus-table__body +{ + overflow: hidden; + flex: 1; -/*----------------------------------------------------------------------------- -2.0 Animations ------------------------------------------------------------------------------*/ + width: 100%; + height: calc(100% - 48px); -.satus-main__container--fade-in-left -{ - animation: fadeInLeft 250ms; + background: #fff; } -.satus-main__container--fade-in-right +.satus-table__row { - animation: fadeInRight 250ms; + display: flex; + + padding: 0 8px; + + box-sizing: border-box; + + border-bottom: 1px solid rgba(0, 0, 0, 0.1); + + flex-wrap: wrap; } -.satus-main__container--fade-out-left +.satus-table__row:nth-child(2n) { - animation: fadeOutLeft 250ms; + background: rgba(0,0,0,.03) } -.satus-main__container--fade-out-right +.satus-table__cell { - animation: fadeOutRight 250ms; -} + overflow: hidden; -@keyframes fadeInLeft -{ - from - { - transform: translateX(-10%); + box-sizing: border-box; + padding: 8px 8px 8px 0; - opacity: 0; - } - to - { - transform: translateX(0%); + white-space: nowrap; + word-wrap: break-word; + -webkit-hyphens: auto; + -moz-hyphens: auto; + hyphens: auto; - opacity: 1; - } + overflow-wrap: break-word; + flex-shrink: 0; + + text-overflow: ellipsis; } -@keyframes fadeInRight +.satus-table__cell:last-child { - from - { - transform: translateX(10%); - - opacity: 0; - } - to - { - transform: translateX(0%); - - opacity: 1; - } + padding-right: 0; } -@keyframes fadeOutLeft +.satus-table__cell input { - from - { - transform: translateX(0%); + width: 100%; + margin: 5px 0; + border: 1px solid #d2d2d2; + box-sizing: border-box; + padding: 1px 2px; + border-radius: 4px; + height: calc(100% - 10px); + outline: none +} - opacity: 1; - } - to - { - transform: translateX(-10%); - opacity: 0; - } -} -@keyframes fadeOutRight +.satus-table__paging > button { - from - { - transform: translateX(0%); + min-width: 32px; + height: 32px; + padding: 0 8px; - opacity: 1; - } - to - { - transform: translateX(10%); + cursor: pointer; - opacity: 0; - } + color: var(--satus-theme-on-surface, #555); + border: none; + background: transparent; +} + +.satus-table__paging > button.active +{ + color: #2979ff; } -.satus-scrollbar +/*--------------------------------------------------------------- +>>> HEADER +----------------------------------------------------------------- +1.0 Basic +2.0 Buttons + 2.1 Back +3.0 Title +4.0 Vertical menu +---------------------------------------------------------------*/ + +/*--------------------------------------------------------------- +1.0 BASIC +---------------------------------------------------------------*/ + +.satus-header { position: relative; + z-index: 1; - overflow: hidden; + display: flex; + box-sizing: border-box; width: 100%; - height: 100%; + min-height: 58px; + padding: 0 8px; + + color: var(--satus-theme-header-text); + background-color: var(--satus-theme-header); + box-shadow: 0 0 3px rgba(0,0,0,.1); + + fill: var(--satus-theme-header-text); } -.satus-scrollbar__wrapper -{ - overflow-y: scroll; - width: 150%; - height: 100%; +/*--------------------------------------------------------------- +2.0 BUTTONS +---------------------------------------------------------------*/ + +.satus-header .satus-button +{ + width: 36px; + height: 36px; + padding: 6px; } -.satus-scrollbar__thumb +.satus-header .satus-button::before { - position: absolute; - z-index: 999; - top: 0; - right: 0; + border-radius: 50%; +} - min-height: 32px; - padding: 0 2px; - transition: background-color 200ms, opacity 300ms; +/*--------------------------------------------------------------- +3.0 BACK BUTTON +---------------------------------------------------------------*/ - opacity: 0; +.satus-button--back +{ + position: absolute; } -.satus-scrollbar__thumb::after +body[data-appearance='home'] .satus-button--back { - display: block; + visibility: hidden; +} - width: 3px; - height: 100%; - content: ''; +/*--------------------------------------------------------------- +3.0 TITLE +---------------------------------------------------------------*/ - background-color: var(--satus-theme-scrollbar); +.satus-header .satus-text--title +{ + font-size: 15px; + font-weight: 400; + + position: absolute; + left: 56px; } -.satus-scrollbar__thumb:hover::after +body[data-appearance='home'] .satus-text--title { - background-color: var(--satus-theme-scrollbar-focus); + left: 14px; } -.satus-scrollbar.active .satus-scrollbar__thumb, -.satus-scrollbar__thumb:hover + +/*--------------------------------------------------------------- +4.0 VERTICAL MENU +---------------------------------------------------------------*/ + +.satus-dialog--vertical-menu .satus-dialog__surface { - transition: background-color 200ms, opacity 100ms; + position: absolute; + top: 8px; + right: 8px; + left: auto; - opacity: 1; + min-width: 180px; + max-width: 220px; + + transform-origin: right top; } - -/*-------------------------------------------------------------- ->>> SECTION ---------------------------------------------------------------*/ -.satus-section +.satus-dialog--vertical-menu .satus-button, +.satus-dialog--vertical-menu .satus-folder { - display: inline-flex; - - box-sizing: border-box; + width: 100%; + height: 36px; + padding: 0 16px; - align-items: center; - flex: 1; + text-align: left; } -.satus-main .satus-section +.satus-dialog--vertical-menu .satus-folder svg, +.satus-dialog--vertical-menu .satus-button svg { - background-color: var(--satus-theme-section); + width: 20px; + height: 18px; + margin: 0 14px 0 0; + + opacity: .75; } /*-------------------------------------------------------------- @@ -602,8 +539,17 @@ opacity: 1; } } + +.satus-section--align-end +{ + justify-content: flex-end; +} -.satus-shortcut +/*-------------------------------------------------------------- +>>> SWITCH +--------------------------------------------------------------*/ + +.satus-switch { position: relative; @@ -612,209 +558,285 @@ box-sizing: border-box; width: 100%; height: 48px; - min-height: 48px; padding: 0 16px; cursor: pointer; + -webkit-user-select: none; + -moz-user-select: -moz-none; + user-select: none; + + background-color: transparent; - justify-content: space-between; align-items: center; + justify-content: space-between; } -.satus-shortcut:hover +.satus-switch:hover { background-color: var(--satus-theme-ripple); } -.satus-shortcut > .label +.satus-switch__input { - margin: 0 0 4px 0; -} + position: absolute; + z-index: 1; + top: 0; + left: 0; -.satus-shortcut > .value -{ - font-size: 13px; + width: 100%; + height: 100%; + margin: 0; + padding: 0; - opacity: .5; + opacity: 0; + outline: none; + + -webkit-appearance: none; } -.satus-shortcut-dialog-label -{ - font-size: 16px; - font-weight: 500; - box-sizing: border-box; - width: 100%; - margin: 4px 0 10px; - padding: 0 16px; -} +/* LABEL */ -.satus-shortcut__canvas +.satus-switch__label { - display: flex; + padding: 0 16px 0 0; +} - box-sizing: border-box; - width: 100%; - height: 68px; - padding: 16px; - background: #dedede; +/* TRACK*/ - align-items: center; +.satus-switch__track +{ + position: relative; + + width: 32px; + min-width: 32px; + height: 18px; + + border-radius: 18px; + background: #bdbdbd; } -.satus-shortcut__key +.satus-switch__track::before { - display: flex; + position: absolute; + top: -7px; + left: -7px; - box-sizing: border-box; - min-width: 32px; + width: 32px; height: 32px; - padding: 4px 8px; - border-radius: 4px; - background: #fff; - box-shadow: 0 1px 3px rgba(0,0,0,.15), inset 0 -3px 0 rgba(0,0,0,.1); + content: ''; + transition: left .1s cubic-bezier(.4, 0, .2, 1), transform 200ms, background-color 200ms; + transform: scale(0); - align-items: center; - justify-content: center; + opacity: 0; + border-radius: 50%; + background-color: #000; + + will-change: left, transform, background-color; } -.satus-shortcut__plus +.satus-switch__input:checked + .satus-switch__track::before { - position: relative; + left: 8px; - width: 12px; - height: 12px; - margin: 8px; + background-color: var(--satus-theme-primary, #54d2a3); } -.satus-shortcut__plus::before +.satus-switch__input:focus + .satus-switch__track::before { - position: absolute; - top: 0; - left: 5px; - - width: 2px; - height: 12px; + transform: scale(1); - content: ''; + opacity: .08; +} - background-color: #aaa; +.satus-switch__input:checked:focus + .satus-switch__track::before +{ + opacity: .25; } -.satus-shortcut__plus::after +.satus-switch__track::after { position: absolute; - top: 5px; - left: 0; - width: 12px; - height: 2px; + width: 14px; + height: 14px; content: ''; + transition: transform .1s cubic-bezier(.4, 0, .2, 1); + transform: translate(2px, 2px); + + border-radius: 50%; + background-color: #fff; + + will-change: transform; +} - background-color: #aaa; +.satus-switch__input:checked + .satus-switch__track +{ + background: var(--satus-theme-primary, #54d2a3); } -.satus-shortcut__mouse +.satus-switch__input:checked + .satus-switch__track::after +{ + transform: translate(16px, 2px); +} + +/*-------------------------------------------------------------- +>>> TABS +--------------------------------------------------------------*/ + +.satus-tabs__bar { position: relative; display: flex; + overflow: hidden; - width: 28px; - height: 36px; + box-sizing: border-box; + width: calc(100% - 16px); + margin: 16px 8px 4px; - border-radius: 50%; - border-top-left-radius: 12px; - border-top-right-radius: 12px; - background: #fff; - box-shadow: 0 1px 3px rgba(0,0,0,.15), inset 0 -3px 0 rgba(0,0,0,.1); + border: 1px solid var(--satus-theme-tabs-border); + border-radius: 18px; + background-color: var(--satus-theme-tabs-background); } -.satus-shortcut__mouse > div +.satus-tabs__bar > div:not(.satus-tabs__bar--select) { - position: absolute; - top: 0; - left: 13px; + z-index: 1; - width: 2px; - height: 11px; + display: flex; - border-radius: 2px; - background: #ccc; + height: 32px; + + cursor: pointer; + transition: opacity .25s; + + opacity: .5; + + flex: 1; + align-items: center; + justify-content: center; } -.satus-shortcut__mouse::before +.satus-tabs__bar > div.active +{ + opacity: 1; +} + +.satus-tabs__bar--select { position: absolute; - top: -4px; - left: 21px; + z-index: 0; + left: 0; - width: 2px; - height: 18px; + width: 50%; + height: 32px; - content: ''; + transition: left .25s; - background: #f96754; + border-radius: 18px; + background: var(--satus-theme-header); + box-shadow: 1px 0 4px rgba(0,0,0,.075); } -.satus-shortcut__mouse.false::after +.satus-tabs__main { - position: absolute; - top: -12px; - left: 17px; + position: relative +} - width: 0; - height: 0; +.satus-tabs__main > div +{ + display: inline-block; + width: 100%; + min-height: 100%; +} - content: ''; +.satus-tabs__main > .old +{ + position: absolute; + left: 0; + top: 0; +} + +.satus-tabs__tab.satus-animation--fade-out-left:not(.old) +{ + z-index: 1 +} + +/*-------------------------------------------------------------- +>>> FOLDER +--------------------------------------------------------------*/ - border-right: 5px solid transparent; - border-bottom: 8px solid #f96754; - border-left: 5px solid transparent; +.satus-folder +{ + text-align: left } + +/*-------------------------------------------------------------- +>>> TEXT +--------------------------------------------------------------*/ -.satus-shortcut__mouse.true::after +.satus-text-field { - position: absolute; - top: 14px; - left: 17px; - - width: 0; - height: 0; + font: inherit; - content: ''; + box-sizing: border-box; + width: 100%; + margin: 0; + padding: 0; + padding: 0 8px; - border-top: 8px solid #f96754; - border-right: 5px solid transparent; - border-left: 5px solid transparent; + color: inherit; + border: none; + outline: none; + background: unset; } + +/*----------------------------------------------------------------------------- +>>> MAIN +------------------------------------------------------------------------------- +1.0 General +-----------------------------------------------------------------------------*/ -.satus-section_shortcut +/*----------------------------------------------------------------------------- +1.0 General +-----------------------------------------------------------------------------*/ + +.satus-main { + position: relative; + + overflow: hidden; + width: 100%; - margin: 8px 0 0; + height: 100%; - justify-content: flex-end; + color: var(--satus-theme-main-text); + background-color: var(--satus-theme-main); + + fill: var(--satus-theme-main-text); } -.satus-button_shortcut +.satus-main__container { - font-weight: 500; + position: absolute; + top: 0; + left: 0; - overflow: hidden; + overflow: auto; - height: 28px; - min-height: 28px; - margin-right: 2px; - padding: 4px 8px; + box-sizing: border-box; + width: 100%; + height: 100%; - text-transform: uppercase; + transition: 250ms; - color: #f96754; - border-radius: 4px; + background-color: var(--satus-theme-main); + + will-change: transform; } /*-------------------------------------------------------------- @@ -987,416 +1009,684 @@ visibility: visible; } -.satus-slider__ring +.satus-slider__ring +{ + position: absolute; + top: -11px; + right: -18px; + + width: 24px; + height: 24px; + + transition: 100ms; + transform: scale(0); + + opacity: 0; + border-radius: 50%; + background-color: var(--satus-theme-primary); +} + +.satus-slider__range:focus + .satus-slider__container .satus-slider__ring +{ + transform: scale(1); + + opacity: .25; +} + +.satus-scrollbar +{ + position: relative; + + overflow: hidden; + + width: 100%; + height: 100%; +} + +.satus-scrollbar__wrapper +{ + overflow-y: scroll; + + width: 150%; + height: 100%; +} + +.satus-scrollbar__content +{ + min-height: 100%; +} + +.satus-scrollbar__thumb +{ + position: absolute; + z-index: 999; + top: 0; + right: 0; + + min-height: 32px; + padding: 0 2px; + + transition: background-color 200ms, opacity 300ms; + + opacity: 0; +} + +.satus-scrollbar__thumb::after +{ + display: block; + + width: 3px; + height: 100%; + + content: ''; + + background-color: var(--satus-theme-scrollbar); +} + +.satus-scrollbar__thumb:hover::after +{ + background-color: var(--satus-theme-scrollbar-focus); +} + +.satus-scrollbar.active .satus-scrollbar__thumb, +.satus-scrollbar__thumb:hover +{ + transition: background-color 200ms, opacity 100ms; + + opacity: 1; +} + +/*-------------------------------------------------------------- +>>> LIST +--------------------------------------------------------------*/ + +.satus-list--compact, +.satus-list--compact li +{ + margin: 0; + padding: 0; + + list-style: none; +} + +/*-------------------------------------------------------------- +>>> DIALOG +--------------------------------------------------------------*/ + +.satus-dialog +{ + position: absolute; + z-index: 100; + top: 0; + left: 0; + + display: flex; + + width: 100%; + height: 100vh; + + justify-content: center; + align-items: center; +} + +.satus-dialog__scrim +{ + position: absolute; + top: 0; + left: 0; + + width: 100%; + height: 100%; + + animation: fadeIn 150ms linear forwards; + + opacity: 0; + background: rgba(25,25,25,.2); + + fill: var(--satus-theme-dialog-text); + backdrop-filter: blur(8px); +} + +.satus-dialog__surface +{ + font-size: 14px; + + display: flex; + flex-direction: column; + + width: 95%; + min-width: 240px; + max-width: 560px; + max-height: 80%; + padding: 8px 0; + + transform: scale(.8); + animation: zoomIn 150ms linear forwards; + animation-delay: 20ms; + + opacity: 0; + color: var(--satus-theme-dialog-text); + border-radius: 6px; + background-color: var(--satus-theme-dialog); + box-shadow: inset 0 -1px 1px 1px rgba(0,0,0,.1), 0 2px 6px rgba(0, 0, 0, .15); +} + +.satus-dialog--closing .satus-dialog__scrim +{ + animation: fadeOut 70ms linear forwards; +} + +.satus-dialog--closing .satus-dialog__surface +{ + animation: zoomOut 70ms linear forwards; +} + +@keyframes fadeIn +{ + from + { + opacity: 0; + } + to + { + opacity: 1; + } +} + +@keyframes fadeOut +{ + from + { + opacity: 1; + } + to + { + opacity: 0; + } +} + +@keyframes zoomIn { - position: absolute; - top: -11px; - right: -18px; - - width: 24px; - height: 24px; + from + { + transform: scale(.8); - transition: 100ms; - transform: scale(0); + opacity: 0; + } + to + { + transform: scale(1); - opacity: 0; - border-radius: 50%; - background-color: var(--satus-theme-primary); + opacity: 1; + } } -.satus-slider__range:focus + .satus-slider__container .satus-slider__ring +@keyframes zoomOut { - transform: scale(1); + from + { + transform: scale(1); - opacity: .25; + opacity: 1; + } + to + { + transform: scale(.8); + + opacity: 0; + } } /*-------------------------------------------------------------- ->>> SWITCH +>>> BUTTON --------------------------------------------------------------*/ -.satus-switch +.satus-button { + font-size: inherit; + font-family: inherit; + position: relative; display: flex; - box-sizing: border-box; - width: 100%; - height: 48px; - padding: 0 16px; + margin: 0; cursor: pointer; - -webkit-user-select: none; - -moz-user-select: -moz-none; - user-select: none; - background-color: transparent; + color: inherit; + border: none; + outline: none; + background-color: var(--satus-theme-button); + -webkit-tap-highlight-color: transparent; align-items: center; - justify-content: space-between; -} - -.satus-switch:hover -{ - background-color: var(--satus-theme-ripple); + -webkit-appearance: none; } -.satus-switch__input +.satus-button::before { position: absolute; - z-index: 1; top: 0; left: 0; width: 100%; height: 100%; - margin: 0; - padding: 0; - opacity: 0; - outline: none; + content: ''; - -webkit-appearance: none; + opacity: 0; + background-color: var(--satus-theme-ripple); } - -/* LABEL */ - -.satus-switch__label +.satus-button:focus::before, +.satus-button:hover::before { - padding: 0 16px 0 0; + opacity: 1; } +.satus-button__icon +{ + width: 24px; + height: 24px; +} -/* TRACK*/ - -.satus-switch__track +.satus-button svg { - position: relative; + width: 100%; - width: 32px; - min-width: 32px; - height: 18px; + color: inherit; +} - border-radius: 18px; - background: #bdbdbd; +.satus-button__icon + .satus-button__label +{ + margin-left: 8px; } + +/*-------------------------------------------------------------- +>>> TEXT +--------------------------------------------------------------*/ -.satus-switch__track::before +.satus-text { - position: absolute; - top: -7px; - left: -7px; + margin: 0; - width: 32px; - height: 32px; + color: inherit; + border: none; + font-family: inherit; +} + +/*-------------------------------------------------------------- +>>> SECTION +--------------------------------------------------------------*/ - content: ''; - transition: left .1s cubic-bezier(.4, 0, .2, 1), transform 200ms, background-color 200ms; - transform: scale(0); +.satus-section +{ + display: inline-flex; - opacity: 0; - border-radius: 50%; - background-color: #000; + box-sizing: border-box; - will-change: left, transform, background-color; + align-items: center; + flex: 1; } -.satus-switch__input:checked + .satus-switch__track::before +.satus-main .satus-section { - left: 8px; + background-color: var(--satus-theme-section); +} - background-color: var(--satus-theme-primary, #54d2a3); +.satus-section--label +{ + font-size: 17px; + display: block; + width: calc(100% - 16px); + max-width: 900px; + margin: 16px auto 8px; } + +/*-------------------------------------------------------------- +>>> COLOR PICKER +--------------------------------------------------------------*/ -.satus-switch__input:focus + .satus-switch__track::before +.satus-color-picker { - transform: scale(1); + font-size: inherit; - opacity: .08; + position: relative; + + display: flex; + + box-sizing: border-box; + margin: 0; + + cursor: pointer; + + color: inherit; + border: none; + outline: none; + background-color: var(--satus-theme-button); + + justify-content: space-between; + -webkit-tap-highlight-color: transparent; + align-items: center; + -webkit-appearance: none; } -.satus-switch__input:checked:focus + .satus-switch__track::before +.satus-color-picker__value { - opacity: .25; + width: 24px; + height: 24px; + + border-radius: 50%; } -.satus-switch__track::after +.satus-dialog--color-picker .satus-scrollbar__content { - position: absolute; + display: flex; + flex-direction: column; - width: 14px; - height: 14px; + align-items: center; +} - content: ''; - transition: transform .1s cubic-bezier(.4, 0, .2, 1); - transform: translate(2px, 2px); +.satus-dialog--color-picker .satus-button +{ + box-sizing: unset; + width: 24px; + height: 24px; + margin: 0 0 8px; + padding: 8px; border-radius: 50%; - background-color: #fff; +} - will-change: transform; +.satus-dialog--color-picker .satus-button:hover +{ + background-color: var(--satus-theme-ripple); } -.satus-switch__input:checked + .satus-switch__track +.satus-dialog--color-picker .satus-button::before { - background: var(--satus-theme-primary, #54d2a3); + content: unset; } -.satus-switch__input:checked + .satus-switch__track::after +.satus-dialog--color-picker .satus-button svg { - transform: translate(16px, 2px); + fill: transparent; + stroke: var(--satus-theme-primary); } - -/*-------------------------------------------------------------- ->>> TABLE ---------------------------------------------------------------*/ -.satus-table +.satus-dialog--color-picker canvas { - font-size: .875rem; + width: 200px; + height: 200px; + margin: 0 0 16px; +} + +/*--------------------------------------------------------------- +>>> SHORTCUT +----------------------------------------------------------------- +# Canvas +# Popup +---------------------------------------------------------------*/ +.satus-shortcut +{ position: relative; - overflow: hidden; display: flex; - flex-direction: column; + overflow: hidden; + box-sizing: border-box; + width: 100%; + height: 48px; + min-height: 48px; + padding: 0 16px; - border: 1px solid #ececec; - border-radius: 6px; + cursor: pointer; + + justify-content: space-between; + align-items: center; +} + +.satus-shortcut:hover +{ + background-color: var(--satus-theme-ripple); +} + +.satus-shortcut .satus-shortcut__label, +.satus-shortcut .satus-shortcut__value +{ + flex: 1 } -.satus-table__head +.satus-shortcut .satus-shortcut__label + .satus-shortcut__value { - position: relative; - z-index: 1; + justify-content: flex-end +} + +/*--------------------------------------------------------------- +# CANVAS +---------------------------------------------------------------*/ + +.satus-shortcut__value +{ display: flex; box-sizing: border-box; width: 100%; - height: 44px; - padding: 0 8px; + height: 48px; - background: #fff; - box-shadow: 0 2px 4px rgba(0, 0, 0, .1); + align-items: center; } -.satus-table__head > div + +.satus-shortcut__key { - font-weight: 600; + font-size: 12px; + line-height: 12px; - display: inline-flex; + display: flex; box-sizing: border-box; - padding: 0 8px 0 0; + min-width: 28px; + height: 28px; + padding: 4px 8px; - opacity: .7; + border: 1px solid rgba(0,0,0,.1); + border-radius: 4px; + background: #fff; + box-shadow: 0 1px 1px rgba(0,0,0,.1), inset 0 -1px 0 rgba(0,0,0,.1); - flex-shrink: 0; align-items: center; + justify-content: center; } -.satus-table__head > div:last-child -{ - padding: 0; -} - -.satus-table__head > div:hover -{ - cursor: pointer; - - opacity: 1; -} - -.satus-table__head > div[data-sorting=false]:hover -{ - cursor: default; -} - -.satus-table__head > div > span +.satus-shortcut__plus { position: relative; + + width: 10px; + height: 10px; + margin: 8px 6px; } -.satus-table__head > div[data-sorting=asc] > span::after +.satus-shortcut__plus::before { position: absolute; - top: calc(50% - 2px); - right: -14px; + top: 0; + left: 4px; - width: 0; - height: 0; + width: 2px; + height: 10px; content: ''; - border-right: 4px solid transparent; - border-bottom: 5px solid currentColor; - border-left: 4px solid transparent; + background-color: #ccc; } -.satus-table__head > div[data-sorting=desc] > span::after +.satus-shortcut__plus::after { position: absolute; - top: calc(50% - 2px); - right: -13px; + top: 4px; + left: 0; - width: 0; - height: 0; + width: 10px; + height: 2px; content: ''; - border-top: 5px solid currentColor; - border-right: 4px solid transparent; - border-left: 4px solid transparent; + background-color: #ccc; } -.satus-table__body +.satus-shortcut__mouse { - overflow: hidden; - flex: 1; + position: relative; - width: 100%; - height: calc(100% - 48px); + display: flex; + width: 24px; + height: 32px; + + border-radius: 50%; + border-top-left-radius: 12px; + border-top-right-radius: 12px; background: #fff; + box-shadow: 0 1px 3px rgba(0,0,0,.15), inset 0 -3px 0 rgba(0,0,0,.1); } -.satus-table__row +.satus-shortcut__mouse > div { - display: flex; - - padding: 0 8px; - - box-sizing: border-box; - - border-bottom: 1px solid rgba(0, 0, 0, 0.1); - - flex-wrap: wrap; -} + position: absolute; + top: 0; + left: 11px; -.satus-table__row:nth-child(2n) -{ - background: rgba(0,0,0,.03) + width: 2px; + height: 10px; + + border-radius: 2px; + background: #ccc; } -.satus-table__cell +.satus-shortcut__mouse::before { - overflow: hidden; + position: absolute; + top: 0; + left: 18px; - box-sizing: border-box; - padding: 8px 8px 8px 0; + width: 2px; + height: 14px; - white-space: nowrap; - word-wrap: break-word; - -webkit-hyphens: auto; - -moz-hyphens: auto; - hyphens: auto; + content: ''; - overflow-wrap: break-word; - flex-shrink: 0; - - text-overflow: ellipsis; + background: #f96754; } -.satus-table__cell:last-child +.satus-shortcut__mouse.false::after { - padding-right: 0; -} + position: absolute; + top: -5px; + left: 15px; -.satus-table__cell input -{ - width: 100%; - margin: 5px 0; - border: 1px solid #d2d2d2; - box-sizing: border-box; - padding: 1px 2px; - border-radius: 4px; - height: calc(100% - 10px); - outline: none -} + width: 0; + height: 0; + content: ''; + border-right: 4px solid transparent; + border-bottom: 6px solid #f96754; + border-left: 4px solid transparent; +} -.satus-table__paging > button +.satus-shortcut__mouse.true::after { - min-width: 32px; - height: 32px; - padding: 0 8px; + position: absolute; + top: 13px; + left: 15px; - cursor: pointer; + width: 0; + height: 0; - color: var(--satus-theme-on-surface, #555); - border: none; - background: transparent; + content: ''; + + border-top: 6px solid #f96754; + border-right: 4px solid transparent; + border-left: 4px solid transparent; } -.satus-table__paging > button.active -{ - color: #2979ff; +.satus-shortcut__mouse.click::before { + position: absolute; + top: 0px; + left: -1px; + width: 10px; + height: 10px; + content: ''; + background: #f96754; + border-radius: 50%; } - -/*-------------------------------------------------------------- ->>> TEXT ---------------------------------------------------------------*/ -.satus-text +.satus-shortcut__mouse.context::before { + position: absolute; + top: 0px; + left: 15px; + width: 10px; + height: 10px; + content: ''; + background: #f96754; + border-radius: 50%; +} + + +/*--------------------------------------------------------------- +# POPUP +---------------------------------------------------------------*/ + +.satus-shortcut-dialog-label { - margin: 0; + font-size: 16px; + font-weight: 500; - color: inherit; - border: none; + box-sizing: border-box; + width: 100%; + margin: 4px 0 10px; + padding: 0 16px; } - -/*-------------------------------------------------------------- ->>> TEXT ---------------------------------------------------------------*/ -.satus-text-field +.satus-shortcut__canvas { - font: inherit; + display: flex; box-sizing: border-box; width: 100%; - margin: 0; - padding: 0; - padding: 0 8px; + height: 68px; + padding: 16px; - color: inherit; - border: none; - outline: none; - background: unset; + background: #dedede; + + align-items: center; } - -/*----------------------------------------------------------------------------- ->>> THEMES -------------------------------------------------------------------------------- -1.0 Default ------------------------------------------------------------------------------*/ -html +.satus-section_shortcut { - --satus-theme-primary: #f6b465; - - --satus-theme-dialog: #f7f7f6; - --satus-theme-dialog-text: #777; - - --satus-theme-header: #fff; - --satus-theme-header-text: #777; + width: 100%; + margin: 8px 0 0; - --satus-theme-main: #f7f7f6; - --satus-theme-main-text: #777; + justify-content: flex-end; +} - --satus-theme-section: #fff; +.satus-button_shortcut +{ + font-weight: 500; - --satus-theme-button: transparent; + overflow: hidden; - --satus-theme-scrollbar: rgba(0, 0, 0, .2); - --satus-theme-scrollbar-focus: rgba(0, 0, 0, .4); + height: 28px; + min-height: 28px; + margin-right: 2px; + padding: 4px 8px; - --satus-theme-tooltip: rgba(0, 0, 0, .4); + text-transform: uppercase; - --satus-theme-ripple: rgba(0, 0, 0, .04); -} \ No newline at end of file + color: #f96754; + border-radius: 4px; +} diff --git a/satus.js b/satus.js index 5ccd74c14..959069050 100644 --- a/satus.js +++ b/satus.js @@ -73,114 +73,6 @@ Satus.getAnimationDuration = function(element) { return Number(window.getComputedStyle(element).getPropertyValue('animation-duration').replace(/[^0-9.]/g, '')) * 1000; }; -/*-------------------------------------------------------------- ->>> CHROMIUM STORAGE ----------------------------------------------------------------- -1.0 Get -2.0 Set -3.0 Import -4.0 Clear ---------------------------------------------------------------*/ - -Satus.storage = {}; - -/*-------------------------------------------------------------- -# GET ---------------------------------------------------------------*/ - -Satus.storage.get = function(name) { - var target = Satus.storage; - - name = name.split('/').filter(function(value) { - return value != ''; - }); - - for (var i = 0, l = name.length; i < l; i++) { - if (Satus.isset(target[name[i]])) { - target = target[name[i]]; - } else { - return undefined; - } - } - - return target; -}; - - -/*-------------------------------------------------------------- -# SET ---------------------------------------------------------------*/ - -Satus.storage.set = function(name, value) { - var items = {}, - target = Satus.storage; - - if (!Satus.isset(name)) { - return false; - } - - name = name.split('/').filter(function(value) { - return value != ''; - }); - - for (var i = 0, l = name.length; i < l; i++) { - var item = name[i]; - - if (i < l - 1) { - - if (target[item]) { - target = target[item]; - } else { - target[item] = {}; - - target = target[item]; - } - } else { - target[item] = value; - } - } - - for (var key in Satus.storage) { - if (typeof Satus.storage[key] !== 'function') { - items[key] = Satus.storage[key]; - } - } - - chrome.storage.local.set(items); -}; - - -/*-------------------------------------------------------------- -# IMPORT ---------------------------------------------------------------*/ - -Satus.storage.import = function(callback) { - chrome.storage.local.get(function(items) { - for (var key in items) { - Satus.storage[key] = items[key]; - } - - if (callback) { - callback(); - } - }); -}; - - -/*-------------------------------------------------------------- -# CLEAR ---------------------------------------------------------------*/ - -Satus.storage.clear = function() { - chrome.storage.local.clear(); - - for (var key in Satus.storage) { - if (typeof Satus.storage[key] !== 'function') { - delete Satus.storage[key]; - } - } -}; - /*-------------------------------------------------------------- # LOCALE --------------------------------------------------------------*/ @@ -244,95 +136,6 @@ Satus.locale.import = function(src, callback) { xhr.send(); }; /*-------------------------------------------------------------- ->>> RENDER ---------------------------------------------------------------*/ - -Satus.render = function(element, container, callback) { - function convert(object) { - if (object && object.type) { - var type = Satus.camelize(object.type), - component = Satus.components[type](object), - excluded_properties = ['type', 'label', 'class', 'title', 'storage']; - - function applyProperties(object, target) { - for (var key in object) { - if (Satus.isset(object[key]) && typeof object[key] === 'object' && !object[key].type) { - if (typeof target[key] !== 'object') { - target[key] = {}; - } - - applyProperties(object[key], target[key]); - } else if (excluded_properties.indexOf(key) === -1) { - target[key] = object[key]; - } - } - } - - applyProperties(object, component); - - component.classList.add('satus-' + object.type); - - if (object.class) { - var class_list = object.class.split(' '); - - for (var i = 0, l = class_list.length; i < l; i++) { - component.classList.add(class_list[i]); - } - } - - if (object.before) { - var component_before = document.createElement('span'); - - component_before.innerHTML = object.before; - - for (var i = component_before.children.length - 1; i > -1; i--) { - component.insertBefore(component_before.children[i], component.firstChild); - } - } - - if (object.after) { - var component_after = document.createElement('span'); - - component_after.innerHTML = object.after; - - for (var i = component_after.children.length - 1; i > -1; i--) { - component.appendChild(component_after.children[i]); - } - } - - (container || document.body).appendChild(component); - - if (typeof component.onClickRender === 'object') { - component.addEventListener('click', function() { - Satus.render(component.onClickRender); - }); - } - - if (Satus.isset(Satus.events.render)) { - for (var i = 0, l = Satus.events.render.length; i < l; i++) { - Satus.events.render[i](component, object); - } - } - - if (typeof component.onrender === 'function') { - component.onrender(object); - } - - if (callback) { - callback(); - } - } - } - - if (element.type) { - convert(element); - } else { - for (var key in element) { - convert(element[key]); - } - } -}; -/*-------------------------------------------------------------- # CLONE NODE STYLES --------------------------------------------------------------*/ @@ -343,84 +146,25 @@ Satus.cloneNodeStyles = function(origin, target) { Satus.cloneNodeStyles(origin.children[i], target.children[i]); } }; -/*----------------------------------------------------------------------------- ->>> «SEARCH» MODULE ------------------------------------------------------------------------------*/ - -Satus.search = function(query, object, callback, categories) { - var threads = 0, - folder = '', - results = {}; - - function parse(items) { - threads++; - - for (var key in items) { - var item = items[key]; - - if (categories === true && item.type === 'folder' && folder !== item.label) { - folder = item.label; - } - - if (['switch', 'select', 'slider'].indexOf(item.type) !== -1 && key.indexOf(query) !== -1) { - if (categories === true) { - if (!results[folder]) { - results[folder] = {}; - } - - results[folder][key] = item; - } else { - results[key] = item; - } - } +Satus.on('render', function(component, data) { + if (data.perspective === true) { + component.style.willChange = 'transform'; + component.style.transformStyle = 'preserve-3d'; + component.style.transition = '.4s'; - if (typeof item === 'object') { - parse(item); - } - } + component.addEventListener('mousemove', function(event) { + var bounding = component.getBoundingClientRect(), + dx = event.clientX - bounding.left - bounding.width / 2, + dy = event.clientY - bounding.top - bounding.height / 2; - threads--; + this.style.transform = 'perspective(440px) rotateX(' + dy * -1 + 'deg) rotateY(' + dx + 'deg) translateZ(0)'; + }); - if (threads === 0) { - callback(results); - } + component.addEventListener('mouseout', function(event) { + this.style.transform = 'perspective(440px) rotateX(0deg) rotateY(0deg) translateZ(0)'; + }); } - - parse(object); -}; - -/*-------------------------------------------------------------- ->>> STORAGE KEYS ---------------------------------------------------------------*/ - -Satus.modules.updateStorageKeys = function(object, callback) { - var threads = 0; - - function parse(items) { - threads++; - - for (var key in items) { - var item = items[key]; - - - if (item.type) { - item.storage_key = key; - } - - if (typeof item === 'object') { - parse(item); - } - } - - threads--; - - if (threads === 0) { - callback(); - } - } - - parse(object); -}; +}); /*----------------------------------------------------------------------------- >>> «USER» MODULE ------------------------------------------------------------------------------- @@ -781,374 +525,844 @@ Satus.modules.user = function() { return data; }; -Satus.on('render', function(component, data) { - if (data.perspective === true) { - component.style.willChange = 'transform'; - component.style.transformStyle = 'preserve-3d'; - component.style.transition = '.4s'; - - component.addEventListener('mousemove', function(event) { - var bounding = component.getBoundingClientRect(), - dx = event.clientX - bounding.left - bounding.width / 2, - dy = event.clientY - bounding.top - bounding.height / 2; +/*-------------------------------------------------------------- +>>> CHROMIUM STORAGE +---------------------------------------------------------------- +1.0 Get +2.0 Set +3.0 Import +4.0 Clear +--------------------------------------------------------------*/ - this.style.transform = 'perspective(440px) rotateX(' + dy * -1 + 'deg) rotateY(' + dx + 'deg) translateZ(0)'; - }); +Satus.storage = { + data: {} +}; - component.addEventListener('mouseout', function(event) { - this.style.transform = 'perspective(440px) rotateX(0deg) rotateY(0deg) translateZ(0)'; - }); - } -}); /*-------------------------------------------------------------- ->>> BUTTON +# GET --------------------------------------------------------------*/ -Satus.components.button = function(element) { - var component = document.createElement('button'); +Satus.storage.get = function(name) { + if (satus.isset(name)) { + var target = Satus.storage; - if (Satus.isset(element.icon)) { - var component_icon = document.createElement('span'); + name = name.split('/').filter(function(value) { + return value != ''; + }); - component_icon.className = 'satus-button__icon'; - component_icon.innerHTML = element.icon; + for (var i = 0, l = name.length; i < l; i++) { + if (Satus.isset(target[name[i]])) { + target = target[name[i]]; + } else { + return undefined; + } + } - component.appendChild(component_icon); + return target; } +}; - if (Satus.isset(element.label)) { - var component_label = document.createElement('span'); - - component_label.className = 'satus-button__label'; - component_label.innerText = Satus.locale.getMessage(element.label); - - component.appendChild(component_label); - } - return component; -}; /*-------------------------------------------------------------- ->>> COLOR PICKER +# SET --------------------------------------------------------------*/ -Satus.components.colorPicker = function(element) { - var component = document.createElement('div'), - component_value = document.createElement('div'); - - element.class = 'satus-button'; - component_value.className = 'satus-color-picker__value'; - component_value.style.backgroundColor = Satus.storage.get(element.storage_key) || element.value || ''; - - if (Satus.isset(element.label)) { - var component_label = document.createElement('span'); - - component_label.className = 'satus-button__label'; - component_label.innerText = Satus.locale.getMessage(element.label); - - component.appendChild(component_label); +Satus.storage.set = function(name, value) { + var items = {}, + target = Satus.storage; + + if (!Satus.isset(name)) { + return false; } - component.addEventListener('click', function() { - var component = document.createElement('div'), - component_canvas = document.createElement('canvas'), - close = document.createElement('button'), - ctx = component_canvas.getContext('2d'), - image = new Image(), - dialog = Satus.components.dialog({}); - - close.className = 'satus-button'; - close.innerHTML = ''; - close.onclick = function() { - dialog.querySelector('.satus-dialog__scrim').click(); - }; - - dialog.className = 'satus-dialog satus-dialog--color-picker'; + name = name.split('/').filter(function(value) { + return value != ''; + }); - component_canvas.width = 200; - component_canvas.height = 200; + for (var i = 0, l = name.length; i < l; i++) { + var item = name[i]; - function select(event) { - var coordinates = component_canvas.getBoundingClientRect(), - x = event.clientX - coordinates.left, - y = event.clientY - coordinates.top, - color = ctx.getImageData(x, y, 1, 1).data; + if (i < l - 1) { - component_value.style.backgroundColor = 'rgb(' + color[0] + ',' + color[1] + ',' + color[2] + ')'; + if (target[item]) { + target = target[item]; + } else { + target[item] = {}; - Satus.storage.set(element.storage_key, component_value.style.backgroundColor); + target = target[item]; + } + } else { + target[item] = value; } + } - function mouseup(event) { - component_canvas.removeEventListener('mousemove', select); - window.removeEventListener('mouseup', mouseup); + for (var key in Satus.storage) { + if (typeof Satus.storage[key] !== 'function') { + items[key] = Satus.storage[key]; } + } - component_canvas.addEventListener('mousedown', function() { - select(event); - this.addEventListener('mousemove', select); - window.addEventListener('mouseup', mouseup); - }); - - image.onload = function() { - ctx.drawImage(image, 0, 0); + chrome.storage.local.set(items); +}; - image.remove(); - }; - image.src = 'data:image/svg+xml;base64,<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 270 270"><defs><radialGradient id="a"><stop offset="0%" stop-color="#fff"/><stop offset="100%" stop-color="#fff" stop-opacity="0"/></radialGradient></defs><g fill="none" stroke-width="133" class="IroWheelHue"><path stroke="hsl(240, 100%, 50%)" d="M201.477 136.74a66.5 66.5 0 00.023-1.74"/><path stroke="hsl(241, 100%, 50%)" d="M201.437 137.9a66.5 66.5 0 00.053-1.74"/><path stroke="hsl(242, 100%, 50%)" d="M201.376 139.06a66.5 66.5 0 00.083-1.74"/><path stroke="hsl(243, 100%, 50%)" d="M201.295 140.218a66.5 66.5 0 00.114-1.738"/><path stroke="hsl(244, 100%, 50%)" d="M201.194 141.374a66.5 66.5 0 00.144-1.735"/><path stroke="hsl(245, 100%, 50%)" d="M201.073 142.528a66.5 66.5 0 00.174-1.732"/><path stroke="hsl(246, 100%, 50%)" d="M200.931 143.68a66.5 66.5 0 00.205-1.729"/><path stroke="hsl(247, 100%, 50%)" d="M200.77 144.83a66.5 66.5 0 00.234-1.726"/><path stroke="hsl(248, 100%, 50%)" d="M200.588 145.976a66.5 66.5 0 00.265-1.721"/><path stroke="hsl(249, 100%, 50%)" d="M200.386 147.119a66.5 66.5 0 00.295-1.716"/><path stroke="hsl(250, 100%, 50%)" d="M200.165 148.258a66.5 66.5 0 00.325-1.71"/><path stroke="hsl(251, 100%, 50%)" d="M199.924 149.393a66.5 66.5 0 00.354-1.704"/><path stroke="hsl(252, 100%, 50%)" d="M199.663 150.524a66.5 66.5 0 00.384-1.698"/><path stroke="hsl(253, 100%, 50%)" d="M199.382 151.65a66.5 66.5 0 00.414-1.69"/><path stroke="hsl(254, 100%, 50%)" d="M199.081 152.771a66.5 66.5 0 00.444-1.683"/><path stroke="hsl(255, 100%, 50%)" d="M198.762 153.887a66.5 66.5 0 00.472-1.676"/><path stroke="hsl(256, 100%, 50%)" d="M198.422 154.997a66.5 66.5 0 00.502-1.667"/><path stroke="hsl(257, 100%, 50%)" d="M198.064 156.1a66.5 66.5 0 00.53-1.657"/><path stroke="hsl(258, 100%, 50%)" d="M197.686 157.198a66.5 66.5 0 00.56-1.648"/><path stroke="hsl(259, 100%, 50%)" d="M197.289 158.289a66.5 66.5 0 00.588-1.639"/><path stroke="hsl(260, 100%, 50%)" d="M196.873 159.372a66.5 66.5 0 00.617-1.628"/><path stroke="hsl(261, 100%, 50%)" d="M196.438 160.448a66.5 66.5 0 00.645-1.617"/><path stroke="hsl(262, 100%, 50%)" d="M195.984 161.517a66.5 66.5 0 00.674-1.606"/><path stroke="hsl(263, 100%, 50%)" d="M195.512 162.577a66.5 66.5 0 00.702-1.593"/><path stroke="hsl(264, 100%, 50%)" d="M195.022 163.629a66.5 66.5 0 00.729-1.581"/><path stroke="hsl(265, 100%, 50%)" d="M194.513 164.672a66.5 66.5 0 00.756-1.568"/><path stroke="hsl(266, 100%, 50%)" d="M193.986 165.706a66.5 66.5 0 00.784-1.554"/><path stroke="hsl(267, 100%, 50%)" d="M193.441 166.731a66.5 66.5 0 00.81-1.54"/><path stroke="hsl(268, 100%, 50%)" d="M192.879 167.746a66.5 66.5 0 00.837-1.526"/><path stroke="hsl(269, 100%, 50%)" d="M192.298 168.751a66.5 66.5 0 00.864-1.511"/><path stroke="hsl(270, 100%, 50%)" d="M191.7 169.746a66.5 66.5 0 00.89-1.496"/><path stroke="hsl(271, 100%, 50%)" d="M191.086 170.73a66.5 66.5 0 00.916-1.48"/><path stroke="hsl(272, 100%, 50%)" d="M190.453 171.704a66.5 66.5 0 00.942-1.464"/><path stroke="hsl(273, 100%, 50%)" d="M189.804 172.666a66.5 66.5 0 00.968-1.448"/><path stroke="hsl(274, 100%, 50%)" d="M189.139 173.617a66.5 66.5 0 00.992-1.43"/><path stroke="hsl(275, 100%, 50%)" d="M188.456 174.556a66.5 66.5 0 001.018-1.413"/><path stroke="hsl(276, 100%, 50%)" d="M187.758 175.483a66.5 66.5 0 001.042-1.395"/><path stroke="hsl(277, 100%, 50%)" d="M187.043 176.397a66.5 66.5 0 001.066-1.376"/><path stroke="hsl(278, 100%, 50%)" d="M186.313 177.3a66.5 66.5 0 001.09-1.359"/><path stroke="hsl(279, 100%, 50%)" d="M185.567 178.188a66.5 66.5 0 001.113-1.338"/><path stroke="hsl(280, 100%, 50%)" d="M184.806 179.064a66.5 66.5 0 001.136-1.319"/><path stroke="hsl(281, 100%, 50%)" d="M184.029 179.927a66.5 66.5 0 001.16-1.3"/><path stroke="hsl(282, 100%, 50%)" d="M183.237 180.776a66.5 66.5 0 001.182-1.279"/><path stroke="hsl(283, 100%, 50%)" d="M182.431 181.61a66.5 66.5 0 001.204-1.257"/><path stroke="hsl(284, 100%, 50%)" d="M181.61 182.431a66.5 66.5 0 001.226-1.236"/><path stroke="hsl(285, 100%, 50%)" d="M180.776 183.237a66.5 66.5 0 001.247-1.214"/><path stroke="hsl(286, 100%, 50%)" d="M179.927 184.029a66.5 66.5 0 001.268-1.193"/><path stroke="hsl(287, 100%, 50%)" d="M179.064 184.806a66.5 66.5 0 001.289-1.171"/><path stroke="hsl(288, 100%, 50%)" d="M178.188 185.567a66.5 66.5 0 001.31-1.148"/><path stroke="hsl(289, 100%, 50%)" d="M177.3 186.313a66.5 66.5 0 001.328-1.125"/><path stroke="hsl(290, 100%, 50%)" d="M176.397 187.043a66.5 66.5 0 001.348-1.101"/><path stroke="hsl(291, 100%, 50%)" d="M175.483 187.758a66.5 66.5 0 001.367-1.078"/><path stroke="hsl(292, 100%, 50%)" d="M174.556 188.456a66.5 66.5 0 001.385-1.053"/><path stroke="hsl(293, 100%, 50%)" d="M173.617 189.139a66.5 66.5 0 001.404-1.03"/><path stroke="hsl(294, 100%, 50%)" d="M172.666 189.804a66.5 66.5 0 001.422-1.004"/><path stroke="hsl(295, 100%, 50%)" d="M171.704 190.453a66.5 66.5 0 001.439-.98"/><path stroke="hsl(296, 100%, 50%)" d="M170.73 191.086a66.5 66.5 0 001.456-.955"/><path stroke="hsl(297, 100%, 50%)" d="M169.746 191.7a66.5 66.5 0 001.472-.928"/><path stroke="hsl(298, 100%, 50%)" d="M168.751 192.298a66.5 66.5 0 001.489-.903"/><path stroke="hsl(299, 100%, 50%)" d="M167.746 192.879a66.5 66.5 0 001.504-.877"/><path stroke="hsl(300, 100%, 50%)" d="M166.731 193.441a66.5 66.5 0 001.519-.85"/><path stroke="hsl(301, 100%, 50%)" d="M165.706 193.986a66.5 66.5 0 001.534-.824"/><path stroke="hsl(302, 100%, 50%)" d="M164.672 194.513a66.5 66.5 0 001.548-.797"/><path stroke="hsl(303, 100%, 50%)" d="M163.629 195.022a66.5 66.5 0 001.561-.77"/><path stroke="hsl(304, 100%, 50%)" d="M162.577 195.512a66.5 66.5 0 001.575-.742"/><path stroke="hsl(305, 100%, 50%)" d="M161.517 195.984a66.5 66.5 0 001.587-.715"/><path stroke="hsl(306, 100%, 50%)" d="M160.448 196.438a66.5 66.5 0 001.6-.687"/><path stroke="hsl(307, 100%, 50%)" d="M159.372 196.873a66.5 66.5 0 001.612-.66"/><path stroke="hsl(308, 100%, 50%)" d="M158.289 197.289a66.5 66.5 0 001.622-.631"/><path stroke="hsl(309, 100%, 50%)" d="M157.198 197.686a66.5 66.5 0 001.633-.603"/><path stroke="hsl(310, 100%, 50%)" d="M156.1 198.064a66.5 66.5 0 001.644-.574"/><path stroke="hsl(311, 100%, 50%)" d="M154.997 198.422a66.5 66.5 0 001.653-.545"/><path stroke="hsl(312, 100%, 50%)" d="M153.887 198.762a66.5 66.5 0 001.663-.517"/><path stroke="hsl(313, 100%, 50%)" d="M152.771 199.081a66.5 66.5 0 001.672-.487"/><path stroke="hsl(314, 100%, 50%)" d="M151.65 199.382a66.5 66.5 0 001.68-.458"/><path stroke="hsl(315, 100%, 50%)" d="M150.524 199.663a66.5 66.5 0 001.687-.429"/><path stroke="hsl(316, 100%, 50%)" d="M149.393 199.924a66.5 66.5 0 001.695-.4"/><path stroke="hsl(317, 100%, 50%)" d="M148.258 200.165a66.5 66.5 0 001.701-.37"/><path stroke="hsl(318, 100%, 50%)" d="M147.119 200.386a66.5 66.5 0 001.707-.34"/><path stroke="hsl(319, 100%, 50%)" d="M145.976 200.588a66.5 66.5 0 001.713-.31"/><path stroke="hsl(320, 100%, 50%)" d="M144.83 200.77a66.5 66.5 0 001.718-.28"/><path stroke="hsl(321, 100%, 50%)" d="M143.68 200.931a66.5 66.5 0 001.723-.25"/><path stroke="hsl(322, 100%, 50%)" d="M142.528 201.073a66.5 66.5 0 001.727-.22"/><path stroke="hsl(323, 100%, 50%)" d="M141.374 201.194a66.5 66.5 0 001.73-.19"/><path stroke="hsl(324, 100%, 50%)" d="M140.218 201.295a66.5 66.5 0 001.733-.16"/><path stroke="hsl(325, 100%, 50%)" d="M139.06 201.376a66.5 66.5 0 001.736-.13"/><path stroke="hsl(326, 100%, 50%)" d="M137.9 201.437a66.5 66.5 0 001.739-.099"/><path stroke="hsl(327, 100%, 50%)" d="M136.74 201.477a66.5 66.5 0 001.74-.068"/><path stroke="hsl(328, 100%, 50%)" d="M135.58 201.497a66.5 66.5 0 001.74-.038"/><path stroke="hsl(329, 100%, 50%)" d="M134.42 201.497a66.5 66.5 0 001.74-.007"/><path stroke="hsl(330, 100%, 50%)" d="M133.26 201.477a66.5 66.5 0 001.74.023"/><path stroke="hsl(331, 100%, 50%)" d="M132.1 201.437a66.5 66.5 0 001.74.053"/><path stroke="hsl(332, 100%, 50%)" d="M130.94 201.376a66.5 66.5 0 001.74.083"/><path stroke="hsl(333, 100%, 50%)" d="M129.782 201.295a66.5 66.5 0 001.738.114"/><path stroke="hsl(334, 100%, 50%)" d="M128.626 201.194a66.5 66.5 0 001.735.144"/><path stroke="hsl(335, 100%, 50%)" d="M127.472 201.073a66.5 66.5 0 001.732.174"/><path stroke="hsl(336, 100%, 50%)" d="M126.32 200.931a66.5 66.5 0 001.729.205"/><path stroke="hsl(337, 100%, 50%)" d="M125.17 200.77a66.5 66.5 0 001.726.234"/><path stroke="hsl(338, 100%, 50%)" d="M124.024 200.588a66.5 66.5 0 001.721.265"/><path stroke="hsl(339, 100%, 50%)" d="M122.881 200.386a66.5 66.5 0 001.716.295"/><path stroke="hsl(340, 100%, 50%)" d="M121.742 200.165a66.5 66.5 0 001.71.325"/><path stroke="hsl(341, 100%, 50%)" d="M120.607 199.924a66.5 66.5 0 001.704.354"/><path stroke="hsl(342, 100%, 50%)" d="M119.476 199.663a66.5 66.5 0 001.698.384"/><path stroke="hsl(343, 100%, 50%)" d="M118.35 199.382a66.5 66.5 0 001.69.414"/><path stroke="hsl(344, 100%, 50%)" d="M117.229 199.081a66.5 66.5 0 001.683.444"/><path stroke="hsl(345, 100%, 50%)" d="M116.113 198.762a66.5 66.5 0 001.676.472"/><path stroke="hsl(346, 100%, 50%)" d="M115.003 198.422a66.5 66.5 0 001.667.502"/><path stroke="hsl(347, 100%, 50%)" d="M113.9 198.064a66.5 66.5 0 001.657.53"/><path stroke="hsl(348, 100%, 50%)" d="M112.802 197.686a66.5 66.5 0 001.648.56"/><path stroke="hsl(349, 100%, 50%)" d="M111.711 197.289a66.5 66.5 0 001.639.588"/><path stroke="hsl(350, 100%, 50%)" d="M110.628 196.873a66.5 66.5 0 001.628.617"/><path stroke="hsl(351, 100%, 50%)" d="M109.552 196.438a66.5 66.5 0 001.617.645"/><path stroke="hsl(352, 100%, 50%)" d="M108.483 195.984a66.5 66.5 0 001.606.674"/><path stroke="hsl(353, 100%, 50%)" d="M107.423 195.512a66.5 66.5 0 001.593.702"/><path stroke="hsl(354, 100%, 50%)" d="M106.371 195.022a66.5 66.5 0 001.581.729"/><path stroke="hsl(355, 100%, 50%)" d="M105.328 194.513a66.5 66.5 0 001.568.756"/><path stroke="hsl(356, 100%, 50%)" d="M104.294 193.986a66.5 66.5 0 001.554.784"/><path stroke="hsl(357, 100%, 50%)" d="M103.269 193.441a66.5 66.5 0 001.54.81"/><path stroke="hsl(358, 100%, 50%)" d="M102.254 192.879a66.5 66.5 0 001.526.837"/><path stroke="hsl(359, 100%, 50%)" d="M101.249 192.298a66.5 66.5 0 001.511.864"/><path stroke="hsl(0, 100%, 50%)" d="M100.254 191.7a66.5 66.5 0 001.496.89"/><path stroke="hsl(1, 100%, 50%)" d="M99.27 191.086a66.5 66.5 0 001.48.916"/><path stroke="hsl(2, 100%, 50%)" d="M98.296 190.453a66.5 66.5 0 001.464.942"/><path stroke="hsl(3, 100%, 50%)" d="M97.334 189.804a66.5 66.5 0 001.448.968"/><path stroke="hsl(4, 100%, 50%)" d="M96.383 189.139a66.5 66.5 0 001.43.992"/><path stroke="hsl(5, 100%, 50%)" d="M95.444 188.456a66.5 66.5 0 001.413 1.018"/><path stroke="hsl(6, 100%, 50%)" d="M94.517 187.758a66.5 66.5 0 001.395 1.042"/><path stroke="hsl(7, 100%, 50%)" d="M93.603 187.043a66.5 66.5 0 001.376 1.066"/><path stroke="hsl(8, 100%, 50%)" d="M92.7 186.313a66.5 66.5 0 001.359 1.09"/><path stroke="hsl(9, 100%, 50%)" d="M91.812 185.567a66.5 66.5 0 001.338 1.113"/><path stroke="hsl(10, 100%, 50%)" d="M90.936 184.806a66.5 66.5 0 001.319 1.136"/><path stroke="hsl(11, 100%, 50%)" d="M90.073 184.029a66.5 66.5 0 001.3 1.16"/><path stroke="hsl(12, 100%, 50%)" d="M89.224 183.237a66.5 66.5 0 001.279 1.182"/><path stroke="hsl(13, 100%, 50%)" d="M88.39 182.431a66.5 66.5 0 001.257 1.204"/><path stroke="hsl(14, 100%, 50%)" d="M87.569 181.61a66.5 66.5 0 001.236 1.226"/><path stroke="hsl(15, 100%, 50%)" d="M86.763 180.776a66.5 66.5 0 001.214 1.247"/><path stroke="hsl(16, 100%, 50%)" d="M85.971 179.927a66.5 66.5 0 001.193 1.268"/><path stroke="hsl(17, 100%, 50%)" d="M85.194 179.064a66.5 66.5 0 001.171 1.289"/><path stroke="hsl(18, 100%, 50%)" d="M84.433 178.188a66.5 66.5 0 001.148 1.31"/><path stroke="hsl(19, 100%, 50%)" d="M83.687 177.3a66.5 66.5 0 001.125 1.328"/><path stroke="hsl(20, 100%, 50%)" d="M82.957 176.397a66.5 66.5 0 001.101 1.348"/><path stroke="hsl(21, 100%, 50%)" d="M82.242 175.483a66.5 66.5 0 001.078 1.367"/><path stroke="hsl(22, 100%, 50%)" d="M81.544 174.556a66.5 66.5 0 001.053 1.385"/><path stroke="hsl(23, 100%, 50%)" d="M80.861 173.617a66.5 66.5 0 001.03 1.404"/><path stroke="hsl(24, 100%, 50%)" d="M80.196 172.666a66.5 66.5 0 001.004 1.422"/><path stroke="hsl(25, 100%, 50%)" d="M79.547 171.704a66.5 66.5 0 00.98 1.439"/><path stroke="hsl(26, 100%, 50%)" d="M78.914 170.73a66.5 66.5 0 00.955 1.456"/><path stroke="hsl(27, 100%, 50%)" d="M78.3 169.746a66.5 66.5 0 00.928 1.472"/><path stroke="hsl(28, 100%, 50%)" d="M77.702 168.751a66.5 66.5 0 00.903 1.489"/><path stroke="hsl(29, 100%, 50%)" d="M77.121 167.746a66.5 66.5 0 00.877 1.504"/><path stroke="hsl(30, 100%, 50%)" d="M76.559 166.731a66.5 66.5 0 00.85 1.519"/><path stroke="hsl(31, 100%, 50%)" d="M76.014 165.706a66.5 66.5 0 00.824 1.534"/><path stroke="hsl(32, 100%, 50%)" d="M75.487 164.672a66.5 66.5 0 00.797 1.548"/><path stroke="hsl(33, 100%, 50%)" d="M74.978 163.629a66.5 66.5 0 00.77 1.561"/><path stroke="hsl(34, 100%, 50%)" d="M74.488 162.577a66.5 66.5 0 00.742 1.575"/><path stroke="hsl(35, 100%, 50%)" d="M74.016 161.517a66.5 66.5 0 00.715 1.587"/><path stroke="hsl(36, 100%, 50%)" d="M73.562 160.448a66.5 66.5 0 00.687 1.6"/><path stroke="hsl(37, 100%, 50%)" d="M73.127 159.372a66.5 66.5 0 00.66 1.612"/><path stroke="hsl(38, 100%, 50%)" d="M72.711 158.289a66.5 66.5 0 00.631 1.622"/><path stroke="hsl(39, 100%, 50%)" d="M72.314 157.198a66.5 66.5 0 00.603 1.633"/><path stroke="hsl(40, 100%, 50%)" d="M71.936 156.1a66.5 66.5 0 00.574 1.644"/><path stroke="hsl(41, 100%, 50%)" d="M71.578 154.997a66.5 66.5 0 00.545 1.653"/><path stroke="hsl(42, 100%, 50%)" d="M71.238 153.887a66.5 66.5 0 00.517 1.663"/><path stroke="hsl(43, 100%, 50%)" d="M70.919 152.771a66.5 66.5 0 00.487 1.672"/><path stroke="hsl(44, 100%, 50%)" d="M70.618 151.65a66.5 66.5 0 00.458 1.68"/><path stroke="hsl(45, 100%, 50%)" d="M70.337 150.524a66.5 66.5 0 00.429 1.687"/><path stroke="hsl(46, 100%, 50%)" d="M70.076 149.393a66.5 66.5 0 00.4 1.695"/><path stroke="hsl(47, 100%, 50%)" d="M69.835 148.258a66.5 66.5 0 00.37 1.701"/><path stroke="hsl(48, 100%, 50%)" d="M69.614 147.119a66.5 66.5 0 00.34 1.707"/><path stroke="hsl(49, 100%, 50%)" d="M69.412 145.976a66.5 66.5 0 00.31 1.713"/><path stroke="hsl(50, 100%, 50%)" d="M69.23 144.83a66.5 66.5 0 00.28 1.718"/><path stroke="hsl(51, 100%, 50%)" d="M69.069 143.68a66.5 66.5 0 00.25 1.723"/><path stroke="hsl(52, 100%, 50%)" d="M68.927 142.528a66.5 66.5 0 00.22 1.727"/><path stroke="hsl(53, 100%, 50%)" d="M68.806 141.374a66.5 66.5 0 00.19 1.73"/><path stroke="hsl(54, 100%, 50%)" d="M68.705 140.218a66.5 66.5 0 00.16 1.733"/><path stroke="hsl(55, 100%, 50%)" d="M68.624 139.06a66.5 66.5 0 00.13 1.736"/><path stroke="hsl(56, 100%, 50%)" d="M68.563 137.9a66.5 66.5 0 00.099 1.739"/><path stroke="hsl(57, 100%, 50%)" d="M68.523 136.74a66.5 66.5 0 00.068 1.74"/><path stroke="hsl(58, 100%, 50%)" d="M68.503 135.58a66.5 66.5 0 00.038 1.74"/><path stroke="hsl(59, 100%, 50%)" d="M68.503 134.42a66.5 66.5 0 00.007 1.74"/><path stroke="hsl(60, 100%, 50%)" d="M68.523 133.26A66.5 66.5 0 0068.5 135"/><path stroke="hsl(61, 100%, 50%)" d="M68.563 132.1a66.5 66.5 0 00-.053 1.74"/><path stroke="hsl(62, 100%, 50%)" d="M68.624 130.94a66.5 66.5 0 00-.083 1.74"/><path stroke="hsl(63, 100%, 50%)" d="M68.705 129.782a66.5 66.5 0 00-.114 1.738"/><path stroke="hsl(64, 100%, 50%)" d="M68.806 128.626a66.5 66.5 0 00-.144 1.735"/><path stroke="hsl(65, 100%, 50%)" d="M68.927 127.472a66.5 66.5 0 00-.174 1.732"/><path stroke="hsl(66, 100%, 50%)" d="M69.069 126.32a66.5 66.5 0 00-.205 1.729"/><path stroke="hsl(67, 100%, 50%)" d="M69.23 125.17a66.5 66.5 0 00-.234 1.726"/><path stroke="hsl(68, 100%, 50%)" d="M69.412 124.024a66.5 66.5 0 00-.265 1.721"/><path stroke="hsl(69, 100%, 50%)" d="M69.614 122.881a66.5 66.5 0 00-.295 1.716"/><path stroke="hsl(70, 100%, 50%)" d="M69.835 121.742a66.5 66.5 0 00-.325 1.71"/><path stroke="hsl(71, 100%, 50%)" d="M70.076 120.607a66.5 66.5 0 00-.354 1.704"/><path stroke="hsl(72, 100%, 50%)" d="M70.337 119.476a66.5 66.5 0 00-.384 1.698"/><path stroke="hsl(73, 100%, 50%)" d="M70.618 118.35a66.5 66.5 0 00-.414 1.69"/><path stroke="hsl(74, 100%, 50%)" d="M70.919 117.229a66.5 66.5 0 00-.444 1.683"/><path stroke="hsl(75, 100%, 50%)" d="M71.238 116.113a66.5 66.5 0 00-.472 1.676"/><path stroke="hsl(76, 100%, 50%)" d="M71.578 115.003a66.5 66.5 0 00-.502 1.667"/><path stroke="hsl(77, 100%, 50%)" d="M71.936 113.9a66.5 66.5 0 00-.53 1.657"/><path stroke="hsl(78, 100%, 50%)" d="M72.314 112.802a66.5 66.5 0 00-.56 1.648"/><path stroke="hsl(79, 100%, 50%)" d="M72.711 111.711a66.5 66.5 0 00-.588 1.639"/><path stroke="hsl(80, 100%, 50%)" d="M73.127 110.628a66.5 66.5 0 00-.617 1.628"/><path stroke="hsl(81, 100%, 50%)" d="M73.562 109.552a66.5 66.5 0 00-.645 1.617"/><path stroke="hsl(82, 100%, 50%)" d="M74.016 108.483a66.5 66.5 0 00-.674 1.606"/><path stroke="hsl(83, 100%, 50%)" d="M74.488 107.423a66.5 66.5 0 00-.702 1.593"/><path stroke="hsl(84, 100%, 50%)" d="M74.978 106.371a66.5 66.5 0 00-.729 1.581"/><path stroke="hsl(85, 100%, 50%)" d="M75.487 105.328a66.5 66.5 0 00-.756 1.568"/><path stroke="hsl(86, 100%, 50%)" d="M76.014 104.294a66.5 66.5 0 00-.784 1.554"/><path stroke="hsl(87, 100%, 50%)" d="M76.559 103.269a66.5 66.5 0 00-.81 1.54"/><path stroke="hsl(88, 100%, 50%)" d="M77.121 102.254a66.5 66.5 0 00-.837 1.526"/><path stroke="hsl(89, 100%, 50%)" d="M77.702 101.249a66.5 66.5 0 00-.864 1.511"/><path stroke="hsl(90, 100%, 50%)" d="M78.3 100.254a66.5 66.5 0 00-.89 1.496"/><path stroke="hsl(91, 100%, 50%)" d="M78.914 99.27a66.5 66.5 0 00-.916 1.48"/><path stroke="hsl(92, 100%, 50%)" d="M79.547 98.296a66.5 66.5 0 00-.942 1.464"/><path stroke="hsl(93, 100%, 50%)" d="M80.196 97.334a66.5 66.5 0 00-.968 1.448"/><path stroke="hsl(94, 100%, 50%)" d="M80.861 96.383a66.5 66.5 0 00-.992 1.43"/><path stroke="hsl(95, 100%, 50%)" d="M81.544 95.444a66.5 66.5 0 00-1.018 1.413"/><path stroke="hsl(96, 100%, 50%)" d="M82.242 94.517a66.5 66.5 0 00-1.042 1.395"/><path stroke="hsl(97, 100%, 50%)" d="M82.957 93.603a66.5 66.5 0 00-1.066 1.376"/><path stroke="hsl(98, 100%, 50%)" d="M83.687 92.7a66.5 66.5 0 00-1.09 1.359"/><path stroke="hsl(99, 100%, 50%)" d="M84.433 91.812a66.5 66.5 0 00-1.113 1.338"/><path stroke="hsl(100, 100%, 50%)" d="M85.194 90.936a66.5 66.5 0 00-1.136 1.319"/><path stroke="hsl(101, 100%, 50%)" d="M85.971 90.073a66.5 66.5 0 00-1.16 1.3"/><path stroke="hsl(102, 100%, 50%)" d="M86.763 89.224a66.5 66.5 0 00-1.182 1.279"/><path stroke="hsl(103, 100%, 50%)" d="M87.569 88.39a66.5 66.5 0 00-1.204 1.257"/><path stroke="hsl(104, 100%, 50%)" d="M88.39 87.569a66.5 66.5 0 00-1.226 1.236"/><path stroke="hsl(105, 100%, 50%)" d="M89.224 86.763a66.5 66.5 0 00-1.247 1.214"/><path stroke="hsl(106, 100%, 50%)" d="M90.073 85.971a66.5 66.5 0 00-1.268 1.193"/><path stroke="hsl(107, 100%, 50%)" d="M90.936 85.194a66.5 66.5 0 00-1.289 1.171"/><path stroke="hsl(108, 100%, 50%)" d="M91.812 84.433a66.5 66.5 0 00-1.31 1.148"/><path stroke="hsl(109, 100%, 50%)" d="M92.7 83.687a66.5 66.5 0 00-1.328 1.125"/><path stroke="hsl(110, 100%, 50%)" d="M93.603 82.957a66.5 66.5 0 00-1.348 1.101"/><path stroke="hsl(111, 100%, 50%)" d="M94.517 82.242a66.5 66.5 0 00-1.367 1.078"/><path stroke="hsl(112, 100%, 50%)" d="M95.444 81.544a66.5 66.5 0 00-1.385 1.053"/><path stroke="hsl(113, 100%, 50%)" d="M96.383 80.861a66.5 66.5 0 00-1.404 1.03"/><path stroke="hsl(114, 100%, 50%)" d="M97.334 80.196a66.5 66.5 0 00-1.422 1.004"/><path stroke="hsl(115, 100%, 50%)" d="M98.296 79.547a66.5 66.5 0 00-1.439.98"/><path stroke="hsl(116, 100%, 50%)" d="M99.27 78.914a66.5 66.5 0 00-1.456.955"/><path stroke="hsl(117, 100%, 50%)" d="M100.254 78.3a66.5 66.5 0 00-1.472.928"/><path stroke="hsl(118, 100%, 50%)" d="M101.249 77.702a66.5 66.5 0 00-1.489.903"/><path stroke="hsl(119, 100%, 50%)" d="M102.254 77.121a66.5 66.5 0 00-1.504.877"/><path stroke="hsl(120, 100%, 50%)" d="M103.269 76.559a66.5 66.5 0 00-1.519.85"/><path stroke="hsl(121, 100%, 50%)" d="M104.294 76.014a66.5 66.5 0 00-1.534.824"/><path stroke="hsl(122, 100%, 50%)" d="M105.328 75.487a66.5 66.5 0 00-1.548.797"/><path stroke="hsl(123, 100%, 50%)" d="M106.371 74.978a66.5 66.5 0 00-1.561.77"/><path stroke="hsl(124, 100%, 50%)" d="M107.423 74.488a66.5 66.5 0 00-1.575.742"/><path stroke="hsl(125, 100%, 50%)" d="M108.483 74.016a66.5 66.5 0 00-1.587.715"/><path stroke="hsl(126, 100%, 50%)" d="M109.552 73.562a66.5 66.5 0 00-1.6.687"/><path stroke="hsl(127, 100%, 50%)" d="M110.628 73.127a66.5 66.5 0 00-1.612.66"/><path stroke="hsl(128, 100%, 50%)" d="M111.711 72.711a66.5 66.5 0 00-1.622.631"/><path stroke="hsl(129, 100%, 50%)" d="M112.802 72.314a66.5 66.5 0 00-1.633.603"/><path stroke="hsl(130, 100%, 50%)" d="M113.9 71.936a66.5 66.5 0 00-1.644.574"/><path stroke="hsl(131, 100%, 50%)" d="M115.003 71.578a66.5 66.5 0 00-1.653.545"/><path stroke="hsl(132, 100%, 50%)" d="M116.113 71.238a66.5 66.5 0 00-1.663.517"/><path stroke="hsl(133, 100%, 50%)" d="M117.229 70.919a66.5 66.5 0 00-1.672.487"/><path stroke="hsl(134, 100%, 50%)" d="M118.35 70.618a66.5 66.5 0 00-1.68.458"/><path stroke="hsl(135, 100%, 50%)" d="M119.476 70.337a66.5 66.5 0 00-1.687.429"/><path stroke="hsl(136, 100%, 50%)" d="M120.607 70.076a66.5 66.5 0 00-1.695.4"/><path stroke="hsl(137, 100%, 50%)" d="M121.742 69.835a66.5 66.5 0 00-1.701.37"/><path stroke="hsl(138, 100%, 50%)" d="M122.881 69.614a66.5 66.5 0 00-1.707.34"/><path stroke="hsl(139, 100%, 50%)" d="M124.024 69.412a66.5 66.5 0 00-1.713.31"/><path stroke="hsl(140, 100%, 50%)" d="M125.17 69.23a66.5 66.5 0 00-1.718.28"/><path stroke="hsl(141, 100%, 50%)" d="M126.32 69.069a66.5 66.5 0 00-1.723.25"/><path stroke="hsl(142, 100%, 50%)" d="M127.472 68.927a66.5 66.5 0 00-1.727.22"/><path stroke="hsl(143, 100%, 50%)" d="M128.626 68.806a66.5 66.5 0 00-1.73.19"/><path stroke="hsl(144, 100%, 50%)" d="M129.782 68.705a66.5 66.5 0 00-1.733.16"/><path stroke="hsl(145, 100%, 50%)" d="M130.94 68.624a66.5 66.5 0 00-1.736.13"/><path stroke="hsl(146, 100%, 50%)" d="M132.1 68.563a66.5 66.5 0 00-1.739.099"/><path stroke="hsl(147, 100%, 50%)" d="M133.26 68.523a66.5 66.5 0 00-1.74.068"/><path stroke="hsl(148, 100%, 50%)" d="M134.42 68.503a66.5 66.5 0 00-1.74.038"/><path stroke="hsl(149, 100%, 50%)" d="M135.58 68.503a66.5 66.5 0 00-1.74.007"/><path stroke="hsl(150, 100%, 50%)" d="M136.74 68.523A66.5 66.5 0 00135 68.5"/><path stroke="hsl(151, 100%, 50%)" d="M137.9 68.563a66.5 66.5 0 00-1.74-.053"/><path stroke="hsl(152, 100%, 50%)" d="M139.06 68.624a66.5 66.5 0 00-1.74-.083"/><path stroke="hsl(153, 100%, 50%)" d="M140.218 68.705a66.5 66.5 0 00-1.738-.114"/><path stroke="hsl(154, 100%, 50%)" d="M141.374 68.806a66.5 66.5 0 00-1.735-.144"/><path stroke="hsl(155, 100%, 50%)" d="M142.528 68.927a66.5 66.5 0 00-1.732-.174"/><path stroke="hsl(156, 100%, 50%)" d="M143.68 69.069a66.5 66.5 0 00-1.729-.205"/><path stroke="hsl(157, 100%, 50%)" d="M144.83 69.23a66.5 66.5 0 00-1.726-.234"/><path stroke="hsl(158, 100%, 50%)" d="M145.976 69.412a66.5 66.5 0 00-1.721-.265"/><path stroke="hsl(159, 100%, 50%)" d="M147.119 69.614a66.5 66.5 0 00-1.716-.295"/><path stroke="hsl(160, 100%, 50%)" d="M148.258 69.835a66.5 66.5 0 00-1.71-.325"/><path stroke="hsl(161, 100%, 50%)" d="M149.393 70.076a66.5 66.5 0 00-1.704-.354"/><path stroke="hsl(162, 100%, 50%)" d="M150.524 70.337a66.5 66.5 0 00-1.698-.384"/><path stroke="hsl(163, 100%, 50%)" d="M151.65 70.618a66.5 66.5 0 00-1.69-.414"/><path stroke="hsl(164, 100%, 50%)" d="M152.771 70.919a66.5 66.5 0 00-1.683-.444"/><path stroke="hsl(165, 100%, 50%)" d="M153.887 71.238a66.5 66.5 0 00-1.676-.472"/><path stroke="hsl(166, 100%, 50%)" d="M154.997 71.578a66.5 66.5 0 00-1.667-.502"/><path stroke="hsl(167, 100%, 50%)" d="M156.1 71.936a66.5 66.5 0 00-1.657-.53"/><path stroke="hsl(168, 100%, 50%)" d="M157.198 72.314a66.5 66.5 0 00-1.648-.56"/><path stroke="hsl(169, 100%, 50%)" d="M158.289 72.711a66.5 66.5 0 00-1.639-.588"/><path stroke="hsl(170, 100%, 50%)" d="M159.372 73.127a66.5 66.5 0 00-1.628-.617"/><path stroke="hsl(171, 100%, 50%)" d="M160.448 73.562a66.5 66.5 0 00-1.617-.645"/><path stroke="hsl(172, 100%, 50%)" d="M161.517 74.016a66.5 66.5 0 00-1.606-.674"/><path stroke="hsl(173, 100%, 50%)" d="M162.577 74.488a66.5 66.5 0 00-1.593-.702"/><path stroke="hsl(174, 100%, 50%)" d="M163.629 74.978a66.5 66.5 0 00-1.581-.729"/><path stroke="hsl(175, 100%, 50%)" d="M164.672 75.487a66.5 66.5 0 00-1.568-.756"/><path stroke="hsl(176, 100%, 50%)" d="M165.706 76.014a66.5 66.5 0 00-1.554-.784"/><path stroke="hsl(177, 100%, 50%)" d="M166.731 76.559a66.5 66.5 0 00-1.54-.81"/><path stroke="hsl(178, 100%, 50%)" d="M167.746 77.121a66.5 66.5 0 00-1.526-.837"/><path stroke="hsl(179, 100%, 50%)" d="M168.751 77.702a66.5 66.5 0 00-1.511-.864"/><path stroke="hsl(180, 100%, 50%)" d="M169.746 78.3a66.5 66.5 0 00-1.496-.89"/><path stroke="hsl(181, 100%, 50%)" d="M170.73 78.914a66.5 66.5 0 00-1.48-.916"/><path stroke="hsl(182, 100%, 50%)" d="M171.704 79.547a66.5 66.5 0 00-1.464-.942"/><path stroke="hsl(183, 100%, 50%)" d="M172.666 80.196a66.5 66.5 0 00-1.448-.968"/><path stroke="hsl(184, 100%, 50%)" d="M173.617 80.861a66.5 66.5 0 00-1.43-.992"/><path stroke="hsl(185, 100%, 50%)" d="M174.556 81.544a66.5 66.5 0 00-1.413-1.018"/><path stroke="hsl(186, 100%, 50%)" d="M175.483 82.242a66.5 66.5 0 00-1.395-1.042"/><path stroke="hsl(187, 100%, 50%)" d="M176.397 82.957a66.5 66.5 0 00-1.376-1.066"/><path stroke="hsl(188, 100%, 50%)" d="M177.3 83.687a66.5 66.5 0 00-1.359-1.09"/><path stroke="hsl(189, 100%, 50%)" d="M178.188 84.433a66.5 66.5 0 00-1.338-1.113"/><path stroke="hsl(190, 100%, 50%)" d="M179.064 85.194a66.5 66.5 0 00-1.319-1.136"/><path stroke="hsl(191, 100%, 50%)" d="M179.927 85.971a66.5 66.5 0 00-1.3-1.16"/><path stroke="hsl(192, 100%, 50%)" d="M180.776 86.763a66.5 66.5 0 00-1.279-1.182"/><path stroke="hsl(193, 100%, 50%)" d="M181.61 87.569a66.5 66.5 0 00-1.257-1.204"/><path stroke="hsl(194, 100%, 50%)" d="M182.431 88.39a66.5 66.5 0 00-1.236-1.226"/><path stroke="hsl(195, 100%, 50%)" d="M183.237 89.224a66.5 66.5 0 00-1.214-1.247"/><path stroke="hsl(196, 100%, 50%)" d="M184.029 90.073a66.5 66.5 0 00-1.193-1.268"/><path stroke="hsl(197, 100%, 50%)" d="M184.806 90.936a66.5 66.5 0 00-1.171-1.289"/><path stroke="hsl(198, 100%, 50%)" d="M185.567 91.812a66.5 66.5 0 00-1.148-1.31"/><path stroke="hsl(199, 100%, 50%)" d="M186.313 92.7a66.5 66.5 0 00-1.125-1.328"/><path stroke="hsl(200, 100%, 50%)" d="M187.043 93.603a66.5 66.5 0 00-1.101-1.348"/><path stroke="hsl(201, 100%, 50%)" d="M187.758 94.517a66.5 66.5 0 00-1.078-1.367"/><path stroke="hsl(202, 100%, 50%)" d="M188.456 95.444a66.5 66.5 0 00-1.053-1.385"/><path stroke="hsl(203, 100%, 50%)" d="M189.139 96.383a66.5 66.5 0 00-1.03-1.404"/><path stroke="hsl(204, 100%, 50%)" d="M189.804 97.334a66.5 66.5 0 00-1.004-1.422"/><path stroke="hsl(205, 100%, 50%)" d="M190.453 98.296a66.5 66.5 0 00-.98-1.439"/><path stroke="hsl(206, 100%, 50%)" d="M191.086 99.27a66.5 66.5 0 00-.955-1.456"/><path stroke="hsl(207, 100%, 50%)" d="M191.7 100.254a66.5 66.5 0 00-.928-1.472"/><path stroke="hsl(208, 100%, 50%)" d="M192.298 101.249a66.5 66.5 0 00-.903-1.489"/><path stroke="hsl(209, 100%, 50%)" d="M192.879 102.254a66.5 66.5 0 00-.877-1.504"/><path stroke="hsl(210, 100%, 50%)" d="M193.441 103.269a66.5 66.5 0 00-.85-1.519"/><path stroke="hsl(211, 100%, 50%)" d="M193.986 104.294a66.5 66.5 0 00-.824-1.534"/><path stroke="hsl(212, 100%, 50%)" d="M194.513 105.328a66.5 66.5 0 00-.797-1.548"/><path stroke="hsl(213, 100%, 50%)" d="M195.022 106.371a66.5 66.5 0 00-.77-1.561"/><path stroke="hsl(214, 100%, 50%)" d="M195.512 107.423a66.5 66.5 0 00-.742-1.575"/><path stroke="hsl(215, 100%, 50%)" d="M195.984 108.483a66.5 66.5 0 00-.715-1.587"/><path stroke="hsl(216, 100%, 50%)" d="M196.438 109.552a66.5 66.5 0 00-.687-1.6"/><path stroke="hsl(217, 100%, 50%)" d="M196.873 110.628a66.5 66.5 0 00-.66-1.612"/><path stroke="hsl(218, 100%, 50%)" d="M197.289 111.711a66.5 66.5 0 00-.631-1.622"/><path stroke="hsl(219, 100%, 50%)" d="M197.686 112.802a66.5 66.5 0 00-.603-1.633"/><path stroke="hsl(220, 100%, 50%)" d="M198.064 113.9a66.5 66.5 0 00-.574-1.644"/><path stroke="hsl(221, 100%, 50%)" d="M198.422 115.003a66.5 66.5 0 00-.545-1.653"/><path stroke="hsl(222, 100%, 50%)" d="M198.762 116.113a66.5 66.5 0 00-.517-1.663"/><path stroke="hsl(223, 100%, 50%)" d="M199.081 117.229a66.5 66.5 0 00-.487-1.672"/><path stroke="hsl(224, 100%, 50%)" d="M199.382 118.35a66.5 66.5 0 00-.458-1.68"/><path stroke="hsl(225, 100%, 50%)" d="M199.663 119.476a66.5 66.5 0 00-.429-1.687"/><path stroke="hsl(226, 100%, 50%)" d="M199.924 120.607a66.5 66.5 0 00-.4-1.695"/><path stroke="hsl(227, 100%, 50%)" d="M200.165 121.742a66.5 66.5 0 00-.37-1.701"/><path stroke="hsl(228, 100%, 50%)" d="M200.386 122.881a66.5 66.5 0 00-.34-1.707"/><path stroke="hsl(229, 100%, 50%)" d="M200.588 124.024a66.5 66.5 0 00-.31-1.713"/><path stroke="hsl(230, 100%, 50%)" d="M200.77 125.17a66.5 66.5 0 00-.28-1.718"/><path stroke="hsl(231, 100%, 50%)" d="M200.931 126.32a66.5 66.5 0 00-.25-1.723"/><path stroke="hsl(232, 100%, 50%)" d="M201.073 127.472a66.5 66.5 0 00-.22-1.727"/><path stroke="hsl(233, 100%, 50%)" d="M201.194 128.626a66.5 66.5 0 00-.19-1.73"/><path stroke="hsl(234, 100%, 50%)" d="M201.295 129.782a66.5 66.5 0 00-.16-1.733"/><path stroke="hsl(235, 100%, 50%)" d="M201.376 130.94a66.5 66.5 0 00-.13-1.736"/><path stroke="hsl(236, 100%, 50%)" d="M201.437 132.1a66.5 66.5 0 00-.099-1.739"/><path stroke="hsl(237, 100%, 50%)" d="M201.477 133.26a66.5 66.5 0 00-.068-1.74"/><path stroke="hsl(238, 100%, 50%)" d="M201.497 134.42a66.5 66.5 0 00-.038-1.74"/><path stroke="hsl(239, 100%, 50%)" d="M201.497 135.58a66.5 66.5 0 00-.007-1.74"/></g><circle cx="135" cy="135" r="133" fill="url(#a)" class="IroWheelSaturation"/><circle cx="135" cy="135" r="133" fill="none" stroke="#fff" stroke-width="2" class="IroWheelBorder"/></svg>'; +/*-------------------------------------------------------------- +# IMPORT +--------------------------------------------------------------*/ - dialog.querySelector('.satus-scrollbar__content').appendChild(close); - dialog.querySelector('.satus-scrollbar__content').appendChild(component_canvas); +Satus.storage.import = function(callback) { + chrome.storage.local.get(function(items) { + for (var key in items) { + Satus.storage[key] = items[key]; + } - document.body.appendChild(dialog); + if (callback) { + callback(); + } }); +}; - component.appendChild(component_value); - return component; -}; /*-------------------------------------------------------------- ->>> DIALOG +# CLEAR --------------------------------------------------------------*/ -Satus.components.dialog = function(element) { - var component = document.createElement('div'), - component_scrim = document.createElement('div'), - component_surface = document.createElement('div'), - component_scrollbar = Satus.components.scrollbar(component_surface), - options = element.options || {}; - - component_scrim.className = 'satus-dialog__scrim'; - component_surface.className = 'satus-dialog__surface'; - - for (var key in element) { - Satus.render(element[key], component_scrollbar); - } - - function close() { - window.removeEventListener('keydown', keydown); - - component.classList.add('satus-dialog--closing'); +Satus.storage.clear = function() { + chrome.storage.local.clear(); - if (typeof element.onclose === 'function') { - element.onclose(); + for (var key in Satus.storage) { + if (typeof Satus.storage[key] !== 'function') { + delete Satus.storage[key]; } - - setTimeout(function() { - component.remove(); - }, Satus.getAnimationDuration(component_surface)); } +}; + +/*----------------------------------------------------------------------------- +>>> «SEARCH» MODULE +-----------------------------------------------------------------------------*/ - function keydown(event) { - if (event.keyCode === 27) { - event.preventDefault(); - close(); - } - - if (event.keyCode === 9) { - var elements = component_surface.querySelectorAll('button, input'), - focused = false; - - event.preventDefault(); +Satus.search = function(query, object, callback, categories) { + var threads = 0, + folder = '', + results = {}; - for (var i = 0, l = elements.length; i < l; i++) { - if (elements[i] === document.activeElement && elements[i + 1]) { - elements[i + 1].focus(); + function parse(items) { + threads++; - focused = true; + for (var key in items) { + var item = items[key]; + + if (categories === true && item.type === 'folder' && folder !== item.label) { + folder = item.label; + } - i = l; + if (['switch', 'select', 'slider'].indexOf(item.type) !== -1 && key.indexOf(query) !== -1) { + if (categories === true) { + if (!results[folder]) { + results[folder] = {}; + } + + results[folder][key] = item; + } else { + results[key] = item; } } - if (focused === false) { - elements[0].focus(); + if (typeof item === 'object') { + parse(item); } } - } - component_scrim.addEventListener('click', close); - window.addEventListener('keydown', keydown); + threads--; - component.appendChild(component_scrim); - component.appendChild(component_surface); + if (threads === 0) { + callback(results); + } + } - // OPTIONS + parse(object); +}; + +/*-------------------------------------------------------------- +>>> RENDER +--------------------------------------------------------------*/ - if (options.left) { - component_surface.style.left = options.left + 'px'; +Satus.render = function(element, container, callback) { + function convert(object) { + if (object && object.type) { + var type = Satus.camelize(object.type), + component = Satus.components[type](object), + excluded_properties = ['type', 'label', 'class', 'title', 'storage']; + + function applyProperties(object, target) { + for (var key in object) { + if (Satus.isset(object[key]) && typeof object[key] === 'object' && !object[key].type) { + if (typeof target[key] !== 'object') { + target[key] = {}; + } + + applyProperties(object[key], target[key]); + } else if (excluded_properties.indexOf(key) === -1) { + target[key] = object[key]; + } + } + } + + applyProperties(object, component); + + component.skelet = object; + + component.classList.add('satus-' + object.type); + + if (object.class) { + var class_list = object.class.split(' '); + + for (var i = 0, l = class_list.length; i < l; i++) { + component.classList.add(class_list[i]); + } + } + + if (object.before) { + var component_before = document.createElement('span'); + + component_before.innerHTML = object.before; + + for (var i = component_before.children.length - 1; i > -1; i--) { + component.insertBefore(component_before.children[i], component.firstChild); + } + } + + if (object.after) { + var component_after = document.createElement('span'); + + component_after.innerHTML = object.after; + + for (var i = component_after.children.length - 1; i > -1; i--) { + component.appendChild(component_after.children[i]); + } + } + + (container || document.body).appendChild(component); + + if (typeof component.onClickRender === 'object') { + component.addEventListener('click', function() { + Satus.render(component.onClickRender); + }); + } + + if (Satus.isset(Satus.events.render)) { + for (var i = 0, l = Satus.events.render.length; i < l; i++) { + Satus.events.render[i](component, object); + } + } + + if (typeof component.onrender === 'function') { + component.onrender(object); + } + + if (callback) { + callback(); + } + } + } + + if (element.type) { + convert(element); + } else { + for (var key in element) { + convert(element[key]); + } + } +}; + +/*-------------------------------------------------------------- +>>> STORAGE KEYS +--------------------------------------------------------------*/ + +Satus.modules.updateStorageKeys = function(object, callback) { + var threads = 0; + + function parse(items) { + threads++; + + for (var key in items) { + var item = items[key]; + + + if (item.type) { + item.storage_key = key; + } + + if (typeof item === 'object') { + parse(item); + } + } + + threads--; + + if (threads === 0) { + callback(); + } } - if (options.top) { - component_surface.style.top = options.top + 'px'; - } + parse(object); +}; +Satus.components.table = function(item) { + var component = document.createElement('div'), + component_head = document.createElement('div'), + component_body = document.createElement('div'), + component_scrollbar = Satus.components.scrollbar(component_body, item.scrollbar), + table = document.createElement('div'); + + table.className = 'satus-table__container'; + component_head.className = 'satus-table__head'; + component_body.className = 'satus-table__body'; + + function update(data) { + var pages = Math.ceil(component.data.length / component.paging), + start = Math.max((component.pagingIndex - 1) * component.paging, 0), + end = component.pagingIndex * component.paging; + + if (end > data.length) { + end = data.length; + } else if (end === 0) { + end = component.paging; + } + + table.innerHTML = ''; + + if (data) { + for (var i = start, l = end; i < l; i++) { + if (data[i]) { + var tr = document.createElement('div'); + + tr.className = 'satus-table__row'; + + for (var j = 0, k = data[i].length; j < k; j++) { + var td = document.createElement('div'); + + + td.className = 'satus-table__cell'; + + if (data[i][j].html) { + td.innerHTML = data[i][j].html; + } else if (data[i][j].text) { + td.innerText = data[i][j].text; + } + + if (item.columns[j].onrender) { + td.onrender = item.columns[j].onrender; + + td.onrender(); + } + + tr.appendChild(td); + } + + table.appendChild(tr); + } + } + } + + component.pagingUpdate(); + } + + function sortArray(array, index, mode) { + if (mode === 'asc') { + if (typeof array[0][index].text === 'number') { + sorted = array.sort(function(a, b) { + return a[index].text - b[index].text; + }); + } else { + sorted = array.sort(function(a, b) { + return a[index].text.localeCompare(b[index].text); + }); + } + } else { + if (typeof array[0][index].text === 'number') { + sorted = array.sort(function(a, b) { + return b[index].text - a[index].text; + }); + } else { + sorted = array.sort(function(a, b) { + return b[index].text.localeCompare(a[index].text); + }); + } + } + + return array; + } + + function sort() { + var mode = this.dataset.sorting, + index = Array.prototype.indexOf.call(this.parentElement.children, this), + sorted; + + if (component.data[0][index] && component.data[0][index].hasOwnProperty('text')) { + if (mode === 'none') { + mode = 'asc'; + } else if (mode === 'asc') { + mode = 'desc'; + } else if (mode === 'desc') { + mode = 'asc'; + } + + if (this.parentNode.querySelector('div[data-sorting=asc], div[data-sorting=desc]')) { + this.parentNode.querySelector('div[data-sorting=asc], div[data-sorting=desc]').dataset.sorting = 'none'; + } + + this.dataset.sorting = mode; + + sorted = sortArray(component.data, index, mode); + + update(sorted); + } else { + this.dataset.sorting = false; + } + } + + function resize() {} + + for (var i = 0, l = item.columns.length; i < l; i++) { + var column = document.createElement('div'); + + column.dataset.sorting = 'none'; + column.addEventListener('click', sort); + column.innerHTML = '' + item.columns[i].title + ''; + + component_head.appendChild(column); + } + + component_scrollbar.appendChild(table); + + component.appendChild(component_head); + component.appendChild(component_body); + + component.data = item.data; + component.paging = item.paging; + component.pagingIndex = 0; + + component.update = function(data, index, mode) { + if (Satus.isset(data)) { + this.data = data; + } + + if (this.querySelector('div[data-sorting=asc], div[data-sorting=desc]')) { + var mode = this.querySelector('div[data-sorting=asc], div[data-sorting=desc]').dataset.sorting, + index = Array.prototype.indexOf.call(this.querySelector('div[data-sorting=asc], div[data-sorting=desc]').parentElement.children, this.querySelector('div[data-sorting=asc], div[data-sorting=desc]')); + + update(sortArray(this.data, index, mode)); + } else { + for (var i = 0, l = item.columns.length; i < l; i++) { + if (item.columns[i].hasOwnProperty('sorting')) { + if (this.data[0][i].hasOwnProperty('text')) { + this.querySelectorAll('.satus-table__head > div')[i].dataset.sorting = item.columns[i].sorting; + } else { + this.querySelectorAll('.satus-table__head > div')[i].dataset.sorting = false; + } + + update(sortArray(this.data, i, item.columns[i].sorting)); + + i = l; + } + } + } + }; + + + // PAGING + + function pagingUpdate() { + if (typeof this.paging === 'number') { + var pages = Math.ceil(this.data.length / this.paging); + + this.querySelector('.satus-table__paging').innerHTML = ''; + + for (var i = 1; i <= pages; i++) { + var button = document.createElement('button'); + + if (i === (this.pagingIndex || 1)) { + button.className = 'active'; + } + + button.innerText = i; + button.parentComponent = this; + button.addEventListener('click', function() { + if (this.parentNode.querySelector('button.active')) { + this.parentNode.querySelector('button.active').classList.remove('active'); + } + + this.classList.add('active'); + + this.parentComponent.pagingIndex = Number(this.innerText); + this.parentComponent.update(this.parentComponent.data); + }); + + this.querySelector('.satus-table__paging').appendChild(button); + } + } + + resize(); + } + + component.pagingUpdate = pagingUpdate; + + component_paging = document.createElement('div'); + + component_paging.className = 'satus-table__paging'; + + component_scrollbar.appendChild(component_paging); - if (options.width) { - component_surface.style.width = options.width + 'px'; + // END PAGING + + if (item.data) { + component.update(item.data); } + + return component; +}; + +/*-------------------------------------------------------------- +>>> HEADER +--------------------------------------------------------------*/ - if (options.height) { - component_surface.style.height = options.height + 'px'; - } +Satus.components.header = function(object) { + var component = document.createElement('header'); - // END OPTIONS + for (var key in object) { + Satus.render(object[key], component); + } return component; }; /*-------------------------------------------------------------- ->>> FOLDER +>>> SELECT --------------------------------------------------------------*/ -Satus.components.folder = function(object) { - var component = document.createElement('button'); - - component.object = object; +Satus.components.select = function(element) { + var component = document.createElement('button'), + component_label = document.createElement('span'), + component_value = document.createElement('span'), + label = Satus.locale.getMessage(element.label); component.classList.add('satus-button'); - component.addEventListener('click', function() { - var parent = document.querySelector(component.object.parent) || document.querySelector('.satus-main'); + component_label.className = 'satus-select__label'; + component_label.innerText = label; - if (!component.object.parent || !parent.classList.contains('satus-main')) { - while (!parent.classList.contains('satus-main')) { - parent = parent.parentNode; + component_value.className = 'satus-select__value'; + + if (element.storage_key) { + var value = Satus.storage.get(element.storage_key); + + component.dataset.storageKey = element.storage_key; + + for (var i = 0, l = element.options.length; i < l; i++) { + if (value === element.options[i].value) { + value = element.options[i].label; } } - parent.open(this.object, object.onopen); - }); + component_value.innerText = Satus.locale.getMessage(value || element.options[0].label); + } - if (Satus.isset(object.label)) { - var component_label = document.createElement('span'); + component.onclick = function() { + var position = this.getBoundingClientRect(), + dialog = { + type: 'dialog', + class: 'satus-dialog--select-component' + }; - component_label.className = 'satus-folder__label'; - component_label.innerText = Satus.locale.getMessage(object.label); + for (var key in element.options) { + dialog[key] = element.options[key]; - component.appendChild(component_label); - } + dialog[key].type = 'button'; + dialog[key].dataset = {}; + dialog[key].dataset.key = element.options[key].label; + dialog[key].dataset.value = element.options[key].value; + dialog[key].onclick = function() { + component_value.innerText = Satus.locale.getMessage(this.dataset.key); - return component; -}; -/*-------------------------------------------------------------- ->>> HEADER ---------------------------------------------------------------*/ + Satus.storage.set(component.dataset.storageKey, this.dataset.value); -Satus.components.header = function(object) { - var component = document.createElement('header'); + var parent = this.parentNode; - for (var key in object) { - Satus.render(object[key], component); - } + while (!parent.classList.contains('satus-dialog')) { + parent = parent.parentNode; + } + + parent.querySelector('.satus-dialog__scrim').click(); + }; + } + + Satus.render(dialog); + }; + + component.appendChild(component_label); + component.appendChild(component_value); return component; }; /*-------------------------------------------------------------- ->>> LIST +>>> SWITCH --------------------------------------------------------------*/ -Satus.components.list = function(object) { - var ul = document.createElement('ul'); +Satus.components.switch = function(element) { + var component = document.createElement('div'), + value; - if (object.compact === true) { - ul.classList.add('satus-list'); - ul.classList.add('satus-list--compact'); + // LABEL + if (Satus.isset(element.label)) { + var component_label = document.createElement('span'); + + component_label.className = 'satus-switch__label'; + component_label.innerText = Satus.locale.getMessage(element.label); + + component.appendChild(component_label); } - for (var key in object) { - if (Satus.isset(object[key].type)) { - var li = document.createElement('li'); - if (object.sortable === true) { - function mousedown(event) { - if (event.button === 0) { - var self = this, - dragging = false, - clone = false, - current_index = Array.from(self.parentNode.children).indexOf(self), - bounding = this.getBoundingClientRect(), - first_x = event.clientX, - first_y = event.clientY, - offset_x = event.clientX - bounding.left, - offset_y = event.clientY - bounding.top; + // INPUT + var component_input = document.createElement('input'); - function mousemove(event) { - if (Math.abs(first_y - event.clientY) <= 5) { - return false; - } - - if (dragging === false) { - clone = self.cloneNode(true); + component_input.type = 'checkbox'; + component_input.className = 'satus-switch__input'; - Satus.cloneNodeStyles(self, clone); - clone.style.position = 'fixed'; - clone.style.pointerEvents = 'none'; - clone.style.backgroundColor = '#fff'; - self.style.visibility = 'hidden'; + if (element.storage_key) { + value = Satus.storage.get(element.storage_key); - document.body.appendChild(clone); + component_input.dataset.storageKey = element.storage_key; + } - dragging = true; - } + if (!Satus.isset(value)) { + value = element.value; + } + + if (value) { + component_input.checked = value; + } + + component_input.addEventListener('change', function() { + Satus.storage.set(this.dataset.storageKey, this.checked); + }); + + component.appendChild(component_input); + + + // TRACK + var component_track = document.createElement('div'); + + component_track.className = 'satus-switch__track'; + + component.appendChild(component_track); + + + // MOUSE MOVE + component_track.addEventListener('mousedown', function(event) { + var prevent = false, + difference = 0; + + function click(event) { + event.preventDefault(); + event.stopPropagation(); + + component.removeEventListener('click', click); + + return false; + } + + function mousemove(event) { + var checkbox = component.querySelector('input'), + movement = event.movementX; + + if (movement * difference < 0) { + difference = 0; + } else { + difference += movement; + + if (prevent === false) { + prevent = true; + component.addEventListener('click', click); + } + } + + if (difference < -5) { + checkbox.checked = false; + } else if (difference > 5) { + checkbox.checked = true; + } + } + + function mouseup(event) { + window.removeEventListener('mousemove', mousemove); + window.removeEventListener('mouseup', mouseup); + } + + window.addEventListener('mousemove', mousemove); + window.addEventListener('mouseup', mouseup); + }); + + + // TOUCH MOVE + component_track.addEventListener('touchstart', function(event) { + var previous_x = 0, + difference = 0; + + function mousemove(event) { + var checkbox = component.querySelector('input'), + movement = event.touches[0].clientX - previous_x; + + previous_x = event.touches[0].clientX; + + if (movement * difference < 0) { + difference = 0; + } else { + difference += movement; + } + + if (difference < -5) { + checkbox.checked = false; + } else if (difference > 5) { + checkbox.checked = true; + } + } + + function mouseup(event) { + window.removeEventListener('touchmove', mousemove); + window.removeEventListener('touchend', mouseup); + } - var x = bounding.left, //event.clientX - offset_x - y = event.clientY - offset_y, - index = Math.floor(y / self.offsetHeight) - 1; + window.addEventListener('touchmove', mousemove); + window.addEventListener('touchend', mouseup); + }); - clone.style.left = x + 'px'; - clone.style.top = y + 'px'; - - //return false; - if (index !== current_index && self.parentNode.children[index]) { - var new_clone = self.cloneNode(true); + return component; +}; +/*-------------------------------------------------------------- +>>> TABS +--------------------------------------------------------------*/ - if (index > 0) { - self.parentNode.insertBefore(new_clone, self.parentNode.children[index].nextSibling); - } else { - self.parentNode.insertBefore(new_clone, self.parentNode.children[index]); - } +Satus.components.tabs = function(object) { + var component = document.createElement('div'), + tabbar = document.createElement('div'), + tabbar_select = document.createElement('div'), + main = document.createElement('div'), + i = 0; + + tabbar.className = 'satus-tabs__bar'; + main.className = 'satus-tabs__main'; + tabbar_select.className = 'satus-tabs__bar--select'; + + tabbar.appendChild(tabbar_select); + + function update() { + var index = Number(this.dataset.key); + + tabbar_select.style.left = this.offsetLeft + 'px'; + + if (this.parentNode.querySelector('.active')) { + var prev_index = Number(this.parentNode.querySelector('.active').dataset.key); + + this.parentNode.querySelector('.active').classList.remove('active'); + } + + this.classList.add('active'); + + var container = document.createElement('div'); + + container.className = 'satus-tabs__tab'; + + satus.render(this.menu, container); + + if (main.children.length >= 1) { + container.classList.add(index > prev_index ? 'satus-animation--fade-in-right' : 'satus-animation--fade-in-left'); + + main.children[0].classList.add('old'); + main.children[0].classList.add(index > prev_index ? 'satus-animation--fade-out-left' : 'satus-animation--fade-out-right'); + + main.appendChild(container); + + setTimeout(function() { + main.children[0].remove(); - self.remove(); + container.classList.remove(index > prev_index ? 'satus-animation--fade-in-right' : 'satus-animation--fade-in-left'); + }, 250); + } else { + main.appendChild(container); + } + } + + for (var key in object) { + if (object[key].type === 'tab') { + var tab = document.createElement('div'); - self = new_clone; + tab.innerText = satus.locale.getMessage(object[key].label); + tab.dataset.key = i; + tab.onclick = update; + tab.menu = Object.assign({}, object[key]); + + delete tab.menu.type; + + tabbar.appendChild(tab); + + i++; + } + } + + tabbar.children[1].click(); - self.addEventListener('mousedown', mousedown); + component.appendChild(tabbar); + component.appendChild(main); - if (typeof object.onchange === 'function') { - object.onchange(current_index, index); - } + return component; +}; + +/*-------------------------------------------------------------- +>>> FOLDER +--------------------------------------------------------------*/ - current_index = index; - } - } +Satus.components.folder = function(object) { + var component = document.createElement('button'); - function mouseup(event) { - if (clone) { - clone.remove(); - self.style.visibility = ''; - } + component.object = object; - window.removeEventListener('mousemove', mousemove); - window.removeEventListener('mouseup', mouseup); - } + component.classList.add('satus-button'); - window.addEventListener('mousemove', mousemove); - window.addEventListener('mouseup', mouseup); - } - } + component.addEventListener('click', function() { + var parent = document.querySelector(component.object.parent) || document.querySelector('.satus-main'); - li.addEventListener('mousedown', mousedown); + if (!component.object.parent || !parent.classList.contains('satus-main')) { + while (!parent.classList.contains('satus-main')) { + parent = parent.parentNode; } + } - Satus.render(object[key], li); + parent.open(this.object, object.onopen); + }); - ul.appendChild(li); - } + if (Satus.isset(object.label)) { + var component_label = document.createElement('span'); + + component_label.className = 'satus-folder__label'; + component_label.innerText = Satus.locale.getMessage(object.label); + + component.appendChild(component_label); } - return ul; -}; - + return component; +}; +/*-------------------------------------------------------------- +>>> TEXT FIELD +--------------------------------------------------------------*/ + +Satus.components.textField = function(element) { + var component = element.rows > 1 ? document.createElement('textarea') : document.createElement('input'); + + component.type = 'text'; + + return component; +}; /*-------------------------------------------------------------- >>> MAIN --------------------------------------------------------------*/ @@ -1165,8 +1379,8 @@ Satus.components.main = function(object) { component_container = document.createElement('div'), component_scrollbar = Satus.components.scrollbar(component_container); - container.classList.add('satus-main__container--fade-out-right'); - component_container.className = 'satus-main__container satus-main__container--fade-in-left'; + container.classList.add('satus-animation--fade-out-right'); + component_container.className = 'satus-main__container satus-animation--fade-in-left'; this.history.pop(); @@ -1185,6 +1399,11 @@ Satus.components.main = function(object) { component_scrollbar.onopen(); } + + var id = this.history[this.history.length - 1].appearanceKey; + + document.body.dataset.appearance = id; + this.dataset.appearance = id; setTimeout(function() { container.remove(); @@ -1197,8 +1416,8 @@ Satus.components.main = function(object) { component_scrollbar = Satus.components.scrollbar(component_container); if (animated !== false) { - container.classList.add('satus-main__container--fade-out-left'); - component_container.className = 'satus-main__container satus-main__container--fade-in-right'; + container.classList.add('satus-animation--fade-out-left'); + component_container.className = 'satus-main__container satus-animation--fade-in-right'; } else { component_container.className = 'satus-main__container'; } @@ -1215,32 +1434,175 @@ Satus.components.main = function(object) { this.historyListener(component_container); } - if (callback) { - component_scrollbar.onopen = callback; + if (callback) { + component_scrollbar.onopen = callback; + + component_scrollbar.onopen(); + } + + var id = this.history[this.history.length - 1].appearanceKey; + + document.body.dataset.appearance = id; + this.dataset.appearance = id; + + setTimeout(function() { + container.remove(); + }, Satus.getAnimationDuration(container)); + }; + + component_container.className = 'satus-main__container'; + + if (object.on && object.on.change) { + component.historyListener = object.on.change; + } + + if (component.historyListener) { + component.historyListener(component_container); + } + + for (var key in object) { + Satus.render(object[key], component_scrollbar); + } + + component.appendChild(component_container); + + return component; +}; + +/*-------------------------------------------------------------- +>>> SLIDER +--------------------------------------------------------------*/ + +Satus.components.slider = function(element) { + var component = document.createElement('div'); + + // LABEL + if (Satus.isset(element.label)) { + var component_label = document.createElement('span'); + + component_label.className = 'satus-slider__label'; + component_label.innerText = Satus.locale.getMessage(element.label); + + component.appendChild(component_label); + } + + + // RANGE + var component_range = document.createElement('input'); + + component_range.type = 'range'; + component_range.className = 'satus-slider__range'; + component_range.min = element.min || 0; + component_range.max = element.max || 10; + component_range.step = element.step || 1; + + component_range.oninput = function() { + var track = this.parentNode.querySelector('.satus-slider__track'), + thumb = this.parentNode.querySelector('.satus-slider__thumb'), + min = Number(this.min) || 0, + max = Number(this.max) || 1, + step = Number(this.step) || 1, + value = Number(this.value) || 0, + offset = (value - min) / (max - min) * 100; + + track.style.width = 'calc(' + offset + '% - ' + Math.floor(offset * 12 / 100) + 'px)'; + + Satus.storage.set(this.dataset.storageKey, Number(this.value)); + + component_thumb.dataset.value = this.value; + + if (component.onchange) { + component.onchange(Number(this.value)); + } + }; + + component.change = function(value) { + component_range.value = value; + + component_thumb.dataset.value = value; + + component_range.oninput(); + }; + + component.addEventListener('mousedown', function() { + function mousemove() { + component.classList.add('satus-slider--dragging'); + } + + function mouseup() { + component.classList.remove('satus-slider--dragging'); + + window.removeEventListener('mousemove', mousemove); + window.removeEventListener('mouseup', mouseup); + } + + window.addEventListener('mousemove', mousemove); + window.addEventListener('mouseup', mouseup); + }); + + if (element.onchange) { + component.onchange = element.onchange; + } + + component.appendChild(component_range); + + + // CONTAINER + var component_container = document.createElement('div'); + + component_container.className = 'satus-slider__container'; + + component.appendChild(component_container); + + + // TRACK + var component_track_container = document.createElement('div'), + component_track = document.createElement('div'); + + component_track_container.className = 'satus-slider__track-container'; + component_track.className = 'satus-slider__track'; + + component_track_container.appendChild(component_track); + component_container.appendChild(component_track_container); + + + // FOCUS RING + var component_ring = document.createElement('div'); + + component_ring.className = 'satus-slider__ring'; + + component_track.appendChild(component_ring); + + + // THUMB + var component_thumb = document.createElement('div'); + + component_thumb.className = 'satus-slider__thumb'; - component_scrollbar.onopen(); - } + component_track.appendChild(component_thumb); - setTimeout(function() { - container.remove(); - }, Satus.getAnimationDuration(container)); - }; + if (element.storage_key) { + var value = Satus.storage.get(element.storage_key) || element.value; - component_container.className = 'satus-main__container'; + component_range.dataset.storageKey = element.storage_key; - if (object.on && object.on.change) { - component.historyListener = object.on.change; - } + if (value) { + component_range.value = value; - if (component.historyListener) { - component.historyListener(component_container); - } + if (!Satus.isset(value)) { + value = element.value; + } - for (var key in object) { - Satus.render(object[key], component_scrollbar); + var offset = (Number(component_range.value) - Number(component_range.min)) / (Number(component_range.max) - Number(component_range.min)) * 100; + + component_track.style.width = 'calc(' + offset + '% - ' + Math.floor(offset * 12 / 100) + 'px)'; + component_thumb.dataset.value = value; + } else { + component_range.value = 0; + component_thumb.dataset.value = 0; + } } - component.appendChild(component_container); return component; }; @@ -1350,813 +1712,624 @@ Satus.components.scrollbar = function(parent, enabled) { return component_content; }; /*-------------------------------------------------------------- ->>> SECTION +>>> DIV --------------------------------------------------------------*/ -Satus.components.section = function(element) { - var component = document.createElement('section'); +Satus.components.div = function(object) { + var component = document.createElement('div'); - for (var key in element) { - Satus.render(element[key], component); + for (var key in object) { + Satus.render(object[key], component); } return component; }; /*-------------------------------------------------------------- ->>> SELECT +>>> LIST --------------------------------------------------------------*/ -Satus.components.select = function(element) { - var component = document.createElement('button'), - component_label = document.createElement('span'), - component_value = document.createElement('span'), - label = Satus.locale.getMessage(element.label); - - component.classList.add('satus-button'); - - component_label.className = 'satus-select__label'; - component_label.innerText = label; - - component_value.className = 'satus-select__value'; - - if (element.storage_key) { - var value = Satus.storage.get(element.storage_key); - - component.dataset.storageKey = element.storage_key; - - for (var i = 0, l = element.options.length; i < l; i++) { - if (value === element.options[i].value) { - value = element.options[i].label; - } - } - - component_value.innerText = Satus.locale.getMessage(value || element.options[0].label); - } - - component.onclick = function() { - var position = this.getBoundingClientRect(), - dialog = { - type: 'dialog', - class: 'satus-dialog--select-component' - }; - - for (var key in element.options) { - dialog[key] = element.options[key]; - - dialog[key].type = 'button'; - dialog[key].dataset = {}; - dialog[key].dataset.key = element.options[key].label; - dialog[key].dataset.value = element.options[key].value; - dialog[key].onclick = function() { - component_value.innerText = Satus.locale.getMessage(this.dataset.key); - - Satus.storage.set(component.dataset.storageKey, this.dataset.value); - - var parent = this.parentNode; - - while (!parent.classList.contains('satus-dialog')) { - parent = parent.parentNode; - } - - parent.querySelector('.satus-dialog__scrim').click(); - }; - } - - Satus.render(dialog); - }; - - component.appendChild(component_label); - component.appendChild(component_value); - - return component; -}; -/*------------------------------------------------------------------------------ ->>> SHORTCUT -------------------------------------------------------------------------------*/ - -Satus.components.shortcut = function(element) { - var self = this, - value = (Satus.storage.get(element.storage_key) ? JSON.parse(Satus.storage.get(element.storage_key)) : false) || element.value || {}, - component = document.createElement('div'), - component_label = document.createElement('span'), - component_value = document.createElement('span'), - mousewheel_timeout = false, - mousewheel_only = false; - - component_label.className = 'satus-shortcut__label'; - component_value.className = 'satus-shortcut__value'; - - function update(canvas) { - let text_value = [], - keys_value = []; - - if (value.altKey === true) { - text_value.push('Alt'); - keys_value.push('