1
0
mirror of https://github.com/axllent/mailpit.git synced 2025-02-09 13:38:37 +02:00

Merge branch 'feature/dark-mode' into develop

This commit is contained in:
Ralph Slooten 2023-06-30 22:44:06 +12:00
commit a5de4e4f65
14 changed files with 515 additions and 512 deletions

View File

@ -14,6 +14,10 @@ const ctx = await esbuild.context(
bundle: true,
minify: doMinify,
sourcemap: false,
define: {
'__VUE_OPTIONS_API__': 'true',
'__VUE_PROD_DEVTOOLS__': 'false',
},
outdir: "server/ui/dist/",
plugins: [pluginVue(), sassPlugin()],
loader: {

View File

@ -304,7 +304,7 @@ func DownloadRaw(w http.ResponseWriter, r *http.Request) {
return
}
w.Header().Set("Content-Type", "text/plain")
w.Header().Set("Content-Type", "text/plain; charset=utf-8")
if dl == "1" {
w.Header().Set("Content-Disposition", "attachment; filename=\""+id+".eml\"")
}
@ -495,7 +495,7 @@ func SetTags(w http.ResponseWriter, r *http.Request) {
_, _ = w.Write([]byte("ok"))
}
// ReleaseMessage (method: POST) will release a message via a preconfigured external SMTP server.
// ReleaseMessage (method: POST) will release a message via a pre-configured external SMTP server.
// If no IDs are provided then all messages are updated.
func ReleaseMessage(w http.ResponseWriter, r *http.Request) {
// swagger:route POST /api/v1/message/{ID}/release message Release

View File

@ -4,6 +4,7 @@ import Message from './templates/Message.vue'
import MessageSummary from './templates/MessageSummary.vue'
import MessageRelease from './templates/MessageRelease.vue'
import MessageToast from './templates/MessageToast.vue'
import ThemeToggle from './templates/ThemeToggle.vue'
import moment from 'moment'
import Tinycon from 'tinycon'
@ -14,7 +15,8 @@ export default {
Message,
MessageSummary,
MessageRelease,
MessageToast
MessageToast,
ThemeToggle,
},
data() {
@ -50,55 +52,55 @@ export default {
watch: {
currentPath(v, old) {
if (v && v.match(/^[a-z0-9]+-[a-z0-9]+-[a-z0-9]+-[a-z0-9]+-[a-z0-9]+$/)) {
this.openMessage();
this.openMessage()
} else {
this.message = false;
this.message = false
}
},
unread(v, old) {
if (v == this.tcStatus) {
return;
return
}
this.tcStatus = v;
this.tcStatus = v
if (v == 0) {
Tinycon.reset();
Tinycon.reset()
} else {
Tinycon.setBubble(v);
Tinycon.setBubble(v)
}
}
},
computed: {
canPrev: function () {
return this.start > 0;
return this.start > 0
},
canNext: function () {
return this.total > (this.start + this.count);
return this.total > (this.start + this.count)
},
unreadInSearch: function () {
if (!this.searching) {
return false;
return false
}
return this.items.filter(i => !i.Read).length;
return this.items.filter(i => !i.Read).length
}
},
mounted() {
this.currentPath = window.location.hash.slice(1);
this.currentPath = window.location.hash.slice(1)
window.addEventListener('hashchange', () => {
this.currentPath = window.location.hash.slice(1);
});
this.currentPath = window.location.hash.slice(1)
})
this.notificationsSupported = window.isSecureContext
&& ("Notification" in window && Notification.permission !== "denied");
this.notificationsEnabled = this.notificationsSupported && Notification.permission == "granted";
&& ("Notification" in window && Notification.permission !== "denied")
this.notificationsEnabled = this.notificationsSupported && Notification.permission == "granted"
Tinycon.setOptions({
height: 11,
background: '#dd0000',
fallback: false
});
})
moment.updateLocale('en', {
relativeTime: {
@ -119,11 +121,11 @@ export default {
y: "a year",
yy: "%d years"
}
});
})
this.connect();
this.getUISettings();
this.loadMessages();
this.connect()
this.getUISettings()
this.loadMessages()
},
methods: {
@ -131,143 +133,143 @@ export default {
let now = Date.now()
// prevent double loading when UI loads & websocket connects
if (this.lastLoaded && now - this.lastLoaded < 250) {
return;
return
}
if (this.start == 0) {
this.lastLoaded = now;
this.lastLoaded = now
}
let self = this;
let params = {};
self.selected = [];
let self = this
let params = {}
self.selected = []
let uri = 'api/v1/messages';
let uri = 'api/v1/messages'
if (self.search) {
self.searching = true;
self.items = [];
self.searching = true
self.items = []
uri = 'api/v1/search'
self.start = 0; // search is displayed on one page
params['query'] = self.search;
params['limit'] = 200;
self.start = 0 // search is displayed on one page
params['query'] = self.search
params['limit'] = 200
} else {
self.searching = false;
params['limit'] = self.limit;
self.searching = false
params['limit'] = self.limit
if (self.start > 0) {
params['start'] = self.start;
params['start'] = self.start
}
}
self.get(uri, params, function (response) {
self.total = response.data.total;
self.unread = response.data.unread;
self.count = response.data.count;
self.start = response.data.start;
self.items = response.data.messages;
self.tags = response.data.tags;
self.existingTags = JSON.parse(JSON.stringify(self.tags));
self.total = response.data.total
self.unread = response.data.unread
self.count = response.data.count
self.start = response.data.start
self.items = response.data.messages
self.tags = response.data.tags
self.existingTags = JSON.parse(JSON.stringify(self.tags))
// if pagination > 0 && results == 0 reload first page (prune)
if (response.data.count == 0 && response.data.start > 0) {
self.start = 0;
return self.loadMessages();
self.start = 0
return self.loadMessages()
}
if (!self.scrollInPlace) {
let mp = document.getElementById('message-page');
let mp = document.getElementById('message-page')
if (mp) {
mp.scrollTop = 0;
mp.scrollTop = 0
}
}
self.scrollInPlace = false;
});
self.scrollInPlace = false
})
},
getUISettings: function () {
let self = this;
let self = this
self.get('api/v1/webui', null, function (response) {
self.relayConfig = response.data;
});
self.relayConfig = response.data
})
},
doSearch: function (e) {
e.preventDefault();
this.loadMessages();
e.preventDefault()
this.loadMessages()
},
tagSearch: function (e, tag) {
e.preventDefault();
e.preventDefault()
if (tag.match(/ /)) {
tag = '"' + tag + '"';
tag = '"' + tag + '"'
}
this.search = 'tag:' + tag;
window.location.hash = "";
this.loadMessages();
this.search = 'tag:' + tag
window.location.hash = ""
this.loadMessages()
},
resetSearch: function (e) {
e.preventDefault();
this.search = '';
this.scrollInPlace = true;
this.loadMessages();
e.preventDefault()
this.search = ''
this.scrollInPlace = true
this.loadMessages()
},
reloadMessages: function () {
this.search = "";
this.start = 0;
this.loadMessages();
this.search = ""
this.start = 0
this.loadMessages()
},
viewNext: function () {
this.start = parseInt(this.start, 10) + parseInt(this.limit, 10);
this.loadMessages();
this.start = parseInt(this.start, 10) + parseInt(this.limit, 10)
this.loadMessages()
},
viewPrev: function () {
let s = this.start - this.limit;
let s = this.start - this.limit
if (s < 0) {
s = 0;
s = 0
}
this.start = s;
this.loadMessages();
this.start = s
this.loadMessages()
},
openMessage: function (id) {
let self = this;
self.selected = [];
self.releaseAddresses = false;
self.toastMessage = false;
self.existingTags = JSON.parse(JSON.stringify(self.tags));
let self = this
self.selected = []
self.releaseAddresses = false
self.toastMessage = false
self.existingTags = JSON.parse(JSON.stringify(self.tags))
let uri = 'api/v1/message/' + self.currentPath
self.get(uri, false, function (response) {
for (let i in self.items) {
if (self.items[i].ID == self.currentPath) {
if (!self.items[i].Read) {
self.items[i].Read = true;
self.unread--;
self.items[i].Read = true
self.unread--
}
}
}
let d = response.data;
let d = response.data
// replace inline images embedded as inline attachments
if (d.HTML && d.Inline) {
for (let i in d.Inline) {
let a = d.Inline[i];
let a = d.Inline[i]
if (a.ContentID != '') {
d.HTML = d.HTML.replace(
new RegExp('(=["\']?)(cid:' + a.ContentID + ')(["|\'|\\s|\\/|>|;])', 'g'),
'$1' + window.location.origin + window.location.pathname + 'api/v1/message/' + d.ID + '/part/' + a.PartID + '$3'
);
)
}
if (a.FileName.match(/^[a-zA-Z0-9\_\-\.]+$/)) {
// some old email clients use the filename
d.HTML = d.HTML.replace(
new RegExp('(=["\']?)(' + a.FileName + ')(["|\'|\\s|\\/|>|;])', 'g'),
'$1' + window.location.origin + window.location.pathname + 'api/v1/message/' + d.ID + '/part/' + a.PartID + '$3'
);
)
}
}
}
@ -275,381 +277,381 @@ export default {
// replace inline images embedded as regular attachments
if (d.HTML && d.Attachments) {
for (let i in d.Attachments) {
let a = d.Attachments[i];
let a = d.Attachments[i]
if (a.ContentID != '') {
d.HTML = d.HTML.replace(
new RegExp('(=["\']?)(cid:' + a.ContentID + ')(["|\'|\\s|\\/|>|;])', 'g'),
'$1' + window.location.origin + window.location.pathname + 'api/v1/message/' + d.ID + '/part/' + a.PartID + '$3'
);
)
}
if (a.FileName.match(/^[a-zA-Z0-9\_\-\.]+$/)) {
// some old email clients use the filename
d.HTML = d.HTML.replace(
new RegExp('(=["\']?)(' + a.FileName + ')(["|\'|\\s|\\/|>|;])', 'g'),
'$1' + window.location.origin + window.location.pathname + 'api/v1/message/' + d.ID + '/part/' + a.PartID + '$3'
);
)
}
}
}
self.message = d;
self.message = d
// generate the prev/next links based on current message list
self.messagePrev = false;
self.messageNext = false;
let found = false;
self.messagePrev = false
self.messageNext = false
let found = false
for (let i in self.items) {
if (self.items[i].ID == self.message.ID) {
found = true;
found = true
} else if (found && !self.messageNext) {
self.messageNext = self.items[i].ID;
break;
self.messageNext = self.items[i].ID
break
} else {
self.messagePrev = self.items[i].ID;
self.messagePrev = self.items[i].ID
}
}
});
})
},
// universal handler to delete current or selected messages
deleteMessages: function () {
let ids = [];
let self = this;
let ids = []
let self = this
if (self.message) {
ids.push(self.message.ID);
ids.push(self.message.ID)
} else {
ids = JSON.parse(JSON.stringify(self.selected));
ids = JSON.parse(JSON.stringify(self.selected))
}
if (!ids.length) {
return false;
return false
}
let uri = 'api/v1/messages';
let uri = 'api/v1/messages'
self.delete(uri, { 'ids': ids }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// delete messages displayed in current search
deleteSearch: function () {
let ids = this.items.map(item => item.ID);
let ids = this.items.map(item => item.ID)
if (!ids.length) {
return false;
return false
}
let self = this;
let uri = 'api/v1/messages';
let self = this
let uri = 'api/v1/messages'
self.delete(uri, { 'ids': ids }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// delete all messages from mailbox
deleteAll: function () {
let self = this;
let uri = 'api/v1/messages';
let self = this
let uri = 'api/v1/messages'
self.delete(uri, false, function (response) {
window.location.hash = "";
self.reloadMessages();
});
window.location.hash = ""
self.reloadMessages()
})
},
// mark current message as read
markUnread: function () {
let self = this;
let self = this
if (!self.message) {
return false;
return false
}
let uri = 'api/v1/messages';
let uri = 'api/v1/messages'
self.put(uri, { 'read': false, 'ids': [self.message.ID] }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// mark all messages in mailbox as read
markAllRead: function () {
let self = this;
let self = this
let uri = 'api/v1/messages'
self.put(uri, { 'read': true }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// mark messages in current search as read
markSearchRead: function () {
let ids = this.items.map(item => item.ID);
let ids = this.items.map(item => item.ID)
if (!ids.length) {
return false;
return false
}
let self = this;
let uri = 'api/v1/messages';
let self = this
let uri = 'api/v1/messages'
self.put(uri, { 'read': true, 'ids': ids }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// mark selected messages as read
markSelectedRead: function () {
let self = this;
let self = this
if (!self.selected.length) {
return false;
return false
}
let uri = 'api/v1/messages';
let uri = 'api/v1/messages'
self.put(uri, { 'read': true, 'ids': self.selected }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// mark selected messages as unread
markSelectedUnread: function () {
let self = this;
let self = this
if (!self.selected.length) {
return false;
return false
}
let uri = 'api/v1/messages';
let uri = 'api/v1/messages'
self.put(uri, { 'read': false, 'ids': self.selected }, function (response) {
window.location.hash = "";
self.scrollInPlace = true;
self.loadMessages();
});
window.location.hash = ""
self.scrollInPlace = true
self.loadMessages()
})
},
// test if any selected emails are unread
selectedHasUnread: function () {
if (!this.selected.length) {
return false;
return false
}
for (let i in this.items) {
if (this.isSelected(this.items[i].ID) && !this.items[i].Read) {
return true;
return true
}
}
return false;
return false
},
// test of any selected emails are read
selectedHasRead: function () {
if (!this.selected.length) {
return false;
return false
}
for (let i in this.items) {
if (this.isSelected(this.items[i].ID) && this.items[i].Read) {
return true;
return true
}
}
return false;
return false
},
// websocket connect
connect: function () {
let wsproto = location.protocol == 'https:' ? 'wss' : 'ws';
let wsproto = location.protocol == 'https:' ? 'wss' : 'ws'
let ws = new WebSocket(
wsproto + "://" + document.location.host + document.location.pathname + "api/events"
);
let self = this;
)
let self = this
ws.onmessage = function (e) {
let response = JSON.parse(e.data);
let response = JSON.parse(e.data)
if (!response) {
return;
return
}
// new messages
if (response.Type == "new" && response.Data) {
if (!self.searching) {
if (self.start < 1) {
// first page
self.items.unshift(response.Data);
self.items.unshift(response.Data)
if (self.items.length > self.limit) {
self.items.pop();
self.items.pop()
}
// first message was open, set messagePrev
if (!self.messagePrev) {
self.messagePrev = response.Data.ID;
self.messagePrev = response.Data.ID
}
} else {
self.start++;
self.start++
}
}
self.total++;
self.unread++;
self.total++
self.unread++
for (let i in response.Data.Tags) {
if (self.tags.indexOf(response.Data.Tags[i]) < 0) {
self.tags.push(response.Data.Tags[i]);
self.tags.sort();
self.tags.push(response.Data.Tags[i])
self.tags.sort()
}
}
let from = response.Data.From != null ? response.Data.From.Address : '[unknown]';
self.browserNotify("New mail from: " + from, response.Data.Subject);
self.setMessageToast(response.Data);
let from = response.Data.From != null ? response.Data.From.Address : '[unknown]'
self.browserNotify("New mail from: " + from, response.Data.Subject)
self.setMessageToast(response.Data)
} else if (response.Type == "prune") {
// messages have been deleted, reload messages to adjust
self.scrollInPlace = true;
self.loadMessages();
self.scrollInPlace = true
self.loadMessages()
}
}
ws.onopen = function () {
self.isConnected = true;
self.loadMessages();
self.isConnected = true
self.loadMessages()
}
ws.onclose = function (e) {
self.isConnected = false;
self.isConnected = false
setTimeout(function () {
self.connect(); // reconnect
}, 1000);
self.connect() // reconnect
}, 1000)
}
ws.onerror = function (err) {
ws.close();
ws.close()
}
},
getPrimaryEmailTo: function (message) {
for (let i in message.To) {
return message.To[i].Address;
return message.To[i].Address
}
return '[ Undisclosed recipients ]';
return '[ Undisclosed recipients ]'
},
getRelativeCreated: function (message) {
let d = new Date(message.Created)
return moment(d).fromNow().toString();
return moment(d).fromNow().toString()
},
browserNotify: function (title, message) {
if (!("Notification" in window)) {
return;
return
}
if (Notification.permission === "granted") {
let b = message.Subject;
let b = message.Subject
let options = {
body: message,
icon: 'notification.png'
}
new Notification(title, options);
new Notification(title, options)
}
},
requestNotifications: function () {
// check if the browser supports notifications
if (!("Notification" in window)) {
alert("This browser does not support desktop notification");
alert("This browser does not support desktop notification")
}
// we need to ask the user for permission
else if (Notification.permission !== "denied") {
let self = this;
let self = this
Notification.requestPermission().then(function (permission) {
// if the user accepts, let's create a notification
if (permission === "granted") {
self.browserNotify("Notifications enabled", "You will receive notifications when new mails are received.");
self.notificationsEnabled = true;
self.browserNotify("Notifications enabled", "You will receive notifications when new mails are received.")
self.notificationsEnabled = true
}
});
})
}
},
toggleSelected: function (e, id) {
e.preventDefault();
e.preventDefault()
if (this.isSelected(id)) {
this.selected = this.selected.filter(function (ele) {
return ele != id;
});
return ele != id
})
} else {
this.selected.push(id);
this.selected.push(id)
}
},
selectRange: function (e, id) {
e.preventDefault();
e.preventDefault()
let selecting = false;
let lastSelected = this.selected.length > 0 && this.selected[this.selected.length - 1];
let selecting = false
let lastSelected = this.selected.length > 0 && this.selected[this.selected.length - 1]
if (lastSelected == id) {
this.selected = this.selected.filter(function (ele) {
return ele != id;
});
return;
return ele != id
})
return
}
if (lastSelected === false) {
this.selected.push(id);
return;
this.selected.push(id)
return
}
for (let d of this.items) {
if (selecting) {
if (!this.isSelected(d.ID)) {
this.selected.push(d.ID);
this.selected.push(d.ID)
}
if (d.ID == lastSelected || d.ID == id) {
// reached backwards select
break;
break
}
} else if (d.ID == id || d.ID == lastSelected) {
if (!this.isSelected(d.ID)) {
this.selected.push(d.ID);
this.selected.push(d.ID)
}
selecting = true;
selecting = true
}
}
},
isSelected: function (id) {
return this.selected.indexOf(id) != -1;
return this.selected.indexOf(id) != -1
},
inSearch: function (tag) {
tag = tag.toLowerCase();
tag = tag.toLowerCase()
if (tag.match(/ /)) {
tag = '"' + tag + '"';
tag = '"' + tag + '"'
}
return this.search.toLowerCase().indexOf('tag:' + tag) > -1;
return this.search.toLowerCase().indexOf('tag:' + tag) > -1
},
loadInfo: function (e) {
e.preventDefault();
let self = this;
e.preventDefault()
let self = this
self.get('api/v1/info', false, function (response) {
self.appInfo = response.data;
self.modal('AppInfoModal').show();
});
self.appInfo = response.data
self.modal('AppInfoModal').show()
})
},
downloadMessageBody: function (str, ext) {
let dl = document.createElement('a');
dl.href = "data:text/plain," + encodeURIComponent(str);
dl.target = '_blank';
dl.download = this.message.ID + '.' + ext;
dl.click();
let dl = document.createElement('a')
dl.href = "data:text/plain," + encodeURIComponent(str)
dl.target = '_blank'
dl.download = this.message.ID + '.' + ext
dl.click()
},
initReleaseModal: function () {
this.releaseAddresses = false;
let addresses = [];
this.releaseAddresses = false
let addresses = []
for (let i in this.message.To) {
addresses.push(this.message.To[i].Address)
}
@ -661,31 +663,31 @@ export default {
}
// include only unique email addresses, regardless of casing
let uAddresses = new Map(addresses.map(a => [a.toLowerCase(), a]));
this.releaseAddresses = [...uAddresses.values()];
let uAddresses = new Map(addresses.map(a => [a.toLowerCase(), a]))
this.releaseAddresses = [...uAddresses.values()]
let self = this;
let self = this
window.setTimeout(function () {
// delay to allow elements to load
self.modal('ReleaseModal').show();
self.modal('ReleaseModal').show()
window.setTimeout(function () {
document.querySelector('#ReleaseModal input[role="combobox"]').focus()
}, 500);
}, 300);
}, 500)
}, 300)
},
setMessageToast: function (m) {
// don't display if browser notifications are enabled, or a toast is already displayed
if (this.notificationsEnabled || this.toastMessage) {
return;
return
}
this.toastMessage = m;
this.toastMessage = m
},
clearMessageToast: function () {
this.toastMessage = false;
this.toastMessage = false
}
}
}
@ -775,13 +777,13 @@ export default {
<img src="mailpit.svg" alt="Mailpit">
<span v-if="!total" class="ms-2">Mailpit</span>
</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"
placeholder="Search mailbox">
<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>
</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>
</button>
</div>
@ -807,8 +809,8 @@ export default {
<i class="bi bi-check2-square"></i>
</button>
<select v-model="limit" v-on:change="loadMessages" class="form-select form-select-sm d-inline w-auto me-2"
v-if="!searching">
<select v-model="limit" v-on:change="loadMessages" 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="50">50</option>
<option value="100">100</option>
@ -920,7 +922,7 @@ export default {
</li>
</ul>
</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"
:style="showTagColors ? { borderLeftColor: colorHash(tag), borderLeftWidth: '4px' } : ''"
v-on:click="tagSearch($event, tag)" :class="inSearch(tag) ? 'active' : ''">
@ -933,11 +935,13 @@ export default {
<MessageSummary v-if="message" :message="message"></MessageSummary>
<div class="position-fixed bottom-0 py-2 text-muted small w-100">
<a href="#" class="text-muted" v-on:click="loadInfo">
<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 btn btn-sm" v-on:click="loadInfo">
<i class="bi bi-info-circle-fill"></i>
About
</a>
<ThemeToggle />
</div>
</div>
@ -1196,53 +1200,4 @@ export default {
</div>
<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>

View File

@ -1,8 +1,7 @@
import { createApp } from 'vue';
import App from './App.vue';
import "./assets/styles.scss";
import "../../node_modules/bootstrap-icons/font/bootstrap-icons.scss";
import "bootstrap-icons/font/bootstrap-icons.scss";
import "bootstrap";
import "./color-modes";
createApp(App).mount('#app');

View File

@ -6,3 +6,4 @@ $link-decoration: none;
$primary: #2c3e50;
$list-group-disabled-color: #adb5bd;
$enable-negative-margins: true;
$body-color-dark: #e7eaed;

View File

@ -56,8 +56,17 @@
z-index: 1500;
}
.message.read:not(.active):not(.selected) {
color: $gray-500;
.message {
&.read {
color: $text-muted;
b {
font-weight: normal;
}
}
&.selected {
background: var(--bs-primary-bg-subtle);
}
}
#nav-plain-text .text-view,
@ -180,20 +189,6 @@
border-top: 0;
}
.message.selected {
background: $gray-300;
.text-muted {
color: $body-color !important;
}
&.read {
b {
font-weight: normal;
}
}
}
body.blur {
.privacy {
filter: blur(3px);
@ -280,8 +275,8 @@ body.blur {
https://prismjs.com/download.html#themes=prism-coy&languages=markup+css */
code[class*="language-"],
pre[class*="language-"] {
color: #000;
background: 0 0;
// color: #000;
// background: 0 0;
font-size: 0.85em;
text-align: left;
white-space: pre;
@ -314,7 +309,7 @@ code[class*="language-"] {
}
:not(pre) > code[class*="language-"],
pre[class*="language-"] {
background-color: #fdfdfd;
// background-color: #fdfdfd;
-webkit-box-sizing: border-box;
-moz-box-sizing: border-box;
box-sizing: border-box;
@ -364,7 +359,7 @@ pre[class*="language-"] {
.token.url,
.token.variable {
color: #a67f59;
background: rgba(255, 255, 255, 0.5);
// background: rgba(255, 255, 255, 0.5);
}
.token.atrule,
.token.attr-value,
@ -379,7 +374,7 @@ pre[class*="language-"] {
.language-css .token.string,
.style .token.string {
color: #a67f59;
background: rgba(255, 255, 255, 0.5);
// background: rgba(255, 255, 255, 0.5);
}
.token.important {
font-weight: 400;
@ -390,9 +385,9 @@ pre[class*="language-"] {
.token.italic {
font-style: italic;
}
.token.entity {
cursor: help;
}
// .token.entity {
// cursor: help;
// }
.token.namespace {
opacity: 0.7;
}

View File

@ -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);
});

View File

@ -14,14 +14,18 @@ export default {
<template>
<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">
<img v-if="isImage(part)" :src="'api/v1/message/'+message.ID+'/part/'+part.PartID+'/thumb'" class="card-img-top" alt="">
<img v-else src="" class="card-img-top" alt="">
<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">
<img v-if="isImage(part)" :src="'api/v1/message/' + message.ID + '/part/' + part.PartID + '/thumb'"
class="card-img-top" alt="">
<img v-else
src=""
class="card-img-top" alt="">
<div class="icon" v-if="!isImage(part)">
<i class="bi" :class="attachmentIcon(part)"></i>
</div>
<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>
<small>{{ getFileSize(part.Size) }}</small>
</p>
@ -29,7 +33,7 @@ export default {
{{ part.FileName != '' ? part.FileName : '[ unknown ]' }}
</p>
</div>
<div class="card-footer small border-0 text-center text-truncate">
<div class="card-footer small border-0 text-center text-truncate">
{{ part.FileName != '' ? part.FileName : '[ unknown ]' }}
</div>
</a>

View File

@ -1,6 +1,6 @@
<script>
import commonMixins from '../mixins.js';
import commonMixins from '../mixins.js'
export default {
props: {
@ -17,9 +17,9 @@ export default {
mounted() {
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.headers = response.data;
self.headers = response.data
});
},
@ -30,7 +30,7 @@ export default {
<div v-if="headers" class="small">
<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-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>
</div>

View File

@ -1,10 +1,10 @@
<script>
import commonMixins from '../mixins.js';
import Prism from "prismjs";
import Tags from "bootstrap5-tags";
import Attachments from './Attachments.vue';
import Headers from './Headers.vue';
import commonMixins from '../mixins.js'
import Prism from "prismjs"
import Tags from "bootstrap5-tags"
import Attachments from './Attachments.vue'
import Headers from './Headers.vue'
export default {
props: {
@ -14,7 +14,7 @@ export default {
components: {
Attachments,
Headers
Headers,
},
mixins: [commonMixins],
@ -41,20 +41,20 @@ export default {
watch: {
message: {
handler() {
let self = this;
self.showTags = false;
self.messageTags = self.message.Tags;
self.allTags = self.existingTags;
self.loadHeaders = false;
let self = this
self.showTags = false
self.messageTags = self.message.Tags
self.allTags = self.existingTags
self.loadHeaders = false
self.scaleHTMLPreview = 'display';// default view
// delay to select first tab and add HTML highlighting (prev/next)
self.$nextTick(function () {
self.renderUI();
self.showTags = true;
self.renderUI()
self.showTags = true
self.$nextTick(function () {
Tags.init("select[multiple]");
});
});
Tags.init("select[multiple]")
})
})
},
// force eager callback execution
immediate: true
@ -62,97 +62,112 @@ export default {
messageTags() {
// save changed to tags
if (this.showTags) {
this.saveTags();
this.saveTags()
}
},
scaleHTMLPreview() {
if (this.scaleHTMLPreview == 'display') {
let self = this;
let self = this
window.setTimeout(function () {
self.resizeIframes();
}, 500);
self.resizeIframes()
}, 500)
}
}
},
mounted() {
let self = this;
self.showTags = false;
self.allTags = self.existingTags;
window.addEventListener("resize", self.resizeIframes);
self.renderUI();
let self = this
self.showTags = false
self.allTags = self.existingTags
window.addEventListener("resize", self.resizeIframes)
self.renderUI()
let headersTab = document.getElementById('nav-headers-tab');
let headersTab = document.getElementById('nav-headers-tab')
headersTab.addEventListener('shown.bs.tab', function (event) {
self.loadHeaders = true;
});
self.loadHeaders = true
})
let rawTab = document.getElementById('nav-raw-tab');
let rawTab = document.getElementById('nav-raw-tab')
rawTab.addEventListener('shown.bs.tab', function (event) {
self.srcURI = 'api/v1/message/' + self.message.ID + '/raw';
self.resizeIframes();
});
self.srcURI = 'api/v1/message/' + self.message.ID + '/raw'
self.resizeIframes()
})
self.showTags = true;
self.showTags = true
self.$nextTick(function () {
Tags.init("select[multiple]");
});
Tags.init("select[multiple]")
})
},
unmounted: function () {
window.removeEventListener("resize", this.resizeIframes);
window.removeEventListener("resize", this.resizeIframes)
},
methods: {
renderUI: function () {
let self = this;
let self = this
// click the first non-disabled tab
document.querySelector('#nav-tab button:not([disabled])').click();
document.activeElement.blur(); // blur focus
document.getElementById('message-view').scrollTop = 0;
document.querySelector('#nav-tab button:not([disabled])').click()
document.activeElement.blur() // blur focus
document.getElementById('message-view').scrollTop = 0
// delay 0.2s until vue has rendered the iframe content
window.setTimeout(function () {
let p = document.getElementById('preview-html');
let p = document.getElementById('preview-html')
if (p) {
// make links open in new window
let anchorEls = p.contentWindow.document.body.querySelectorAll('a');
let anchorEls = p.contentWindow.document.body.querySelectorAll('a')
for (var i = 0; i < anchorEls.length; i++) {
let anchorEl = anchorEls[i];
let href = anchorEl.getAttribute('href');
let anchorEl = anchorEls[i]
let href = anchorEl.getAttribute('href')
if (href && href.match(/^http/)) {
anchorEl.setAttribute('target', '_blank');
anchorEl.setAttribute('target', '_blank')
}
}
self.resizeIframes();
self.resizeIframes()
}
}, 200);
}, 200)
// html highlighting
window.Prism = window.Prism || {};
window.Prism.manual = true;
Prism.highlightAll();
window.Prism = window.Prism || {}
window.Prism.manual = true
Prism.highlightAll()
},
resizeIframe: function (el) {
let i = el.target;
i.style.height = i.contentWindow.document.body.scrollHeight + 50 + 'px';
let i = el.target
i.style.height = i.contentWindow.document.body.scrollHeight + 50 + 'px'
},
resizeIframes: function () {
if (this.scaleHTMLPreview != 'display') {
return;
return
}
let h = document.getElementById('preview-html');
let h = document.getElementById('preview-html')
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 () {
let self = this;
let self = this
var data = {
ids: [this.message.ID],
@ -160,9 +175,9 @@ export default {
}
self.put('api/v1/tags', data, function (response) {
self.scrollInPlace = true;
self.$emit('loadMessages');
});
self.scrollInPlace = true
self.$emit('loadMessages')
})
},
// Convert plain text to HTML including anchor links
@ -195,7 +210,7 @@ export default {
</script>
<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="col-md">
<table class="messageHeaders">
@ -221,7 +236,7 @@ export default {
<template v-if="i > 0">, </template>
<span class="text-nowrap">{{ t.Name + " &lt;" + t.Address + "&gt;" }}</span>
</span>
<span v-else class="text-muted">[Undisclosed recipients]</span>
<span v-else class="text-body-secondary">[Undisclosed recipients]</span>
</td>
</tr>
<tr v-if="message.Cc && message.Cc.length" class="small">
@ -243,7 +258,7 @@ export default {
</tr>
<tr v-if="message.ReplyTo && message.ReplyTo.length" class="small">
<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">
<template v-if="i > 0">,</template>
{{ t.Name + " &lt;" + t.Address + "&gt;" }} </span>
@ -251,13 +266,13 @@ export default {
</tr>
<tr v-if="message.ReturnPath && message.ReturnPath != message.From.Address" class="small">
<th class="text-nowrap">Return-Path</th>
<td class="privacy text-muted">&lt;{{ message.ReturnPath }}&gt;</td>
<td class="privacy text-body-secondary">&lt;{{ message.ReturnPath }}&gt;</td>
</tr>
<tr>
<th class="small">Subject</th>
<td>
<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>
</tr>
<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">
<template v-for=" vals, key in responsiveSizes ">
<button class="btn" :class="scaleHTMLPreview == key ? 'btn-outline-primary' : ''"
:disabled="scaleHTMLPreview == key" :title="'Switch to ' + key + ' view'"
<button class="btn" :disabled="scaleHTMLPreview == key" :title="'Switch to ' + key + ' view'"
v-on:click=" scaleHTMLPreview = key">
<i class="bi" :class="'bi-' + key"></i>
</button>
@ -352,8 +366,8 @@ export default {
<Headers v-if="loadHeaders" :message="message"></Headers>
</div>
<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"
style="width: 100%; height: 300px; background: #fff; color: #15141A"></iframe>
<iframe v-if="srcURI" :src="srcURI" v-on:load="initRawIframe" frameborder="0"
style="width: 100%; height: 300px"></iframe>
</div>
</div>
</div>

View File

@ -1,7 +1,7 @@
<script>
import Tags from "bootstrap5-tags";
import commonMixins from '../mixins.js';
import Tags from "bootstrap5-tags"
import commonMixins from '../mixins.js'
export default {
props: {
@ -19,19 +19,19 @@ export default {
mixins: [commonMixins],
mounted() {
this.addresses = JSON.parse(JSON.stringify(this.releaseAddresses));
this.addresses = JSON.parse(JSON.stringify(this.releaseAddresses))
this.$nextTick(function () {
Tags.init("select[multiple]");
});
Tags.init("select[multiple]")
})
},
methods: {
releaseMessage: function () {
let self = this;
let self = this
// set timeout to allow for user clicking send before the tag filter has applied the tag
window.setTimeout(function () {
if (!self.addresses.length) {
return false;
return false
}
let data = {
@ -39,9 +39,9 @@ export default {
}
self.post('api/v1/message/' + self.message.ID + '/release', data, function (response) {
self.modal("ReleaseModal").hide();
});
}, 100);
self.modal("ReleaseModal").hide()
})
}, 100)
}
}
}
@ -57,19 +57,21 @@ export default {
<div class="modal-body">
<h6>Send this message to one or more addresses specified below.</h6>
<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">
<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 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">
<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 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">
<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..."

View File

@ -1,5 +1,5 @@
<script>
import commonMixins from '../mixins.js';
import commonMixins from '../mixins.js'
export default {
props: {
@ -12,7 +12,7 @@ export default {
<template>
<div class="card mt-4">
<div class="card-body text-muted small">
<div class="card-body text-body-secondary small">
<p class="card-text">
<b>Message date:</b><br>
<small>{{ messageDate(message.Date) }}</small>

View File

@ -1,5 +1,5 @@
<script>
import { Toast } from 'bootstrap';
import { Toast } from 'bootstrap'
export default {
props: {
@ -7,15 +7,15 @@ export default {
},
mounted() {
let self = this;
let el = document.getElementById('messageToast');
let self = this
let el = document.getElementById('messageToast')
if (el) {
el.addEventListener('hidden.bs.toast', () => {
self.$emit("clearMessageToast");
self.$emit("clearMessageToast")
})
let b = Toast.getOrCreateInstance(el);
b.show();
let b = Toast.getOrCreateInstance(el)
b.show()
}
}
}
@ -33,7 +33,7 @@ export default {
<div class="toast-body">
<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-else>[ no subject ]</template>
</a>

View 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>