mirror of
https://github.com/axllent/mailpit.git
synced 2025-04-17 12:06:22 +02:00
UI: Theme toggler - auto, light and dark themes
This commit is contained in:
parent
497086cb65
commit
7748846b88
@ -4,6 +4,7 @@ import Message from './templates/Message.vue'
|
|||||||
import MessageSummary from './templates/MessageSummary.vue'
|
import MessageSummary from './templates/MessageSummary.vue'
|
||||||
import MessageRelease from './templates/MessageRelease.vue'
|
import MessageRelease from './templates/MessageRelease.vue'
|
||||||
import MessageToast from './templates/MessageToast.vue'
|
import MessageToast from './templates/MessageToast.vue'
|
||||||
|
import ThemeToggle from './templates/ThemeToggle.vue'
|
||||||
import moment from 'moment'
|
import moment from 'moment'
|
||||||
import Tinycon from 'tinycon'
|
import Tinycon from 'tinycon'
|
||||||
|
|
||||||
@ -14,7 +15,8 @@ export default {
|
|||||||
Message,
|
Message,
|
||||||
MessageSummary,
|
MessageSummary,
|
||||||
MessageRelease,
|
MessageRelease,
|
||||||
MessageToast
|
MessageToast,
|
||||||
|
ThemeToggle,
|
||||||
},
|
},
|
||||||
|
|
||||||
data() {
|
data() {
|
||||||
@ -775,13 +777,13 @@ export default {
|
|||||||
<img src="mailpit.svg" alt="Mailpit">
|
<img src="mailpit.svg" alt="Mailpit">
|
||||||
<span v-if="!total" class="ms-2">Mailpit</span>
|
<span v-if="!total" class="ms-2">Mailpit</span>
|
||||||
</a>
|
</a>
|
||||||
<div v-if="total" class="ms-md-2 d-flex bg-white border rounded-start flex-fill position-relative">
|
<div v-if="total" class="ms-md-2 d-flex border bg-body rounded-start flex-fill position-relative">
|
||||||
<input type="text" class="form-control border-0" aria-label="Search" v-model.trim="search"
|
<input type="text" class="form-control border-0" aria-label="Search" v-model.trim="search"
|
||||||
placeholder="Search mailbox">
|
placeholder="Search mailbox">
|
||||||
<span class="btn btn-link position-absolute end-0 text-muted" v-if="search"
|
<span class="btn btn-link position-absolute end-0 text-muted" v-if="search"
|
||||||
v-on:click="resetSearch"><i class="bi bi-x-circle"></i></span>
|
v-on:click="resetSearch"><i class="bi bi-x-circle"></i></span>
|
||||||
</div>
|
</div>
|
||||||
<button v-if="total" class="btn btn-outline-light" type="submit">
|
<button v-if="total" class="btn btn-outline-secondary" type="submit">
|
||||||
<i class="bi bi-search"></i>
|
<i class="bi bi-search"></i>
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
@ -807,8 +809,8 @@ export default {
|
|||||||
<i class="bi bi-check2-square"></i>
|
<i class="bi bi-check2-square"></i>
|
||||||
</button>
|
</button>
|
||||||
|
|
||||||
<select v-model="limit" v-on:change="loadMessages" class="form-select form-select-sm d-inline w-auto me-2"
|
<select v-model="limit" v-on:change="loadMessages" v-if="!searching"
|
||||||
v-if="!searching">
|
class="form-select form-select-sm d-none d-md-inline w-auto me-2">
|
||||||
<option value="25">25</option>
|
<option value="25">25</option>
|
||||||
<option value="50">50</option>
|
<option value="50">50</option>
|
||||||
<option value="100">100</option>
|
<option value="100">100</option>
|
||||||
@ -920,7 +922,7 @@ export default {
|
|||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
</div>
|
</div>
|
||||||
<div class="list-group mt-1 mb-5">
|
<div class="list-group mt-1 mb-5 pb-3">
|
||||||
<button class="list-group-item list-group-item-action small px-2" v-for="tag in tags"
|
<button class="list-group-item list-group-item-action small px-2" v-for="tag in tags"
|
||||||
:style="showTagColors ? { borderLeftColor: colorHash(tag), borderLeftWidth: '4px' } : ''"
|
:style="showTagColors ? { borderLeftColor: colorHash(tag), borderLeftWidth: '4px' } : ''"
|
||||||
v-on:click="tagSearch($event, tag)" :class="inSearch(tag) ? 'active' : ''">
|
v-on:click="tagSearch($event, tag)" :class="inSearch(tag) ? 'active' : ''">
|
||||||
@ -933,11 +935,13 @@ export default {
|
|||||||
|
|
||||||
<MessageSummary v-if="message" :message="message"></MessageSummary>
|
<MessageSummary v-if="message" :message="message"></MessageSummary>
|
||||||
|
|
||||||
<div class="position-fixed bottom-0 py-2 text-muted small w-100">
|
<div class="position-fixed bg-body bottom-0 py-2 text-muted small col-lg-2 col-md-3 pe-3 z-3">
|
||||||
<a href="#" class="text-muted" v-on:click="loadInfo">
|
<a href="#" class="text-muted btn btn-sm" v-on:click="loadInfo">
|
||||||
<i class="bi bi-info-circle-fill"></i>
|
<i class="bi bi-info-circle-fill"></i>
|
||||||
About
|
About
|
||||||
</a>
|
</a>
|
||||||
|
|
||||||
|
<ThemeToggle />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
@ -1196,53 +1200,4 @@ export default {
|
|||||||
</div>
|
</div>
|
||||||
|
|
||||||
<MessageToast v-if="toastMessage" :message="toastMessage" @clearMessageToast="clearMessageToast"></MessageToast>
|
<MessageToast v-if="toastMessage" :message="toastMessage" @clearMessageToast="clearMessageToast"></MessageToast>
|
||||||
|
|
||||||
<!-- Toggle theme -->
|
|
||||||
<svg xmlns="http://www.w3.org/2000/svg" style="display: none;">
|
|
||||||
<symbol id="bootstrap" viewBox="0 0 512 408" fill="currentcolor">
|
|
||||||
<path d="M106.342 0c-29.214 0-50.827 25.58-49.86 53.32.927 26.647-.278 61.165-8.966 89.31C38.802 170.862 24.07 188.707 0 191v26c24.069 2.293 38.802 20.138 47.516 48.37 8.688 28.145 9.893 62.663 8.965 89.311C55.515 382.42 77.128 408 106.342 408h299.353c29.214 0 50.827-25.58 49.861-53.319-.928-26.648.277-61.166 8.964-89.311 8.715-28.232 23.411-46.077 47.48-48.37v-26c-24.069-2.293-38.765-20.138-47.48-48.37-8.687-28.145-9.892-62.663-8.964-89.31C456.522 25.58 434.909 0 405.695 0H106.342zm236.559 251.102c0 38.197-28.501 61.355-75.798 61.355h-87.202a2 2 0 01-2-2v-213a2 2 0 012-2h86.74c39.439 0 65.322 21.354 65.322 54.138 0 23.008-17.409 43.61-39.594 47.219v1.203c30.196 3.309 50.532 24.212 50.532 53.085zm-84.58-128.125h-45.91v64.814h38.669c29.888 0 46.373-12.03 46.373-33.535 0-20.151-14.174-31.279-39.132-31.279zm-45.91 90.53v71.431h47.605c31.12 0 47.605-12.482 47.605-35.941 0-23.46-16.947-35.49-49.608-35.49h-45.602z"/>
|
|
||||||
</symbol>
|
|
||||||
<symbol id="check2" viewBox="0 0 16 16" fill="currentcolor">
|
|
||||||
<path d="M13.854 3.646a.5.5 0 0 1 0 .708l-7 7a.5.5 0 0 1-.708 0l-3.5-3.5a.5.5 0 1 1 .708-.708L6.5 10.293l6.646-6.647a.5.5 0 0 1 .708 0z"/>
|
|
||||||
</symbol>
|
|
||||||
<symbol id="circle-half" viewBox="0 0 16 16" fill="currentcolor">
|
|
||||||
<path d="M8 15A7 7 0 1 0 8 1v14zm0 1A8 8 0 1 1 8 0a8 8 0 0 1 0 16z"/>
|
|
||||||
</symbol>
|
|
||||||
<symbol id="moon-stars-fill" viewBox="0 0 16 16" fill="currentcolor">
|
|
||||||
<path d="M6 .278a.768.768 0 0 1 .08.858 7.208 7.208 0 0 0-.878 3.46c0 4.021 3.278 7.277 7.318 7.277.527 0 1.04-.055 1.533-.16a.787.787 0 0 1 .81.316.733.733 0 0 1-.031.893A8.349 8.349 0 0 1 8.344 16C3.734 16 0 12.286 0 7.71 0 4.266 2.114 1.312 5.124.06A.752.752 0 0 1 6 .278z"/>
|
|
||||||
<path d="M10.794 3.148a.217.217 0 0 1 .412 0l.387 1.162c.173.518.579.924 1.097 1.097l1.162.387a.217.217 0 0 1 0 .412l-1.162.387a1.734 1.734 0 0 0-1.097 1.097l-.387 1.162a.217.217 0 0 1-.412 0l-.387-1.162A1.734 1.734 0 0 0 9.31 6.593l-1.162-.387a.217.217 0 0 1 0-.412l1.162-.387a1.734 1.734 0 0 0 1.097-1.097l.387-1.162zM13.863.099a.145.145 0 0 1 .274 0l.258.774c.115.346.386.617.732.732l.774.258a.145.145 0 0 1 0 .274l-.774.258a1.156 1.156 0 0 0-.732.732l-.258.774a.145.145 0 0 1-.274 0l-.258-.774a1.156 1.156 0 0 0-.732-.732l-.774-.258a.145.145 0 0 1 0-.274l.774-.258c.346-.115.617-.386.732-.732L13.863.1z"/>
|
|
||||||
</symbol>
|
|
||||||
<symbol id="sun-fill" viewBox="0 0 16 16" fill="currentcolor">
|
|
||||||
<path d="M8 12a4 4 0 1 0 0-8 4 4 0 0 0 0 8zM8 0a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-1 0v-2A.5.5 0 0 1 8 0zm0 13a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-1 0v-2A.5.5 0 0 1 8 13zm8-5a.5.5 0 0 1-.5.5h-2a.5.5 0 0 1 0-1h2a.5.5 0 0 1 .5.5zM3 8a.5.5 0 0 1-.5.5h-2a.5.5 0 0 1 0-1h2A.5.5 0 0 1 3 8zm10.657-5.657a.5.5 0 0 1 0 .707l-1.414 1.415a.5.5 0 1 1-.707-.708l1.414-1.414a.5.5 0 0 1 .707 0zm-9.193 9.193a.5.5 0 0 1 0 .707L3.05 13.657a.5.5 0 0 1-.707-.707l1.414-1.414a.5.5 0 0 1 .707 0zm9.193 2.121a.5.5 0 0 1-.707 0l-1.414-1.414a.5.5 0 0 1 .707-.707l1.414 1.414a.5.5 0 0 1 0 .707zM4.464 4.465a.5.5 0 0 1-.707 0L2.343 3.05a.5.5 0 1 1 .707-.707l1.414 1.414a.5.5 0 0 1 0 .708z"/>
|
|
||||||
</symbol>
|
|
||||||
</svg>
|
|
||||||
<div class="dropdown position-fixed bottom-0 end-0 mb-3 me-3 bd-mode-toggle">
|
|
||||||
<button class="btn btn-primary py-2 dropdown-toggle d-flex align-items-center" id="bd-theme" type="button" aria-expanded="false" data-bs-toggle="dropdown" aria-label="Toggle theme (light)">
|
|
||||||
<svg class="bi my-1 theme-icon-active" width="1em" height="1em"><use href="#sun-fill"></use></svg>
|
|
||||||
<span class="visually-hidden" id="bd-theme-text">Toggle theme</span>
|
|
||||||
</button>
|
|
||||||
<ul class="dropdown-menu dropdown-menu-end shadow" aria-labelledby="bd-theme-text" style="">
|
|
||||||
<li>
|
|
||||||
<button type="button" class="dropdown-item d-flex align-items-center active" data-bs-theme-value="light" aria-pressed="true">
|
|
||||||
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em"><use href="#sun-fill"></use></svg>
|
|
||||||
Light
|
|
||||||
<svg class="bi ms-auto d-none" width="1em" height="1em"><use href="#check2"></use></svg>
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<button type="button" class="dropdown-item d-flex align-items-center" data-bs-theme-value="dark" aria-pressed="false">
|
|
||||||
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em"><use href="#moon-stars-fill"></use></svg>
|
|
||||||
Dark
|
|
||||||
<svg class="bi ms-auto d-none" width="1em" height="1em"><use href="#check2"></use></svg>
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
<li>
|
|
||||||
<button type="button" class="dropdown-item d-flex align-items-center" data-bs-theme-value="auto" aria-pressed="false">
|
|
||||||
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em"><use href="#circle-half"></use></svg>
|
|
||||||
Auto
|
|
||||||
<svg class="bi ms-auto d-none" width="1em" height="1em"><use href="#check2"></use></svg>
|
|
||||||
</button>
|
|
||||||
</li>
|
|
||||||
</ul>
|
|
||||||
</div>
|
|
||||||
</template>
|
</template>
|
||||||
|
@ -1,8 +1,7 @@
|
|||||||
import { createApp } from 'vue';
|
import { createApp } from 'vue';
|
||||||
import App from './App.vue';
|
import App from './App.vue';
|
||||||
import "./assets/styles.scss";
|
import "./assets/styles.scss";
|
||||||
import "../../node_modules/bootstrap-icons/font/bootstrap-icons.scss";
|
import "bootstrap-icons/font/bootstrap-icons.scss";
|
||||||
import "bootstrap";
|
import "bootstrap";
|
||||||
import "./color-modes";
|
|
||||||
|
|
||||||
createApp(App).mount('#app');
|
createApp(App).mount('#app');
|
||||||
|
@ -6,3 +6,4 @@ $link-decoration: none;
|
|||||||
$primary: #2c3e50;
|
$primary: #2c3e50;
|
||||||
$list-group-disabled-color: #adb5bd;
|
$list-group-disabled-color: #adb5bd;
|
||||||
$enable-negative-margins: true;
|
$enable-negative-margins: true;
|
||||||
|
$body-color-dark: #e7eaed;
|
||||||
|
@ -56,8 +56,17 @@
|
|||||||
z-index: 1500;
|
z-index: 1500;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message.read:not(.active):not(.selected) {
|
.message {
|
||||||
color: $gray-500;
|
&.read {
|
||||||
|
color: $text-muted;
|
||||||
|
|
||||||
|
b {
|
||||||
|
font-weight: normal;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&.selected {
|
||||||
|
background: var(--bs-primary-bg-subtle);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#nav-plain-text .text-view,
|
#nav-plain-text .text-view,
|
||||||
@ -180,20 +189,6 @@
|
|||||||
border-top: 0;
|
border-top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.message.selected {
|
|
||||||
background: $gray-300;
|
|
||||||
|
|
||||||
.text-muted {
|
|
||||||
color: $body-color !important;
|
|
||||||
}
|
|
||||||
|
|
||||||
&.read {
|
|
||||||
b {
|
|
||||||
font-weight: normal;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
body.blur {
|
body.blur {
|
||||||
.privacy {
|
.privacy {
|
||||||
filter: blur(3px);
|
filter: blur(3px);
|
||||||
@ -280,8 +275,8 @@ body.blur {
|
|||||||
https://prismjs.com/download.html#themes=prism-coy&languages=markup+css */
|
https://prismjs.com/download.html#themes=prism-coy&languages=markup+css */
|
||||||
code[class*="language-"],
|
code[class*="language-"],
|
||||||
pre[class*="language-"] {
|
pre[class*="language-"] {
|
||||||
color: #000;
|
// color: #000;
|
||||||
background: 0 0;
|
// background: 0 0;
|
||||||
font-size: 0.85em;
|
font-size: 0.85em;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
white-space: pre;
|
white-space: pre;
|
||||||
@ -314,7 +309,7 @@ code[class*="language-"] {
|
|||||||
}
|
}
|
||||||
:not(pre) > code[class*="language-"],
|
:not(pre) > code[class*="language-"],
|
||||||
pre[class*="language-"] {
|
pre[class*="language-"] {
|
||||||
background-color: #fdfdfd;
|
// background-color: #fdfdfd;
|
||||||
-webkit-box-sizing: border-box;
|
-webkit-box-sizing: border-box;
|
||||||
-moz-box-sizing: border-box;
|
-moz-box-sizing: border-box;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
@ -364,7 +359,7 @@ pre[class*="language-"] {
|
|||||||
.token.url,
|
.token.url,
|
||||||
.token.variable {
|
.token.variable {
|
||||||
color: #a67f59;
|
color: #a67f59;
|
||||||
background: rgba(255, 255, 255, 0.5);
|
// background: rgba(255, 255, 255, 0.5);
|
||||||
}
|
}
|
||||||
.token.atrule,
|
.token.atrule,
|
||||||
.token.attr-value,
|
.token.attr-value,
|
||||||
@ -379,7 +374,7 @@ pre[class*="language-"] {
|
|||||||
.language-css .token.string,
|
.language-css .token.string,
|
||||||
.style .token.string {
|
.style .token.string {
|
||||||
color: #a67f59;
|
color: #a67f59;
|
||||||
background: rgba(255, 255, 255, 0.5);
|
// background: rgba(255, 255, 255, 0.5);
|
||||||
}
|
}
|
||||||
.token.important {
|
.token.important {
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
@ -390,9 +385,9 @@ pre[class*="language-"] {
|
|||||||
.token.italic {
|
.token.italic {
|
||||||
font-style: italic;
|
font-style: italic;
|
||||||
}
|
}
|
||||||
.token.entity {
|
// .token.entity {
|
||||||
cursor: help;
|
// cursor: help;
|
||||||
}
|
// }
|
||||||
.token.namespace {
|
.token.namespace {
|
||||||
opacity: 0.7;
|
opacity: 0.7;
|
||||||
}
|
}
|
||||||
|
@ -1,94 +0,0 @@
|
|||||||
/*!
|
|
||||||
* Color mode toggler for Bootstrap's docs (https://getbootstrap.com/)
|
|
||||||
* Copyright 2011-2023 The Bootstrap Authors
|
|
||||||
* Licensed under the Creative Commons Attribution 3.0 Unported License.
|
|
||||||
*/
|
|
||||||
|
|
||||||
(() => {
|
|
||||||
'use strict';
|
|
||||||
|
|
||||||
const getStoredTheme = () => localStorage.getItem('theme');
|
|
||||||
const setStoredTheme = (theme) => localStorage.setItem('theme', theme);
|
|
||||||
|
|
||||||
const getPreferredTheme = () => {
|
|
||||||
const storedTheme = getStoredTheme();
|
|
||||||
if (storedTheme) {
|
|
||||||
return storedTheme;
|
|
||||||
}
|
|
||||||
|
|
||||||
return window.matchMedia('(prefers-color-scheme: dark)').matches
|
|
||||||
? 'dark'
|
|
||||||
: 'light';
|
|
||||||
};
|
|
||||||
|
|
||||||
const setTheme = (theme) => {
|
|
||||||
if (
|
|
||||||
theme === 'auto' &&
|
|
||||||
window.matchMedia('(prefers-color-scheme: dark)').matches
|
|
||||||
) {
|
|
||||||
document.documentElement.setAttribute('data-bs-theme', 'dark');
|
|
||||||
} else {
|
|
||||||
document.documentElement.setAttribute('data-bs-theme', theme);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
setTheme(getPreferredTheme());
|
|
||||||
|
|
||||||
const showActiveTheme = (theme, focus = false) => {
|
|
||||||
const themeSwitcher = document.querySelector('#bd-theme');
|
|
||||||
|
|
||||||
if (!themeSwitcher) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const themeSwitcherText = document.querySelector('#bd-theme-text');
|
|
||||||
const activeThemeIcon = document.querySelector('.theme-icon-active use');
|
|
||||||
const btnToActive = document.querySelector(
|
|
||||||
`[data-bs-theme-value="${theme}"]`
|
|
||||||
);
|
|
||||||
const svgOfActiveBtn = btnToActive
|
|
||||||
.querySelector('svg use')
|
|
||||||
.getAttribute('href');
|
|
||||||
|
|
||||||
document.querySelectorAll('[data-bs-theme-value]').forEach((element) => {
|
|
||||||
element.classList.remove('active');
|
|
||||||
element.setAttribute('aria-pressed', 'false');
|
|
||||||
});
|
|
||||||
|
|
||||||
btnToActive.classList.add('active');
|
|
||||||
btnToActive.setAttribute('aria-pressed', 'true');
|
|
||||||
activeThemeIcon.setAttribute('href', svgOfActiveBtn);
|
|
||||||
const themeSwitcherLabel = `${themeSwitcherText.textContent} (${btnToActive.dataset.bsThemeValue})`;
|
|
||||||
themeSwitcher.setAttribute('aria-label', themeSwitcherLabel);
|
|
||||||
|
|
||||||
if (focus) {
|
|
||||||
themeSwitcher.focus();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
window
|
|
||||||
.matchMedia('(prefers-color-scheme: dark)')
|
|
||||||
.addEventListener('change', () => {
|
|
||||||
const storedTheme = getStoredTheme();
|
|
||||||
if (storedTheme !== 'light' && storedTheme !== 'dark') {
|
|
||||||
setTheme(getPreferredTheme());
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
window.addEventListener('DOMContentLoaded', () => {
|
|
||||||
showActiveTheme(getPreferredTheme());
|
|
||||||
|
|
||||||
document.querySelectorAll('[data-bs-theme-value]').forEach((toggle) => {
|
|
||||||
toggle.addEventListener('click', () => {
|
|
||||||
const theme = toggle.getAttribute('data-bs-theme-value');
|
|
||||||
setStoredTheme(theme);
|
|
||||||
setTheme(theme);
|
|
||||||
showActiveTheme(theme, true);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
});
|
|
||||||
})();
|
|
||||||
|
|
||||||
document.querySelectorAll('[data-bs-toggle="popover"]').forEach((popover) => {
|
|
||||||
new bootstrap.Popover(popover);
|
|
||||||
});
|
|
@ -14,14 +14,18 @@ export default {
|
|||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="mt-4 border-top pt-4">
|
<div class="mt-4 border-top pt-4">
|
||||||
<a v-for="part in attachments" :href="'api/v1/message/'+message.ID+'/part/'+part.PartID" class="card attachment float-start me-3 mb-3" target="_blank" style="width: 180px">
|
<a v-for="part in attachments" :href="'api/v1/message/' + message.ID + '/part/' + part.PartID"
|
||||||
<img v-if="isImage(part)" :src="'api/v1/message/'+message.ID+'/part/'+part.PartID+'/thumb'" class="card-img-top" alt="">
|
class="card attachment float-start me-3 mb-3" target="_blank" style="width: 180px">
|
||||||
<img v-else src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAALQAAAB4AQMAAABhKUq+AAAAA1BMVEX///+nxBvIAAAAGUlEQVQYGe3BgQAAAADDoPtTT+EA1QAAgFsLQAAB12s2WgAAAABJRU5ErkJggg==" class="card-img-top" alt="">
|
<img v-if="isImage(part)" :src="'api/v1/message/' + message.ID + '/part/' + part.PartID + '/thumb'"
|
||||||
|
class="card-img-top" alt="">
|
||||||
|
<img v-else
|
||||||
|
src="data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAALQAAAB4AQMAAABhKUq+AAAAA1BMVEX///+nxBvIAAAAGUlEQVQYGe3BgQAAAADDoPtTT+EA1QAAgFsLQAAB12s2WgAAAABJRU5ErkJggg=="
|
||||||
|
class="card-img-top" alt="">
|
||||||
<div class="icon" v-if="!isImage(part)">
|
<div class="icon" v-if="!isImage(part)">
|
||||||
<i class="bi" :class="attachmentIcon(part)"></i>
|
<i class="bi" :class="attachmentIcon(part)"></i>
|
||||||
</div>
|
</div>
|
||||||
<div class="card-body border-0">
|
<div class="card-body border-0">
|
||||||
<p class="mb-1 text-muted">
|
<p class="mb-1">
|
||||||
<i class="bi me-1" :class="attachmentIcon(part)"></i>
|
<i class="bi me-1" :class="attachmentIcon(part)"></i>
|
||||||
<small>{{ getFileSize(part.Size) }}</small>
|
<small>{{ getFileSize(part.Size) }}</small>
|
||||||
</p>
|
</p>
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
|
|
||||||
<script>
|
<script>
|
||||||
import commonMixins from '../mixins.js';
|
import commonMixins from '../mixins.js'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
@ -17,9 +17,9 @@ export default {
|
|||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
let self = this;
|
let self = this;
|
||||||
let uri = 'api/v1/message/' + self.message.ID + '/headers';
|
let uri = 'api/v1/message/' + self.message.ID + '/headers'
|
||||||
self.get(uri, false, function (response) {
|
self.get(uri, false, function (response) {
|
||||||
self.headers = response.data;
|
self.headers = response.data
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -30,7 +30,7 @@ export default {
|
|||||||
<div v-if="headers" class="small">
|
<div v-if="headers" class="small">
|
||||||
<div v-for="vals, k in headers" class="row mb-2 pb-2 border-bottom w-100">
|
<div v-for="vals, k in headers" class="row mb-2 pb-2 border-bottom w-100">
|
||||||
<div class="col-md-4 col-lg-3 col-xl-2 mb-2"><b>{{ k }}</b></div>
|
<div class="col-md-4 col-lg-3 col-xl-2 mb-2"><b>{{ k }}</b></div>
|
||||||
<div class="col-md-8 col-lg-9 col-xl-10 text-muted">
|
<div class="col-md-8 col-lg-9 col-xl-10 text-body-secondary">
|
||||||
<div v-for="x in vals" class="mb-2 text-break">{{ x }}</div>
|
<div v-for="x in vals" class="mb-2 text-break">{{ x }}</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -142,15 +142,30 @@ export default {
|
|||||||
|
|
||||||
resizeIframes: function () {
|
resizeIframes: function () {
|
||||||
if (this.scaleHTMLPreview != 'display') {
|
if (this.scaleHTMLPreview != 'display') {
|
||||||
return;
|
return
|
||||||
}
|
}
|
||||||
let h = document.getElementById('preview-html');
|
let h = document.getElementById('preview-html')
|
||||||
if (h) {
|
if (h) {
|
||||||
h.style.height = h.contentWindow.document.body.scrollHeight + 50 + 'px';
|
h.style.height = h.contentWindow.document.body.scrollHeight + 50 + 'px'
|
||||||
}
|
}
|
||||||
|
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// set the iframe body & text colors based on current theme
|
||||||
|
initRawIframe: function (el) {
|
||||||
|
let bodyStyles = window.getComputedStyle(document.body, null)
|
||||||
|
let bg = bodyStyles.getPropertyValue('background-color')
|
||||||
|
let txt = bodyStyles.getPropertyValue('color')
|
||||||
|
|
||||||
|
let body = el.target.contentWindow.document.querySelector('body')
|
||||||
|
if (body) {
|
||||||
|
body.style.color = txt
|
||||||
|
body.style.backgroundColor = bg
|
||||||
|
}
|
||||||
|
|
||||||
|
this.resizeIframe(el)
|
||||||
|
},
|
||||||
|
|
||||||
saveTags: function () {
|
saveTags: function () {
|
||||||
let self = this;
|
let self = this;
|
||||||
|
|
||||||
@ -195,7 +210,7 @@ export default {
|
|||||||
</script>
|
</script>
|
||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div v-if="message" id="message-view" class="mh-100" style="overflow-y: scroll;">
|
<div v-if="message" id="message-view" class="px-2 px-md-0 mh-100" style="overflow-y: scroll;">
|
||||||
<div class="row w-100">
|
<div class="row w-100">
|
||||||
<div class="col-md">
|
<div class="col-md">
|
||||||
<table class="messageHeaders">
|
<table class="messageHeaders">
|
||||||
@ -221,7 +236,7 @@ export default {
|
|||||||
<template v-if="i > 0">, </template>
|
<template v-if="i > 0">, </template>
|
||||||
<span class="text-nowrap">{{ t.Name + " <" + t.Address + ">" }}</span>
|
<span class="text-nowrap">{{ t.Name + " <" + t.Address + ">" }}</span>
|
||||||
</span>
|
</span>
|
||||||
<span v-else class="text-muted">[Undisclosed recipients]</span>
|
<span v-else class="text-body-secondary">[Undisclosed recipients]</span>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr v-if="message.Cc && message.Cc.length" class="small">
|
<tr v-if="message.Cc && message.Cc.length" class="small">
|
||||||
@ -243,7 +258,7 @@ export default {
|
|||||||
</tr>
|
</tr>
|
||||||
<tr v-if="message.ReplyTo && message.ReplyTo.length" class="small">
|
<tr v-if="message.ReplyTo && message.ReplyTo.length" class="small">
|
||||||
<th class="text-nowrap">Reply-To</th>
|
<th class="text-nowrap">Reply-To</th>
|
||||||
<td class="privacy text-muted">
|
<td class="privacy text-body-secondary">
|
||||||
<span v-for="(t, i) in message.ReplyTo">
|
<span v-for="(t, i) in message.ReplyTo">
|
||||||
<template v-if="i > 0">,</template>
|
<template v-if="i > 0">,</template>
|
||||||
{{ t.Name + " <" + t.Address + ">" }} </span>
|
{{ t.Name + " <" + t.Address + ">" }} </span>
|
||||||
@ -251,13 +266,13 @@ export default {
|
|||||||
</tr>
|
</tr>
|
||||||
<tr v-if="message.ReturnPath && message.ReturnPath != message.From.Address" class="small">
|
<tr v-if="message.ReturnPath && message.ReturnPath != message.From.Address" class="small">
|
||||||
<th class="text-nowrap">Return-Path</th>
|
<th class="text-nowrap">Return-Path</th>
|
||||||
<td class="privacy text-muted"><{{ message.ReturnPath }}></td>
|
<td class="privacy text-body-secondary"><{{ message.ReturnPath }}></td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<th class="small">Subject</th>
|
<th class="small">Subject</th>
|
||||||
<td>
|
<td>
|
||||||
<strong v-if="message.Subject != ''">{{ message.Subject }}</strong>
|
<strong v-if="message.Subject != ''">{{ message.Subject }}</strong>
|
||||||
<small class="text-muted" v-else>[ no subject ]</small>
|
<small class="text-body-secondary" v-else>[ no subject ]</small>
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr class="d-md-none small">
|
<tr class="d-md-none small">
|
||||||
@ -317,8 +332,7 @@ export default {
|
|||||||
|
|
||||||
<div class="d-none d-lg-block ms-auto me-2" v-if="showMobileBtns">
|
<div class="d-none d-lg-block ms-auto me-2" v-if="showMobileBtns">
|
||||||
<template v-for=" vals, key in responsiveSizes ">
|
<template v-for=" vals, key in responsiveSizes ">
|
||||||
<button class="btn" :class="scaleHTMLPreview == key ? 'btn-outline-primary' : ''"
|
<button class="btn" :disabled="scaleHTMLPreview == key" :title="'Switch to ' + key + ' view'"
|
||||||
:disabled="scaleHTMLPreview == key" :title="'Switch to ' + key + ' view'"
|
|
||||||
v-on:click=" scaleHTMLPreview = key">
|
v-on:click=" scaleHTMLPreview = key">
|
||||||
<i class="bi" :class="'bi-' + key"></i>
|
<i class="bi" :class="'bi-' + key"></i>
|
||||||
</button>
|
</button>
|
||||||
@ -352,8 +366,8 @@ export default {
|
|||||||
<Headers v-if="loadHeaders" :message="message"></Headers>
|
<Headers v-if="loadHeaders" :message="message"></Headers>
|
||||||
</div>
|
</div>
|
||||||
<div class="tab-pane fade" id="nav-raw" role="tabpanel" aria-labelledby="nav-raw-tab" tabindex="0">
|
<div class="tab-pane fade" id="nav-raw" role="tabpanel" aria-labelledby="nav-raw-tab" tabindex="0">
|
||||||
<iframe v-if="srcURI" :src="srcURI" v-on:load="resizeIframe" frameborder="0"
|
<iframe v-if="srcURI" :src="srcURI" v-on:load="initRawIframe" frameborder="0"
|
||||||
style="width: 100%; height: 300px; background: #fff; color: #15141A"></iframe>
|
style="width: 100%; height: 300px"></iframe>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
@ -57,19 +57,21 @@ export default {
|
|||||||
<div class="modal-body">
|
<div class="modal-body">
|
||||||
<h6>Send this message to one or more addresses specified below.</h6>
|
<h6>Send this message to one or more addresses specified below.</h6>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<label class="col-sm-2 col-form-label text-muted">From</label>
|
<label class="col-sm-2 col-form-label text-body-secondary">From</label>
|
||||||
<div class="col-sm-10">
|
<div class="col-sm-10">
|
||||||
<input type="text" readonly class="form-control-plaintext" :value="message.From.Address">
|
<input type="text" aria-label="From address" readonly class="form-control-plaintext"
|
||||||
|
:value="message.From.Address">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row">
|
<div class="row">
|
||||||
<label class=" col-sm-2 col-form-label text-muted">Subject</label>
|
<label class=" col-sm-2 col-form-label text-body-secondary">Subject</label>
|
||||||
<div class="col-sm-10">
|
<div class="col-sm-10">
|
||||||
<input type="text" readonly class="form-control-plaintext" :value="message.Subject">
|
<input type="text" aria-label="Subject" readonly class="form-control-plaintext"
|
||||||
|
:value="message.Subject">
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="row mb-3">
|
<div class="row mb-3">
|
||||||
<label class="col-sm-2 col-form-label text-muted">Send to</label>
|
<label class="col-sm-2 col-form-label text-body-secondary">Send to</label>
|
||||||
<div class="col-sm-10">
|
<div class="col-sm-10">
|
||||||
<select class="form-select tag-selector" v-model="addresses" multiple data-allow-new="true"
|
<select class="form-select tag-selector" v-model="addresses" multiple data-allow-new="true"
|
||||||
data-clear-end="true" data-allow-clear="true" data-placeholder="Enter email addresses..."
|
data-clear-end="true" data-allow-clear="true" data-placeholder="Enter email addresses..."
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import commonMixins from '../mixins.js';
|
import commonMixins from '../mixins.js'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
@ -12,7 +12,7 @@ export default {
|
|||||||
|
|
||||||
<template>
|
<template>
|
||||||
<div class="card mt-4">
|
<div class="card mt-4">
|
||||||
<div class="card-body text-muted small">
|
<div class="card-body text-body-secondary small">
|
||||||
<p class="card-text">
|
<p class="card-text">
|
||||||
<b>Message date:</b><br>
|
<b>Message date:</b><br>
|
||||||
<small>{{ messageDate(message.Date) }}</small>
|
<small>{{ messageDate(message.Date) }}</small>
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
<script>
|
<script>
|
||||||
import { Toast } from 'bootstrap';
|
import { Toast } from 'bootstrap'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
props: {
|
props: {
|
||||||
@ -7,15 +7,15 @@ export default {
|
|||||||
},
|
},
|
||||||
|
|
||||||
mounted() {
|
mounted() {
|
||||||
let self = this;
|
let self = this
|
||||||
let el = document.getElementById('messageToast');
|
let el = document.getElementById('messageToast')
|
||||||
if (el) {
|
if (el) {
|
||||||
el.addEventListener('hidden.bs.toast', () => {
|
el.addEventListener('hidden.bs.toast', () => {
|
||||||
self.$emit("clearMessageToast");
|
self.$emit("clearMessageToast")
|
||||||
})
|
})
|
||||||
|
|
||||||
let b = Toast.getOrCreateInstance(el);
|
let b = Toast.getOrCreateInstance(el)
|
||||||
b.show();
|
b.show()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -33,7 +33,7 @@ export default {
|
|||||||
|
|
||||||
<div class="toast-body">
|
<div class="toast-body">
|
||||||
<div>
|
<div>
|
||||||
<a :href="'#' + message.ID" class="d-block text-truncate text-muted">
|
<a :href="'#' + message.ID" class="d-block text-truncate text-body-secondary">
|
||||||
<template v-if="message.Subject != ''">{{ message.Subject }}</template>
|
<template v-if="message.Subject != ''">{{ message.Subject }}</template>
|
||||||
<template v-else>[ no subject ]</template>
|
<template v-else>[ no subject ]</template>
|
||||||
</a>
|
</a>
|
||||||
|
123
server/ui-src/templates/ThemeToggle.vue
Normal file
123
server/ui-src/templates/ThemeToggle.vue
Normal file
@ -0,0 +1,123 @@
|
|||||||
|
<script>
|
||||||
|
export default {
|
||||||
|
data() {
|
||||||
|
return {
|
||||||
|
theme: 'auto',
|
||||||
|
icon: '#circle-half',
|
||||||
|
icons: {
|
||||||
|
'auto': '#circle-half',
|
||||||
|
'light': '#sun-fill',
|
||||||
|
'dark': '#moon-stars-fill'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
mounted() {
|
||||||
|
this.setTheme(this.getPreferredTheme())
|
||||||
|
},
|
||||||
|
|
||||||
|
methods: {
|
||||||
|
getStoredTheme: function () {
|
||||||
|
let theme = localStorage.getItem('theme')
|
||||||
|
if (!theme) {
|
||||||
|
theme = 'auto'
|
||||||
|
}
|
||||||
|
|
||||||
|
return theme
|
||||||
|
},
|
||||||
|
|
||||||
|
setStoredTheme: function (theme) {
|
||||||
|
localStorage.setItem('theme', theme)
|
||||||
|
this.setTheme(theme)
|
||||||
|
},
|
||||||
|
|
||||||
|
getPreferredTheme: function () {
|
||||||
|
const storedTheme = this.getStoredTheme()
|
||||||
|
if (storedTheme) {
|
||||||
|
return storedTheme
|
||||||
|
}
|
||||||
|
|
||||||
|
return window.matchMedia('(prefers-color-scheme: dark)').matches
|
||||||
|
? 'dark'
|
||||||
|
: 'light'
|
||||||
|
},
|
||||||
|
|
||||||
|
setTheme: function (theme) {
|
||||||
|
this.icon = this.icons[theme]
|
||||||
|
this.theme = theme
|
||||||
|
if (
|
||||||
|
theme === 'auto' &&
|
||||||
|
window.matchMedia('(prefers-color-scheme: dark)').matches
|
||||||
|
) {
|
||||||
|
document.documentElement.setAttribute('data-bs-theme', 'dark')
|
||||||
|
} else {
|
||||||
|
document.documentElement.setAttribute('data-bs-theme', theme)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<svg xmlns="http://www.w3.org/2000/svg" style="display: none;">
|
||||||
|
<symbol id="bootstrap" viewBox="0 0 512 408" fill="currentcolor">
|
||||||
|
<path
|
||||||
|
d="M106.342 0c-29.214 0-50.827 25.58-49.86 53.32.927 26.647-.278 61.165-8.966 89.31C38.802 170.862 24.07 188.707 0 191v26c24.069 2.293 38.802 20.138 47.516 48.37 8.688 28.145 9.893 62.663 8.965 89.311C55.515 382.42 77.128 408 106.342 408h299.353c29.214 0 50.827-25.58 49.861-53.319-.928-26.648.277-61.166 8.964-89.311 8.715-28.232 23.411-46.077 47.48-48.37v-26c-24.069-2.293-38.765-20.138-47.48-48.37-8.687-28.145-9.892-62.663-8.964-89.31C456.522 25.58 434.909 0 405.695 0H106.342zm236.559 251.102c0 38.197-28.501 61.355-75.798 61.355h-87.202a2 2 0 01-2-2v-213a2 2 0 012-2h86.74c39.439 0 65.322 21.354 65.322 54.138 0 23.008-17.409 43.61-39.594 47.219v1.203c30.196 3.309 50.532 24.212 50.532 53.085zm-84.58-128.125h-45.91v64.814h38.669c29.888 0 46.373-12.03 46.373-33.535 0-20.151-14.174-31.279-39.132-31.279zm-45.91 90.53v71.431h47.605c31.12 0 47.605-12.482 47.605-35.941 0-23.46-16.947-35.49-49.608-35.49h-45.602z" />
|
||||||
|
</symbol>
|
||||||
|
<symbol id="check2" viewBox="0 0 16 16" fill="currentcolor">
|
||||||
|
<path
|
||||||
|
d="M13.854 3.646a.5.5 0 0 1 0 .708l-7 7a.5.5 0 0 1-.708 0l-3.5-3.5a.5.5 0 1 1 .708-.708L6.5 10.293l6.646-6.647a.5.5 0 0 1 .708 0z" />
|
||||||
|
</symbol>
|
||||||
|
<symbol id="circle-half" viewBox="0 0 16 16" fill="currentcolor">
|
||||||
|
<path d="M8 15A7 7 0 1 0 8 1v14zm0 1A8 8 0 1 1 8 0a8 8 0 0 1 0 16z" />
|
||||||
|
</symbol>
|
||||||
|
<symbol id="moon-stars-fill" viewBox="0 0 16 16" fill="currentcolor">
|
||||||
|
<path
|
||||||
|
d="M6 .278a.768.768 0 0 1 .08.858 7.208 7.208 0 0 0-.878 3.46c0 4.021 3.278 7.277 7.318 7.277.527 0 1.04-.055 1.533-.16a.787.787 0 0 1 .81.316.733.733 0 0 1-.031.893A8.349 8.349 0 0 1 8.344 16C3.734 16 0 12.286 0 7.71 0 4.266 2.114 1.312 5.124.06A.752.752 0 0 1 6 .278z" />
|
||||||
|
<path
|
||||||
|
d="M10.794 3.148a.217.217 0 0 1 .412 0l.387 1.162c.173.518.579.924 1.097 1.097l1.162.387a.217.217 0 0 1 0 .412l-1.162.387a1.734 1.734 0 0 0-1.097 1.097l-.387 1.162a.217.217 0 0 1-.412 0l-.387-1.162A1.734 1.734 0 0 0 9.31 6.593l-1.162-.387a.217.217 0 0 1 0-.412l1.162-.387a1.734 1.734 0 0 0 1.097-1.097l.387-1.162zM13.863.099a.145.145 0 0 1 .274 0l.258.774c.115.346.386.617.732.732l.774.258a.145.145 0 0 1 0 .274l-.774.258a1.156 1.156 0 0 0-.732.732l-.258.774a.145.145 0 0 1-.274 0l-.258-.774a1.156 1.156 0 0 0-.732-.732l-.774-.258a.145.145 0 0 1 0-.274l.774-.258c.346-.115.617-.386.732-.732L13.863.1z" />
|
||||||
|
</symbol>
|
||||||
|
<symbol id="sun-fill" viewBox="0 0 16 16" fill="currentcolor">
|
||||||
|
<path
|
||||||
|
d="M8 12a4 4 0 1 0 0-8 4 4 0 0 0 0 8zM8 0a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-1 0v-2A.5.5 0 0 1 8 0zm0 13a.5.5 0 0 1 .5.5v2a.5.5 0 0 1-1 0v-2A.5.5 0 0 1 8 13zm8-5a.5.5 0 0 1-.5.5h-2a.5.5 0 0 1 0-1h2a.5.5 0 0 1 .5.5zM3 8a.5.5 0 0 1-.5.5h-2a.5.5 0 0 1 0-1h2A.5.5 0 0 1 3 8zm10.657-5.657a.5.5 0 0 1 0 .707l-1.414 1.415a.5.5 0 1 1-.707-.708l1.414-1.414a.5.5 0 0 1 .707 0zm-9.193 9.193a.5.5 0 0 1 0 .707L3.05 13.657a.5.5 0 0 1-.707-.707l1.414-1.414a.5.5 0 0 1 .707 0zm9.193 2.121a.5.5 0 0 1-.707 0l-1.414-1.414a.5.5 0 0 1 .707-.707l1.414 1.414a.5.5 0 0 1 0 .707zM4.464 4.465a.5.5 0 0 1-.707 0L2.343 3.05a.5.5 0 1 1 .707-.707l1.414 1.414a.5.5 0 0 1 0 .708z" />
|
||||||
|
</symbol>
|
||||||
|
</svg>
|
||||||
|
<div class="dropdown bd-mode-toggle float-end me-2 d-inline-block">
|
||||||
|
<button class="btn btn-sm btn-outline-secondary dropdown-toggle" type="button" aria-expanded="false"
|
||||||
|
title="Toggle theme" data-bs-toggle="dropdown" aria-label="Toggle theme">
|
||||||
|
<svg class="bi my-1 theme-icon-active" width="1em" height="1em">
|
||||||
|
<use :href="icon"></use>
|
||||||
|
</svg>
|
||||||
|
<span class="visually-hidden" id="bd-theme-text">Toggle theme</span>
|
||||||
|
</button>
|
||||||
|
<ul class="dropdown-menu dropdown-menu-end shadow" aria-labelledby="bd-theme-text">
|
||||||
|
<li>
|
||||||
|
<button type="button" class="dropdown-item d-flex align-items-center"
|
||||||
|
:class="theme == 'light' ? 'active' : ''" @click="setStoredTheme('light')">
|
||||||
|
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em">
|
||||||
|
<use href="#sun-fill"></use>
|
||||||
|
</svg>
|
||||||
|
Light
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button type="button" class="dropdown-item d-flex align-items-center"
|
||||||
|
:class="theme == 'dark' ? 'active' : ''" @click="setStoredTheme('dark')">
|
||||||
|
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em">
|
||||||
|
<use href="#moon-stars-fill"></use>
|
||||||
|
</svg>
|
||||||
|
Dark
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<button type="button" class="dropdown-item d-flex align-items-center"
|
||||||
|
:class="theme == 'auto' ? 'active' : ''" @click="setStoredTheme('auto')">
|
||||||
|
<svg class="bi me-2 opacity-50 theme-icon" width="1em" height="1em">
|
||||||
|
<use href="#circle-half"></use>
|
||||||
|
</svg>
|
||||||
|
Auto
|
||||||
|
</button>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</template>
|
Loading…
x
Reference in New Issue
Block a user