2019-07-08 13:52:41 +10:00
|
|
|
// Copyright 2019 miruka
|
|
|
|
// This file is part of harmonyqml, licensed under LGPLv3.
|
|
|
|
|
2019-07-06 09:54:16 +10:00
|
|
|
function arrayToModelItem(keys_name, array) {
|
|
|
|
// Convert an array to an object suitable to be in a model, example:
|
|
|
|
// [1, 2, 3] → [{keys_name: 1}, {keys_name: 2}, {keys_name: 3}]
|
|
|
|
var items = []
|
|
|
|
|
|
|
|
for (var i = 0; i < array.length; i++) {
|
|
|
|
var obj = {}
|
|
|
|
obj[keys_name] = array[i]
|
|
|
|
items.push(obj)
|
|
|
|
}
|
|
|
|
return items
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-04 12:31:29 +10:00
|
|
|
function hueFrom(string) {
|
|
|
|
// Calculate and return a unique hue between 0 and 1 for the string
|
|
|
|
var hue = 0
|
|
|
|
for (var i = 0; i < string.length; i++) {
|
|
|
|
hue += string.charCodeAt(i) * 99
|
|
|
|
}
|
|
|
|
return hue % 360 / 360
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-05 06:56:34 +10:00
|
|
|
function avatarColor(name) {
|
2019-07-04 12:31:29 +10:00
|
|
|
return Qt.hsla(
|
|
|
|
hueFrom(name),
|
2019-07-07 07:50:55 +10:00
|
|
|
theme.avatar.background.saturation,
|
|
|
|
theme.avatar.background.lightness,
|
|
|
|
theme.avatar.background.alpha
|
2019-07-04 12:31:29 +10:00
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-05 06:56:34 +10:00
|
|
|
function nameColor(name) {
|
2019-07-04 12:31:29 +10:00
|
|
|
return Qt.hsla(
|
|
|
|
hueFrom(name),
|
2019-07-07 07:50:55 +10:00
|
|
|
theme.displayName.saturation,
|
|
|
|
theme.displayName.lightness,
|
2019-07-04 12:31:29 +10:00
|
|
|
1
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-16 10:10:43 +10:00
|
|
|
function coloredNameHtml(name, user_id, display_text) {
|
2019-07-07 14:24:23 +10:00
|
|
|
// substring: remove leading @
|
2019-07-16 10:10:43 +10:00
|
|
|
return "<font color='" + nameColor(name || alt_id.substring(1)) + "'>" +
|
|
|
|
escapeHtml(display_text || name || alt_id) +
|
2019-07-05 06:56:34 +10:00
|
|
|
"</font>"
|
2019-07-04 14:24:21 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-04 12:31:29 +10:00
|
|
|
function escapeHtml(string) {
|
|
|
|
// Replace special HTML characters by encoded alternatives
|
|
|
|
return string.replace("&", "&")
|
|
|
|
.replace("<", "<")
|
|
|
|
.replace(">", ">")
|
|
|
|
.replace('"', """)
|
|
|
|
.replace("'", "'")
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-04 14:24:21 +10:00
|
|
|
function eventIsMessage(ev) {
|
|
|
|
return /^RoomMessage($|[A-Z])/.test(ev.eventType)
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-04 12:31:29 +10:00
|
|
|
function translatedEventContent(ev) {
|
2019-07-05 06:01:44 +10:00
|
|
|
// ev: timelines item
|
2019-07-04 12:31:29 +10:00
|
|
|
if (ev.translatable == false) { return ev.content }
|
|
|
|
|
|
|
|
// %S → sender display name
|
2019-07-08 12:41:32 +10:00
|
|
|
var name = users.find(ev.senderId).displayName
|
2019-07-04 14:24:21 +10:00
|
|
|
var text = ev.content.replace("%S", coloredNameHtml(name, ev.senderId))
|
2019-07-04 12:31:29 +10:00
|
|
|
|
|
|
|
// %T → target (event state_key) display name
|
2019-07-09 03:24:05 +10:00
|
|
|
if (ev.targetUserId) {
|
|
|
|
var tname = users.find(ev.targetUserId).displayName
|
|
|
|
text = text.replace("%T", coloredNameHtml(tname, ev.targetUserId))
|
2019-07-04 12:31:29 +10:00
|
|
|
}
|
|
|
|
|
|
|
|
text = qsTr(text)
|
2019-07-04 14:24:21 +10:00
|
|
|
if (ev.translatable == true) { return text }
|
2019-07-04 12:31:29 +10:00
|
|
|
|
|
|
|
// Else, model.translatable should be an array of args
|
2019-07-04 14:24:21 +10:00
|
|
|
for (var i = 0; ev.translatable.length; i++) {
|
|
|
|
text = text.arg(ev.translatable[i])
|
2019-07-04 12:31:29 +10:00
|
|
|
}
|
|
|
|
return text
|
|
|
|
}
|
2019-07-07 19:49:02 +10:00
|
|
|
|
|
|
|
|
|
|
|
function filterMatches(filter, text) {
|
|
|
|
filter = filter.toLowerCase()
|
|
|
|
text = text.toLowerCase()
|
|
|
|
|
|
|
|
var words = filter.split(" ")
|
|
|
|
|
|
|
|
for (var i = 0; i < words.length; i++) {
|
|
|
|
if (words[i] && text.indexOf(words[i]) == -1) {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
}
|
2019-07-14 10:15:20 +10:00
|
|
|
|
|
|
|
|
|
|
|
function thumbnailParametersFor(width, height) {
|
|
|
|
// https://matrix.org/docs/spec/client_server/latest#thumbnails
|
|
|
|
|
|
|
|
if (width > 640 || height > 480)
|
|
|
|
return {width: 800, height: 600, fillMode: Image.PreserveAspectFit}
|
|
|
|
|
|
|
|
if (width > 320 || height > 240)
|
|
|
|
return {width: 640, height: 480, fillMode: Image.PreserveAspectFit}
|
|
|
|
|
|
|
|
if (width > 96 || height > 96)
|
|
|
|
return {width: 320, height: 240, fillMode: Image.PreserveAspectFit}
|
|
|
|
|
|
|
|
if (width > 32 || height > 32)
|
|
|
|
return {width: 96, height: 96, fillMode: Image.PreserveAspectCrop}
|
|
|
|
|
|
|
|
return {width: 32, height: 32, fillMode: Image.PreserveAspectCrop}
|
|
|
|
}
|