2020-06-22 11:51:15 -04:00
|
|
|
pragma Singleton
|
|
|
|
|
|
|
|
import QtQuick 2.13
|
2021-10-28 00:27:49 +03:00
|
|
|
import shared 1.0
|
2020-06-22 11:51:15 -04:00
|
|
|
|
|
|
|
QtObject {
|
|
|
|
function isHex(value) {
|
2020-08-12 13:40:25 +10:00
|
|
|
return /^(-0x|0x)?[0-9a-fA-F]*$/i.test(value)
|
2020-06-22 11:51:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
function startsWith0x(value) {
|
|
|
|
return value.startsWith('0x')
|
|
|
|
}
|
|
|
|
|
|
|
|
function isChatKey(value) {
|
|
|
|
return startsWith0x(value) && isHex(value) && value.length === 132
|
|
|
|
}
|
2020-06-22 13:26:47 -04:00
|
|
|
|
2020-06-25 09:26:58 -04:00
|
|
|
function isValidETHNamePrefix(value) {
|
|
|
|
return !(value.trim() === "" || value.endsWith(".") || value.indexOf("..") > -1)
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:26:47 -04:00
|
|
|
function isAddress(value) {
|
|
|
|
return startsWith0x(value) && isHex(value) && value.length === 42
|
|
|
|
}
|
|
|
|
|
|
|
|
function isPrivateKey(value) {
|
|
|
|
return isHex(value) && ((startsWith0x(value) && value.length === 66) ||
|
|
|
|
(!startsWith0x(value) && value.length === 64))
|
|
|
|
}
|
2020-06-22 13:57:06 -04:00
|
|
|
|
2021-03-16 15:19:48 -04:00
|
|
|
function getCurrentThemeAccountColor(color) {
|
|
|
|
const upperCaseColor = color.toUpperCase()
|
|
|
|
if (Style.current.accountColors.indexOf(upperCaseColor) > -1) {
|
|
|
|
return upperCaseColor
|
|
|
|
}
|
|
|
|
|
|
|
|
let colorIndex
|
|
|
|
if (Style.current.name === Constants.lightThemeName) {
|
|
|
|
colorIndex = Style.darkTheme.accountColors.indexOf(upperCaseColor)
|
|
|
|
} else {
|
|
|
|
colorIndex = Style.lightTheme.accountColors.indexOf(upperCaseColor)
|
|
|
|
}
|
|
|
|
if (colorIndex === -1) {
|
|
|
|
// Unknown color
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return Style.current.accountColors[colorIndex]
|
|
|
|
}
|
|
|
|
|
2021-12-10 12:29:33 +01:00
|
|
|
function getMessageWithStyle(msg, isCurrentUser, hoveredLink = "") {
|
2021-03-23 14:15:26 -04:00
|
|
|
return `<style type="text/css">` +
|
|
|
|
`p, img, a, del, code, blockquote { margin: 0; padding: 0; }` +
|
|
|
|
`code {` +
|
2021-08-24 17:14:19 +03:00
|
|
|
`font-family: ${Style.current.fontCodeRegular.name};` +
|
|
|
|
`font-weight: 400;` +
|
|
|
|
`font-size: ${Style.current.secondaryTextFontSize};` +
|
|
|
|
`padding: 2px 4px;` +
|
|
|
|
`border-radius: 4px;` +
|
2021-03-23 14:15:26 -04:00
|
|
|
`background-color: ${Style.current.codeBackground};` +
|
2021-08-24 17:14:19 +03:00
|
|
|
`color: ${Style.current.black};` +
|
2021-03-23 14:15:26 -04:00
|
|
|
`white-space: pre;` +
|
|
|
|
`}` +
|
|
|
|
`p {` +
|
|
|
|
`line-height: 22px;` +
|
|
|
|
`}` +
|
|
|
|
`a {` +
|
2021-05-05 14:23:43 -04:00
|
|
|
`color: ${Style.current.linkColor};` +
|
2021-03-23 14:15:26 -04:00
|
|
|
`}` +
|
|
|
|
`a.mention {` +
|
|
|
|
`color: ${Style.current.mentionColor};` +
|
|
|
|
`background-color: ${Style.current.mentionBgColor};` +
|
|
|
|
`text-decoration: none;` +
|
2021-05-07 09:40:23 +10:00
|
|
|
`padding: 0px 2px;` +
|
2021-03-23 14:15:26 -04:00
|
|
|
`}` +
|
2021-05-07 09:40:23 +10:00
|
|
|
(hoveredLink !== "" ? `a.mention[href="${hoveredLink}"] { background-color: ${Style.current.mentionBgHoverColor}; }` : ``) +
|
2021-03-23 14:15:26 -04:00
|
|
|
`del {` +
|
|
|
|
`text-decoration: line-through;` +
|
|
|
|
`}` +
|
|
|
|
`table.blockquote td {` +
|
|
|
|
`padding-left: 10px;` +
|
|
|
|
`color: ${isCurrentUser ? Style.current.chatReplyCurrentUser : Style.current.secondaryText};` +
|
|
|
|
`}` +
|
|
|
|
`table.blockquote td.quoteline {` +
|
|
|
|
`background-color: ${isCurrentUser ? Style.current.chatReplyCurrentUser : Style.current.secondaryText};` +
|
|
|
|
`height: 100%;` +
|
|
|
|
`padding-left: 0;` +
|
|
|
|
`}` +
|
|
|
|
`.emoji {` +
|
|
|
|
`vertical-align: bottom;` +
|
|
|
|
`}` +
|
2021-06-29 10:49:32 -04:00
|
|
|
`span.isEdited {` +
|
|
|
|
`color: ${Style.current.secondaryText};` +
|
|
|
|
`margin-left: 5px` +
|
|
|
|
`}` +
|
2021-03-23 14:15:26 -04:00
|
|
|
`</style>` +
|
|
|
|
`${msg}`
|
|
|
|
}
|
|
|
|
|
2021-12-10 12:29:33 +01:00
|
|
|
function getReplyMessageStyle(msg, isCurrentUser) {
|
2021-06-11 15:50:52 -04:00
|
|
|
return `<style type="text/css">`+
|
2021-08-24 17:14:19 +03:00
|
|
|
`a {`+
|
2021-12-10 12:29:33 +01:00
|
|
|
`color: ${isCurrentUser? Style.current.white : Style.current.textColor};`+
|
2021-08-24 17:14:19 +03:00
|
|
|
`}`+
|
|
|
|
`a.mention {`+
|
|
|
|
`color: ${isCurrentUser ? Style.current.mentionColor : Style.current.turquoise};`+
|
|
|
|
`background-color: ${Style.current.mentionBgColor};` +
|
|
|
|
`}`+
|
|
|
|
`</style>`+
|
|
|
|
`</head>`+
|
|
|
|
`<body>`+
|
|
|
|
`${msg}`+
|
|
|
|
`</body>`+
|
|
|
|
`</html>`
|
2021-06-11 15:50:52 -04:00
|
|
|
}
|
|
|
|
|
2021-10-21 11:15:33 +02:00
|
|
|
function getLinkStyle(link, hoveredLink, textColor) {
|
2021-10-01 18:58:36 +03:00
|
|
|
return `<style type="text/css">` +
|
|
|
|
`a {` +
|
2021-10-21 11:15:33 +02:00
|
|
|
`color: ${textColor};` +
|
2021-10-01 18:58:36 +03:00
|
|
|
`text-decoration: none;` +
|
|
|
|
`}` +
|
|
|
|
(hoveredLink !== "" ? `a[href="${hoveredLink}"] { text-decoration: underline; }` : "") +
|
|
|
|
`</style>` +
|
|
|
|
`<a href="${link}">${link}</a>`
|
|
|
|
}
|
|
|
|
|
2020-06-22 13:57:06 -04:00
|
|
|
function isMnemonic(value) {
|
2020-07-27 11:11:24 -04:00
|
|
|
if(!value.match(/^([a-z\s]+)$/)){
|
|
|
|
return false;
|
|
|
|
}
|
2021-04-12 11:40:49 -04:00
|
|
|
return Utils.seedPhraseValidWordCount(value);
|
2020-06-22 13:57:06 -04:00
|
|
|
}
|
2020-06-25 09:26:58 -04:00
|
|
|
|
|
|
|
function compactAddress(addr, numberOfChars) {
|
|
|
|
if(addr.length <= 5 + (numberOfChars * 2)){ // 5 represents these chars 0x...
|
|
|
|
return addr;
|
|
|
|
}
|
|
|
|
return addr.substring(0, 2 + numberOfChars) + "..." + addr.substring(addr.length - numberOfChars);
|
|
|
|
}
|
2020-07-15 17:04:14 -04:00
|
|
|
|
|
|
|
function linkifyAndXSS(inputText) {
|
|
|
|
//URLs starting with http://, https://, or ftp://
|
2021-02-25 14:32:39 -05:00
|
|
|
var replacePattern1 = /(\b(https?|ftp|statusim):\/\/[-A-Z0-9+&@#\/%?=~_|!:,.;]*[-A-Z0-9+&@#\/%=~_|])/gim;
|
2020-07-15 17:04:14 -04:00
|
|
|
var replacedText = inputText.replace(replacePattern1, "<a href='$1'>$1</a>");
|
|
|
|
|
|
|
|
//URLs starting with "www." (without // before it, or it'd re-link the ones done above).
|
|
|
|
var replacePattern2 = /(^|[^\/])(www\.[\S]+(\b|$))/gim;
|
|
|
|
replacedText = replacedText.replace(replacePattern2, "$1<a href='http://$2'>$2</a>");
|
|
|
|
|
2020-11-05 15:03:53 -05:00
|
|
|
return XSS.filterXSS(replacedText)
|
|
|
|
}
|
|
|
|
|
|
|
|
function filterXSS(inputText) {
|
|
|
|
return XSS.filterXSS(inputText)
|
2020-07-15 17:04:14 -04:00
|
|
|
}
|
2020-07-27 13:30:20 -04:00
|
|
|
|
2020-09-16 13:11:31 +02:00
|
|
|
function toLocaleString(val, locale, options) {
|
|
|
|
return NumberPolyFill.toLocaleString(val, locale, options)
|
|
|
|
}
|
|
|
|
|
2020-07-27 13:30:20 -04:00
|
|
|
function isOnlyEmoji(inputText) {
|
|
|
|
var emoji_regex = /^(?:[\u2700-\u27bf]|(?:\ud83c[\udde6-\uddff]){2}|[\ud800-\udbff][\udc00-\udfff]|[\u0023-\u0039]\ufe0f?\u20e3|\u3299|\u3297|\u303d|\u3030|\u24c2|\ud83c[\udd70-\udd71]|\ud83c[\udd7e-\udd7f]|\ud83c\udd8e|\ud83c[\udd91-\udd9a]|\ud83c[\udde6-\uddff]|[\ud83c[\ude01-\ude02]|\ud83c\ude1a|\ud83c\ude2f|[\ud83c[\ude32-\ude3a]|[\ud83c[\ude50-\ude51]|\u203c|\u2049|[\u25aa-\u25ab]|\u25b6|\u25c0|[\u25fb-\u25fe]|\u00a9|\u00ae|\u2122|\u2139|\ud83c\udc04|[\u2600-\u26FF]|\u2b05|\u2b06|\u2b07|\u2b1b|\u2b1c|\u2b50|\u2b55|\u231a|\u231b|\u2328|\u23cf|[\u23e9-\u23f3]|[\u23f8-\u23fa]|\ud83c\udccf|\u2934|\u2935|[\u2190-\u21ff]|\s)+$/;
|
|
|
|
return emoji_regex.test(inputText);
|
2020-08-07 15:26:51 -04:00
|
|
|
}
|
2020-07-27 13:30:20 -04:00
|
|
|
|
2020-08-07 15:26:51 -04:00
|
|
|
function removeStatusEns(userName){
|
|
|
|
return userName.endsWith(".stateofus.eth") ? userName.substr(0, userName.length - 14) : userName
|
2020-07-27 13:30:20 -04:00
|
|
|
}
|
2020-08-12 13:40:25 +10:00
|
|
|
|
2021-08-10 14:01:37 -04:00
|
|
|
function addStatusEns(userName){
|
|
|
|
return userName.endsWith(".eth") ? userName : userName + ".stateofus.eth"
|
|
|
|
}
|
|
|
|
|
2020-08-12 13:40:25 +10:00
|
|
|
function isValidAddress(inputValue) {
|
2020-11-04 13:37:53 +01:00
|
|
|
return inputValue !== "0x" && /^0x[a-fA-F0-9]{40}$/.test(inputValue)
|
2020-08-12 13:40:25 +10:00
|
|
|
}
|
2020-08-13 18:24:51 +10:00
|
|
|
|
2020-09-09 13:04:01 +02:00
|
|
|
function isValidEns(inputValue) {
|
2020-11-04 13:37:53 +01:00
|
|
|
if (!inputValue) {
|
|
|
|
return false
|
|
|
|
}
|
2020-09-09 13:04:01 +02:00
|
|
|
const isEmail = /(?:[a-z0-9!#$%&'*+/=?^_`{|}~-]+(?:\.[a-z0-9!#$%&'*+/=?^_`{|}~-]+)*|"(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21\x23-\x5b\x5d-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])*")@(?:(?:[a-z0-9](?:[a-z0-9-]*[a-z0-9])?\.)+[a-z0-9](?:[a-z0-9-]*[a-z0-9])?|\[(?:(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?)\.){3}(?:25[0-5]|2[0-4][0-9]|[01]?[0-9][0-9]?|[a-z0-9-]*[a-z0-9]:(?:[\x01-\x08\x0b\x0c\x0e-\x1f\x21-\x5a\x53-\x7f]|\\[\x01-\x09\x0b\x0c\x0e-\x7f])+)\])/.test(inputValue)
|
|
|
|
const isDomain = /(?:(?:(?<thld>[\w\-]*)(?:\.))?(?<sld>[\w\-]*))\.(?<tld>[\w\-]*)/.test(inputValue)
|
2020-11-04 13:37:53 +01:00
|
|
|
return isEmail || isDomain || (inputValue.startsWith("@") && inputValue.length > 1)
|
2020-09-09 13:04:01 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-08-13 18:24:51 +10:00
|
|
|
/**
|
|
|
|
* Removes trailing zeros from a string-representation of a number. Throws
|
|
|
|
* if parameter is not a string
|
|
|
|
*/
|
|
|
|
function stripTrailingZeros(strNumber) {
|
|
|
|
if (!(typeof strNumber === "string")) {
|
2020-09-01 13:49:05 +10:00
|
|
|
try {
|
|
|
|
strNumber = strNumber.toString()
|
|
|
|
} catch(e) {
|
|
|
|
throw "[Utils.stripTrailingZeros] input parameter must be a string"
|
|
|
|
}
|
2020-08-13 18:24:51 +10:00
|
|
|
}
|
|
|
|
return strNumber.replace(/(\.[0-9]*[1-9])0+$|\.0*$/,'$1')
|
|
|
|
}
|
2020-08-17 16:46:13 -04:00
|
|
|
|
2020-09-08 17:24:12 -04:00
|
|
|
/**
|
|
|
|
* Removes starting zeros from a string-representation of a number. Throws
|
|
|
|
* if parameter is not a string
|
|
|
|
*/
|
|
|
|
function stripStartingZeros(strNumber) {
|
|
|
|
if (!(typeof strNumber === "string")) {
|
|
|
|
try {
|
|
|
|
strNumber = strNumber.toString()
|
|
|
|
} catch(e) {
|
|
|
|
throw "[Utils.stripStartingZeros] input parameter must be a string"
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return strNumber.replace(/^(0*)([0-9\.]+)/, "$2")
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-08-17 16:46:13 -04:00
|
|
|
function setColorAlpha(color, alpha) {
|
|
|
|
return Qt.hsla(color.hslHue, color.hslSaturation, color.hslLightness, alpha)
|
|
|
|
}
|
2020-08-26 11:15:40 -04:00
|
|
|
|
|
|
|
function formatTime(timestamp) {
|
|
|
|
let messageDate = new Date(Math.floor(timestamp))
|
|
|
|
let minutes = messageDate.getMinutes();
|
|
|
|
let hours = messageDate.getHours();
|
|
|
|
return (hours < 10 ? "0" + hours : hours) + ":" + (minutes < 10 ? "0" + minutes : minutes)
|
|
|
|
}
|
2020-09-01 13:49:05 +10:00
|
|
|
|
2021-07-15 17:59:30 +03:00
|
|
|
function formatAgeFromTime(timestamp, epoch) {
|
|
|
|
epoch++ // pretending the parameter is not unused
|
2021-07-15 17:20:17 +03:00
|
|
|
const now = new Date()
|
|
|
|
const messageDate = new Date(Math.floor(timestamp))
|
|
|
|
const diffMs = now - messageDate
|
|
|
|
const diffMin = Math.floor(diffMs / 60000)
|
|
|
|
if (diffMin < 1) {
|
2021-07-22 17:03:59 +02:00
|
|
|
//% "NOW"
|
|
|
|
return qsTrId("now")
|
2021-07-15 17:20:17 +03:00
|
|
|
}
|
|
|
|
const diffHour = Math.floor(diffMin / 60)
|
|
|
|
if (diffHour < 1) {
|
2021-07-22 17:03:59 +02:00
|
|
|
//% "%1M"
|
|
|
|
return qsTrId("-1m").arg(diffMin)
|
2021-07-15 17:20:17 +03:00
|
|
|
}
|
|
|
|
const diffDay = Math.floor(diffHour / 24)
|
|
|
|
if (diffDay < 1) {
|
2021-07-22 17:03:59 +02:00
|
|
|
//% "%1H"
|
|
|
|
return qsTrId("-1h").arg(diffHour)
|
2021-07-15 17:20:17 +03:00
|
|
|
}
|
2021-07-22 17:03:59 +02:00
|
|
|
//% "%1D"
|
|
|
|
return qsTrId("-1d").arg(diffDay)
|
2021-07-15 17:20:17 +03:00
|
|
|
}
|
|
|
|
|
2021-01-20 00:16:42 +04:00
|
|
|
function formatShortDateStr(longStr) {
|
|
|
|
const dmKeys = {
|
|
|
|
// Days
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Sun"
|
2021-02-18 11:36:05 -05:00
|
|
|
Sunday: qsTrId("sun"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Mon"
|
2021-02-18 11:36:05 -05:00
|
|
|
Monday: qsTrId("mon"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Tue"
|
2021-02-18 11:36:05 -05:00
|
|
|
Tuesday: qsTrId("tue"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Wed"
|
2021-02-18 11:36:05 -05:00
|
|
|
Wednesday: qsTrId("wed"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Thu"
|
2021-02-18 11:36:05 -05:00
|
|
|
Thursday: qsTrId("thu"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Fri"
|
2021-02-18 11:36:05 -05:00
|
|
|
Friday: qsTrId("fri"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Sat"
|
2021-02-18 11:36:05 -05:00
|
|
|
Saturday: qsTrId("sat"),
|
2021-01-20 00:16:42 +04:00
|
|
|
// Months
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Jan"
|
2021-02-18 11:36:05 -05:00
|
|
|
January: qsTrId("jan"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Feb"
|
2021-02-18 11:36:05 -05:00
|
|
|
February: qsTrId("feb"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Mar"
|
2021-02-18 11:36:05 -05:00
|
|
|
March: qsTrId("mar"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Apr"
|
2021-02-18 11:36:05 -05:00
|
|
|
April: qsTrId("apr"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "May"
|
2021-02-18 11:36:05 -05:00
|
|
|
May: qsTrId("may"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Jun"
|
2021-02-18 11:36:05 -05:00
|
|
|
June: qsTrId("jun"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Jul"
|
2021-02-18 11:36:05 -05:00
|
|
|
July: qsTrId("jul"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Aug"
|
2021-02-18 11:36:05 -05:00
|
|
|
August: qsTrId("aug"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Sep"
|
2021-02-18 11:36:05 -05:00
|
|
|
September: qsTrId("sep"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Oct"
|
2021-02-18 11:36:05 -05:00
|
|
|
October: qsTrId("oct"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Nov"
|
2021-02-18 11:36:05 -05:00
|
|
|
November: qsTrId("nov"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Dec"
|
2021-02-18 11:36:05 -05:00
|
|
|
December: qsTrId("dec")
|
2021-01-20 00:16:42 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
let shortStr = longStr;
|
|
|
|
for (const [key, value] of Object.entries(dmKeys)) {
|
|
|
|
shortStr = shortStr.replace(key, value);
|
|
|
|
shortStr = shortStr.replace(key.toLowerCase(), value);
|
|
|
|
shortStr = shortStr.replace(key.toUpperCase(), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
return shortStr;
|
|
|
|
}
|
|
|
|
|
2021-01-04 15:47:25 +01:00
|
|
|
function formatDateTime(timestamp, locale) {
|
2020-12-22 11:22:04 +01:00
|
|
|
let now = new Date()
|
|
|
|
let yesterday = new Date()
|
|
|
|
yesterday.setDate(now.getDate()-1)
|
|
|
|
let messageDate = new Date(Math.floor(timestamp))
|
|
|
|
let lastWeek = new Date()
|
|
|
|
lastWeek.setDate(now.getDate()-7)
|
|
|
|
|
|
|
|
let minutes = messageDate.getMinutes();
|
|
|
|
let hours = messageDate.getHours();
|
|
|
|
|
|
|
|
if (now.toDateString() === messageDate.toDateString()) {
|
|
|
|
return (hours < 10 ? "0" + hours : hours) + ":" + (minutes < 10 ? "0" + minutes : minutes)
|
|
|
|
} else if (yesterday.toDateString() === messageDate.toDateString()) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Yesterday"
|
2021-02-18 11:36:05 -05:00
|
|
|
return qsTrId("yesterday")
|
2020-12-22 11:22:04 +01:00
|
|
|
} else if (lastWeek.getTime() < messageDate.getTime()) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Sunday"
|
2021-02-18 11:36:05 -05:00
|
|
|
let days = [qsTrId("sunday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Monday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("monday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Tuesday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("tuesday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Wednesday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("wednesday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Thursday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("thursday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Friday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("friday"),
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Saturday"
|
2021-02-18 11:36:05 -05:00
|
|
|
qsTrId("saturday")];
|
2020-12-22 11:22:04 +01:00
|
|
|
return days[messageDate.getDay()];
|
|
|
|
} else {
|
2021-01-20 00:16:42 +04:00
|
|
|
return formatShortDateStr(new Date().toLocaleDateString(Qt.locale(locale)))
|
2020-12-22 11:22:04 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-01 13:49:05 +10:00
|
|
|
function findAssetBySymbol(assets, symbolToFind) {
|
|
|
|
for(var i=0; i<assets.rowCount(); i++) {
|
|
|
|
const symbol = assets.rowData(i, "symbol")
|
|
|
|
if (symbol.toLowerCase() === symbolToFind.toLowerCase()) {
|
|
|
|
return {
|
|
|
|
name: assets.rowData(i, "name"),
|
|
|
|
symbol,
|
2021-11-04 15:56:26 -04:00
|
|
|
value: assets.rowData(i, "balance"),
|
|
|
|
fiatBalanceDisplay: assets.rowData(i, "currencyBalance"),
|
2020-09-01 13:49:05 +10:00
|
|
|
address: assets.rowData(i, "address"),
|
2021-11-04 15:56:26 -04:00
|
|
|
fiatBalance: assets.rowData(i, "currencyBalance")
|
2020-09-01 13:49:05 +10:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-09-24 17:41:59 -04:00
|
|
|
|
2020-11-06 14:46:21 -05:00
|
|
|
function isValidChannelName(channelName) {
|
|
|
|
return (/^[a-z0-9\-]+$/.test(channelName))
|
|
|
|
}
|
|
|
|
|
2020-10-29 11:19:27 -04:00
|
|
|
function isURL(text) {
|
2021-04-29 15:51:16 -04:00
|
|
|
return (/https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}(\.[a-zA-Z0-9()]{1,6})?\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/.test(text))
|
2020-10-29 11:19:27 -04:00
|
|
|
}
|
|
|
|
|
2021-04-19 13:34:03 +03:00
|
|
|
function isURLWithOptionalProtocol(text) {
|
|
|
|
return (/^(www\.)?[-a-zA-Z0-9@:%._\+~#=]{1,256}\.[a-zA-Z0-9()]{1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)/.test(text))
|
|
|
|
}
|
|
|
|
|
2020-12-11 15:29:46 -05:00
|
|
|
function isHexColor(c) {
|
2021-01-06 09:58:44 -05:00
|
|
|
return (/^#([0-9A-Fa-f]{6}|[0-9A-Fa-f]{3})$/i.test(c))
|
2020-12-11 15:29:46 -05:00
|
|
|
}
|
|
|
|
|
2020-09-28 11:23:04 -04:00
|
|
|
function isSpace(c) {
|
|
|
|
return (/( |\t|\n|\r)/.test(c))
|
2020-09-24 17:41:59 -04:00
|
|
|
}
|
|
|
|
|
2020-10-03 15:53:46 +02:00
|
|
|
function getTick(wordCount) {
|
|
|
|
return (wordCount === 12 || wordCount === 15 ||
|
|
|
|
wordCount === 18 || wordCount === 21 || wordCount === 24)
|
|
|
|
? "✓ " : "";
|
|
|
|
}
|
|
|
|
|
2021-04-12 11:40:49 -04:00
|
|
|
function isValidNumberOfWords(wordCount) {
|
|
|
|
return !!getTick(wordCount);
|
|
|
|
}
|
|
|
|
|
2020-10-03 15:53:46 +02:00
|
|
|
function countWords(text) {
|
|
|
|
if (text.trim() === "")
|
|
|
|
return 0;
|
2021-04-12 11:40:49 -04:00
|
|
|
return text.trim().replace(/ +/g, " ").split(" ").length;
|
|
|
|
}
|
|
|
|
|
|
|
|
function seedPhraseValidWordCount(text) {
|
|
|
|
return isValidNumberOfWords(countWords(text))
|
2020-10-03 15:53:46 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns text in the format "✓ 12 words" for seed phrases input boxes
|
|
|
|
*/
|
|
|
|
function seedPhraseWordCountText(text) {
|
|
|
|
let wordCount = countWords(text);
|
2021-02-18 11:36:05 -05:00
|
|
|
//% "words"
|
|
|
|
return getTick(wordCount) + wordCount.toString() + " " + qsTrId("words")
|
2020-10-03 15:53:46 +02:00
|
|
|
}
|
2020-10-07 13:47:21 +11:00
|
|
|
|
|
|
|
function uuid() {
|
|
|
|
return Date.now().toString(36) + Math.random().toString(36).substr(2, 5)
|
|
|
|
}
|
2020-11-23 16:41:57 -04:00
|
|
|
|
|
|
|
function getNetworkName(network){
|
|
|
|
switch(network){
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Mainnet with upstream RPC"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkMainnet: return qsTrId("mainnet-with-upstream-rpc")
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "POA Network"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkPOA: return qsTrId("poa-network")
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "xDai Chain"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkXDai: return qsTrId("xdai-chain")
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Goerli with upstream RPC"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkGoerli: return qsTrId("goerli-with-upstream-rpc")
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Rinkeby with upstream RPC"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkRinkeby: return qsTrId("rinkeby-with-upstream-rpc")
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Ropsten with upstream RPC"
|
2021-02-18 11:36:05 -05:00
|
|
|
case Constants.networkRopsten: return qsTrId("ropsten-with-upstream-rpc")
|
2020-11-23 16:41:57 -04:00
|
|
|
default: return network
|
|
|
|
}
|
|
|
|
}
|
2020-12-04 00:26:05 +02:00
|
|
|
|
2020-12-04 16:53:00 +02:00
|
|
|
function validatePasswords(item, firstPasswordField, repeatPasswordField) {
|
2020-12-04 00:26:05 +02:00
|
|
|
switch (item) {
|
|
|
|
case "first":
|
|
|
|
if (firstPasswordField.text === "") {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "You need to enter a password"
|
2020-12-04 00:26:05 +02:00
|
|
|
return [false, qsTrId("you-need-to-enter-a-password")];
|
|
|
|
} else if (firstPasswordField.text.length < 6) {
|
2021-07-30 12:02:22 -04:00
|
|
|
//% "Password needs to be 6 characters or more"
|
|
|
|
return [false, qsTrId("password-needs-to-be-6-characters-or-more")];
|
2020-12-04 00:26:05 +02:00
|
|
|
}
|
|
|
|
return [true, ""];
|
|
|
|
|
|
|
|
case "repeat":
|
|
|
|
if (repeatPasswordField.text === "") {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "You need to repeat your password"
|
2020-12-04 00:26:05 +02:00
|
|
|
return [false, qsTrId("you-need-to-repeat-your-password")];
|
|
|
|
} else if (repeatPasswordField.text !== firstPasswordField.text) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Passwords don't match"
|
|
|
|
return [false, qsTrId("passwords-don-t-match")];
|
2020-12-04 00:26:05 +02:00
|
|
|
}
|
|
|
|
return [true, ""];
|
|
|
|
|
|
|
|
default:
|
|
|
|
return [false, ""];
|
|
|
|
}
|
|
|
|
}
|
feat: whitelist gifs (no url extension needed)
Fixes #1377.
Fixes #1479.
Two sites have been added to the whitelist: giphy.com and tenor.com.
`imageUrls` in its entirety has been removed and instead all links are being handle through the message `linkUrls`. This prevents double-handling of urls that may or may not be images.
The logic to automatically show links previews works like this:
1. If the setting "display chat images" is enabled, all links that *contain* ".png", ".jpg", ".jpeg", ".svg", ".gif" will be automatically shown. If the URL doesn't contain the extension, we are not downloading it. This was meant to be somewhat of a security compromise as we do not want to download each and every link posted in a message just to find out its true content type.
2. If the above setting is *disabled*, then we follow the whitelist settings for tenor and giphy. This allows us to preview gifs that do not have a file extension in their url.
feat: bump status-go to the commit that supports the new whitelist (https://github.com/status-im/status-go/pull/2094), and also lets us get link preview data from urls in the whitelist. NOTE: this commit was branched off status-go `develop`, so once it is merged, and we update this PR to the new commit, we will effectively be getting status-go develop changes. We *could* base that status-go PR off of master if it makes things easier.
fix: height on settings update issue
feat: move date/time of message below links
fix: layout issues when changing setting `neverAskAboutUnfurlingAgain`
feat: Add MessageBorder component to aid in showing rounded corners with different radius
2020-12-11 11:53:44 +11:00
|
|
|
|
2021-09-24 15:03:57 +03:00
|
|
|
function validatePINs(item, firstPINField, repeatPINField) {
|
|
|
|
switch (item) {
|
|
|
|
case "first":
|
|
|
|
if (firstPINField.text === "") {
|
|
|
|
return [false, qsTr("You need to enter a PIN")];
|
|
|
|
} else if (!/^\d+$/.test(firstPINField.text)) {
|
|
|
|
return [false, qsTr("The PIN must contain only digits")];
|
|
|
|
} else if (firstPINField.text.length != 6) {
|
|
|
|
return [false, qsTr("The PIN must be exactly 6 digits")];
|
|
|
|
}
|
|
|
|
return [true, ""];
|
|
|
|
|
|
|
|
case "repeat":
|
|
|
|
if (repeatPINField.text === "") {
|
|
|
|
return [false, qsTr("You need to repeat your PIN")];
|
|
|
|
} else if (repeatPINField.text !== firstPINField.text) {
|
|
|
|
return [false, qsTr("PIN don't match")];
|
|
|
|
}
|
|
|
|
return [true, ""];
|
|
|
|
|
|
|
|
default:
|
|
|
|
return [false, ""];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
feat: whitelist gifs (no url extension needed)
Fixes #1377.
Fixes #1479.
Two sites have been added to the whitelist: giphy.com and tenor.com.
`imageUrls` in its entirety has been removed and instead all links are being handle through the message `linkUrls`. This prevents double-handling of urls that may or may not be images.
The logic to automatically show links previews works like this:
1. If the setting "display chat images" is enabled, all links that *contain* ".png", ".jpg", ".jpeg", ".svg", ".gif" will be automatically shown. If the URL doesn't contain the extension, we are not downloading it. This was meant to be somewhat of a security compromise as we do not want to download each and every link posted in a message just to find out its true content type.
2. If the above setting is *disabled*, then we follow the whitelist settings for tenor and giphy. This allows us to preview gifs that do not have a file extension in their url.
feat: bump status-go to the commit that supports the new whitelist (https://github.com/status-im/status-go/pull/2094), and also lets us get link preview data from urls in the whitelist. NOTE: this commit was branched off status-go `develop`, so once it is merged, and we update this PR to the new commit, we will effectively be getting status-go develop changes. We *could* base that status-go PR off of master if it makes things easier.
fix: height on settings update issue
feat: move date/time of message below links
fix: layout issues when changing setting `neverAskAboutUnfurlingAgain`
feat: Add MessageBorder component to aid in showing rounded corners with different radius
2020-12-11 11:53:44 +11:00
|
|
|
function getHostname(url) {
|
|
|
|
const rgx = /\:\/\/(?:[a-zA-Z0-9\-]*\.{1,}){1,}[a-zA-Z0-9]*/i
|
|
|
|
const matches = rgx.exec(url)
|
2021-02-25 14:32:39 -05:00
|
|
|
if (!matches || !matches.length) {
|
|
|
|
if (url.includes(Constants.deepLinkPrefix)) {
|
|
|
|
return Constants.deepLinkPrefix
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
feat: whitelist gifs (no url extension needed)
Fixes #1377.
Fixes #1479.
Two sites have been added to the whitelist: giphy.com and tenor.com.
`imageUrls` in its entirety has been removed and instead all links are being handle through the message `linkUrls`. This prevents double-handling of urls that may or may not be images.
The logic to automatically show links previews works like this:
1. If the setting "display chat images" is enabled, all links that *contain* ".png", ".jpg", ".jpeg", ".svg", ".gif" will be automatically shown. If the URL doesn't contain the extension, we are not downloading it. This was meant to be somewhat of a security compromise as we do not want to download each and every link posted in a message just to find out its true content type.
2. If the above setting is *disabled*, then we follow the whitelist settings for tenor and giphy. This allows us to preview gifs that do not have a file extension in their url.
feat: bump status-go to the commit that supports the new whitelist (https://github.com/status-im/status-go/pull/2094), and also lets us get link preview data from urls in the whitelist. NOTE: this commit was branched off status-go `develop`, so once it is merged, and we update this PR to the new commit, we will effectively be getting status-go develop changes. We *could* base that status-go PR off of master if it makes things easier.
fix: height on settings update issue
feat: move date/time of message below links
fix: layout issues when changing setting `neverAskAboutUnfurlingAgain`
feat: Add MessageBorder component to aid in showing rounded corners with different radius
2020-12-11 11:53:44 +11:00
|
|
|
return matches[0].substring(3)
|
|
|
|
}
|
|
|
|
|
|
|
|
function hasImageExtension(url) {
|
2021-07-13 10:34:13 +03:00
|
|
|
return Constants.acceptedImageExtensions.some(ext => url.toLowerCase().includes(ext))
|
2021-03-10 15:59:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
function hasDragNDropImageExtension(url) {
|
2021-07-13 10:34:13 +03:00
|
|
|
return Constants.acceptedDragNDropImageExtensions.some(ext => url.toLowerCase().includes(ext))
|
2021-03-10 15:59:01 +11:00
|
|
|
}
|
|
|
|
|
|
|
|
function deduplicate(array) {
|
|
|
|
return Array.from(new Set(array))
|
feat: whitelist gifs (no url extension needed)
Fixes #1377.
Fixes #1479.
Two sites have been added to the whitelist: giphy.com and tenor.com.
`imageUrls` in its entirety has been removed and instead all links are being handle through the message `linkUrls`. This prevents double-handling of urls that may or may not be images.
The logic to automatically show links previews works like this:
1. If the setting "display chat images" is enabled, all links that *contain* ".png", ".jpg", ".jpeg", ".svg", ".gif" will be automatically shown. If the URL doesn't contain the extension, we are not downloading it. This was meant to be somewhat of a security compromise as we do not want to download each and every link posted in a message just to find out its true content type.
2. If the above setting is *disabled*, then we follow the whitelist settings for tenor and giphy. This allows us to preview gifs that do not have a file extension in their url.
feat: bump status-go to the commit that supports the new whitelist (https://github.com/status-im/status-go/pull/2094), and also lets us get link preview data from urls in the whitelist. NOTE: this commit was branched off status-go `develop`, so once it is merged, and we update this PR to the new commit, we will effectively be getting status-go develop changes. We *could* base that status-go PR off of master if it makes things easier.
fix: height on settings update issue
feat: move date/time of message below links
fix: layout issues when changing setting `neverAskAboutUnfurlingAgain`
feat: Add MessageBorder component to aid in showing rounded corners with different radius
2020-12-11 11:53:44 +11:00
|
|
|
}
|
2021-04-12 11:40:49 -04:00
|
|
|
|
2021-06-29 14:47:28 +02:00
|
|
|
function hasUpperCaseLetter(str) {
|
|
|
|
return (/[A-Z]/.test(str))
|
|
|
|
}
|
|
|
|
|
|
|
|
function convertSpacesToDashesAndUpperToLowerCase(str)
|
|
|
|
{
|
|
|
|
if (str.includes(" "))
|
|
|
|
str = str.replace(/ /g, "-")
|
|
|
|
|
|
|
|
if(hasUpperCaseLetter(str))
|
|
|
|
str = str.toLowerCase()
|
|
|
|
|
|
|
|
return str
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Validation section start */
|
|
|
|
|
|
|
|
enum Validate {
|
|
|
|
NoEmpty = 0x01,
|
|
|
|
TextLength = 0x02,
|
|
|
|
TextHexColor = 0x04,
|
|
|
|
TextLowercaseLettersNumberAndDashes = 0x08
|
|
|
|
}
|
|
|
|
|
|
|
|
function validateAndReturnError(str, validation, fieldName = "field", limit = 0)
|
|
|
|
{
|
|
|
|
let errMsg = ""
|
|
|
|
|
|
|
|
if(validation & Utils.Validate.NoEmpty && str === "") {
|
2021-07-22 17:03:59 +02:00
|
|
|
//% "You need to enter a %1"
|
|
|
|
errMsg = qsTrId("you-need-to-enter-a--1").arg(fieldName)
|
2021-06-29 14:47:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(validation & Utils.Validate.TextLength && str.length > limit) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "The %1 cannot exceed %2 characters"
|
2021-07-16 22:22:50 +02:00
|
|
|
errMsg = qsTrId("the--1-cannot-exceed--2-characters").arg(fieldName, limit)
|
2021-06-29 14:47:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(validation & Utils.Validate.TextHexColor && !isHexColor(str)) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Must be an hexadecimal color (eg: #4360DF)"
|
2021-07-16 22:22:50 +02:00
|
|
|
errMsg = qsTrId("must-be-an-hexadecimal-color--eg---4360df-")
|
2021-06-29 14:47:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if(validation & Utils.Validate.TextLowercaseLettersNumberAndDashes && !isValidChannelName(str)) {
|
2021-07-30 11:54:19 -04:00
|
|
|
//% "Use only lowercase letters (a to z), numbers & dashes (-). Do not use chat keys."
|
2021-07-16 22:22:50 +02:00
|
|
|
errMsg = qsTrId("use-only-lowercase-letters--a-to-z---numbers---dashes------do-not-use-chat-keys-")
|
2021-06-29 14:47:28 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return errMsg
|
|
|
|
}
|
|
|
|
|
2021-08-16 15:46:00 +02:00
|
|
|
function getErrorMessage(errors, fieldName) {
|
|
|
|
if (errors) {
|
|
|
|
if (errors.minLength) {
|
|
|
|
return errors.minLength.min === 1 ?
|
|
|
|
qsTr("You need to enter a %1").arg(fieldName) :
|
|
|
|
qsTr("Value has to be at least %1 characters long").arg(errors.minLength.min)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
2021-06-29 14:47:28 +02:00
|
|
|
/* Validation section end */
|
|
|
|
|
|
|
|
|
2021-12-31 13:29:51 +01:00
|
|
|
function getContactDetailsAsJson(publicKey) {
|
|
|
|
let jsonObj = mainModule.getContactDetailsAsJson(publicKey)
|
2022-01-04 13:06:05 +01:00
|
|
|
try {
|
|
|
|
let obj = JSON.parse(jsonObj)
|
|
|
|
return obj
|
|
|
|
}
|
|
|
|
catch (e) {
|
2021-12-31 13:29:51 +01:00
|
|
|
// This log is available only in debug mode, if it's annoying we can remove it
|
2022-01-04 13:06:05 +01:00
|
|
|
console.debug("error parsing contact details for public key: ", publicKey, " error: ", e.message)
|
|
|
|
|
2021-12-31 13:29:51 +01:00
|
|
|
return {
|
|
|
|
displayName: "",
|
|
|
|
displayIcon: "",
|
|
|
|
isDisplayIconIdenticon: true,
|
|
|
|
publicKey: publicKey,
|
|
|
|
name: "",
|
|
|
|
ensVerified: false,
|
|
|
|
alias: "",
|
|
|
|
lastUpdated: 0,
|
|
|
|
lastUpdatedLocally: 0,
|
|
|
|
localNickname: "",
|
|
|
|
identicon: "",
|
|
|
|
thumbnailImage: "",
|
|
|
|
largeImage: "",
|
|
|
|
isContact: false,
|
|
|
|
isBlocked: false,
|
|
|
|
requestReceived: false,
|
|
|
|
isSyncing: false,
|
|
|
|
removed: false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-29 14:47:28 +02:00
|
|
|
|
2021-04-12 11:40:49 -04:00
|
|
|
// Leave this function at the bottom of the file as QT Creator messes up the code color after this
|
|
|
|
function isPunct(c) {
|
|
|
|
return /(!|\@|#|\$|%|\^|&|\*|\(|\)|_|\+|\||-|=|\\|{|}|[|]|"|;|'|<|>|\?|,|\.|\/)/.test(c)
|
|
|
|
}
|
fix: prevent crash on generate account wrong password
Fixes #2448.
Currently, if a wrong password is entered when generating a wallet account, the app will crash due to attempting to decode a `GeneratedAccount ` from an rpc response containing only an error.
With this PR, we are detecting if an error is returned in the response, and if so, raising a StatusGoException. This exception is caught in the call chain, and translated in to a `StatusGoError` which is serialised and sent to the QML view, where it is parsed and displayed as an invalid password error in the input box.
refactor: remove string return values as error messages in wallet model
In the wallet model, we were passing back empty strings for no error, or an error as a string. This is not only confusing, but does not benefit from leaning on the compiler and strong types. One has to read the entire code to understand if a string result is returned when there is no error instead of implicitly being able to understand there is no return type.
To alleviate this, account creation fundtions that do not need to return a value have been changed to a void return type, and raise `StatusGoException` if there is an error encountered. This can be caught in the call chain and used as necessary (ie to pass to QML).
refactor: move invalid password string detection to Utils
Currently, we are reading returned view model values and checking to see if they include a known string from Status Go that means there was an invalid password used. This string was placed in the codebased in mulitple locations.
This PR moves the string check to a Utils function and updates all the references to use the function in Utils.
2021-05-13 14:41:48 +10:00
|
|
|
|
|
|
|
function isInvalidPasswordMessage(msg) {
|
|
|
|
return (
|
|
|
|
msg.includes("could not decrypt key with given password") ||
|
|
|
|
msg.includes("invalid password")
|
|
|
|
);
|
|
|
|
}
|
2020-06-22 11:51:15 -04:00
|
|
|
}
|