Merge branch 'clearnet' into multi-device

pull/578/head
Beaudan Brown 6 years ago
commit 8d6fee4aec

@ -688,7 +688,7 @@
<!-- Profile -->
<div class='page'>
<div class='display-name-input'>
<div class='input-header'>Enter a name that will be shown to all your contacts</div>
<div class='input-header'>Enter your public display name (alphanumeric characters and spaces only)</div>
<input class='form-control' type='text' id='display-name' placeholder='Display Name (optional)' autocomplete='off' spellcheck='false' maxlength='25'>
</div>
<div class='password-inputs'>

@ -466,6 +466,7 @@
timestamp: this.get('timestamp'),
title: this.getTitle(),
unreadCount: this.get('unreadCount') || 0,
mentionedUs: this.get('mentionedUs') || false,
showFriendRequestIndicator: this.isPendingFriendRequest(),
isBlocked: this.isBlocked(),
@ -2007,6 +2008,21 @@
const unreadCount = unreadMessages.length - read.length;
this.set({ unreadCount });
const mentionRead = (() => {
const stillUnread = unreadMessages.filter(
m => m.get('received_at') > newestUnreadDate
);
const ourNumber = textsecure.storage.user.getNumber();
return !stillUnread.some(
m => m.propsForMessage.text.indexOf(`@${ourNumber}`) !== -1
);
})();
if (mentionRead) {
this.set({ mentionedUs: false });
}
await window.Signal.Data.updateConversation(this.id, this.attributes, {
Conversation: Whisper.Conversation,
});

@ -1988,6 +1988,12 @@
c.onReadMessage(message);
}
} else {
const ourNumber = textsecure.storage.user.getNumber();
if (message.attributes.body.indexOf(`@${ourNumber}`) !== -1) {
conversation.set({ mentionedUs: true });
}
conversation.set({
unreadCount: conversation.get('unreadCount') + 1,
isArchived: false,

@ -314,10 +314,11 @@
this.selectMember = this.selectMember.bind(this);
const updateMemberList = async () => {
const maxToFetch = 1000;
const allMessages = await window.Signal.Data.getMessagesByConversation(
this.model.id,
{
limit: Number.MAX_SAFE_INTEGER,
limit: maxToFetch,
MessageCollection: Whisper.MessageCollection,
}
);
@ -1603,11 +1604,16 @@
this.$messageField.val(),
cursorPos
);
let firstHalf = `${prev}@${member.authorPhoneNumber}`;
const handle = this.memberView.addPubkeyMapping(
member.authorProfileName,
member.authorPhoneNumber
);
let firstHalf = `${prev}${handle}`;
let newCursorPos = firstHalf.length;
const needExtraWhitespace =
end.length === 0 || /[a-fA-F0-9@]/.test(end[0]);
const needExtraWhitespace = end.length === 0 || /\b/.test(end[0]);
if (needExtraWhitespace) {
firstHalf += ' ';
newCursorPos += 1;
@ -1810,7 +1816,9 @@
this.model.clearTypingTimers();
const input = this.$messageField;
const message = window.Signal.Emoji.replaceColons(input.val()).trim();
let message = this.memberView.replaceMentions(input.val());
message = window.Signal.Emoji.replaceColons(message).trim();
let toast;
if (extension.expired()) {
@ -1853,6 +1861,7 @@
);
input.val('');
this.memberView.deleteMention();
this.setQuoteMessage(null);
this.resetLinkPreview();
this.focusMessageFieldAndClearDisabled();
@ -2186,12 +2195,172 @@
}
},
handleDeleteOrBackspace(event, isDelete) {
const $input = this.$messageField[0];
const text = this.$messageField.val();
// Only handle the case when nothing is selected
if ($input.selectionDirection !== 'none') {
// Note: if this ends up deleting a handle, we should
// (ideally) check if we need to update the mapping in
// `this.memberView`, but that's not vital as we already
// reset it on every 'send'
return;
}
const mentions = this.memberView.pendingMentions();
const _ = window.Lodash; // no underscore.js please
const predicate = isDelete ? _.startsWith : _.endsWith;
const pos = $input.selectionStart;
const part = isDelete ? text.substr(pos) : text.substr(0, pos);
const curMention = _.keys(mentions).find(key => predicate(part, key));
if (!curMention) {
return;
}
event.preventDefault();
const beforeMention = isDelete
? text.substr(0, pos)
: text.substr(0, pos - curMention.length);
const afterMention = isDelete
? text.substr(pos + curMention.length)
: text.substr(pos);
const resText = beforeMention + afterMention;
// NOTE: this doesn't work well with undo/redo, perhaps
// we should fix it one day
this.$messageField.val(resText);
const nextPos = isDelete ? pos : pos - curMention.length;
$input.selectionStart = nextPos;
$input.selectionEnd = nextPos;
this.memberView.deleteMention(curMention);
},
handleLeftRight(event, isLeft) {
// Return next cursor position candidate before we take
// various modifier keys into account
const nextPos = (text, cursorPos, isLeft2, isAltPressed) => {
// If the next char is ' ', skip it if Alt is pressed
let pos = cursorPos;
if (isAltPressed) {
const nextChar = isLeft2
? text.substr(pos - 1, 1)
: text.substr(pos, 1);
if (nextChar === ' ') {
pos = isLeft2 ? pos - 1 : pos + 1;
}
}
const part = isLeft2 ? text.substr(0, pos) : text.substr(pos);
const mentions = this.memberView.pendingMentions();
const predicate = isLeft2
? window.Lodash.endsWith
: window.Lodash.startsWith;
const curMention = _.keys(mentions).find(key => predicate(part, key));
const offset = curMention ? curMention.length : 1;
const resPos = isLeft2 ? Math.max(0, pos - offset) : pos + offset;
return resPos;
};
event.preventDefault();
const $input = this.$messageField[0];
const posStart = $input.selectionStart;
const posEnd = $input.selectionEnd;
const text = this.$messageField.val();
const posToChange =
$input.selectionDirection === 'forward' ? posEnd : posStart;
let newPos = nextPos(text, posToChange, isLeft, event.altKey);
// If command (macos) key is pressed, go to the beginning/end
// (this shouldn't affect Windows, but we should double check that)
if (event.metaKey) {
newPos = isLeft ? 0 : text.length;
}
// Alt would normally make the cursor go until the next whitespace,
// but we need to take the presence of a mention into account
if (event.altKey) {
const searchFrom = isLeft ? posToChange - 1 : posToChange + 1;
const toSearch = isLeft
? text.substr(0, searchFrom)
: text.substr(searchFrom);
// Note: we don't seem to support tabs etc, thus no /\s/
let nextAltPos = isLeft
? toSearch.lastIndexOf(' ')
: toSearch.indexOf(' ');
if (nextAltPos === -1) {
nextAltPos = isLeft ? 0 : text.length;
} else if (isLeft) {
nextAltPos += 1;
}
if (isLeft) {
newPos = Math.min(newPos, nextAltPos);
} else {
newPos = Math.max(newPos, nextAltPos + searchFrom);
}
}
// ==== Handle selection business ====
let newPosStart = newPos;
let newPosEnd = newPos;
let direction = $input.selectionDirection;
if (event.shiftKey) {
if (direction === 'none') {
if (isLeft) {
direction = 'backward';
} else {
direction = 'forward';
}
}
} else {
direction = 'none';
}
if (direction === 'forward') {
newPosStart = posStart;
} else if (direction === 'backward') {
newPosEnd = posEnd;
}
if (newPosStart === newPosEnd) {
direction = 'none';
}
$input.setSelectionRange(newPosStart, newPosEnd, direction);
},
// Note: not only input, but keypresses too (rename?)
handleInputEvent(event) {
// Note: schedule the member list handler shortly afterwards, so
// that the input element has time to update its cursor position to
// what the user would expect
window.requestAnimationFrame(this.maybeShowMembers.bind(this, event));
if (this.model.isPublic()) {
window.requestAnimationFrame(this.maybeShowMembers.bind(this, event));
}
const keyCode = event.which || event.keyCode;
@ -2235,6 +2404,20 @@
if (keyPressedLeft || keyPressedRight) {
this.$messageField.trigger('input');
this.handleLeftRight(event, keyPressedLeft);
return;
}
const keyPressedDelete = keyCode === 46;
const keyPressedBackspace = keyCode === 8;
if (keyPressedDelete) {
this.handleDeleteOrBackspace(event, true);
}
if (keyPressedBackspace) {
this.handleDeleteOrBackspace(event, false);
}
this.updateMessageFieldSize();
@ -2309,6 +2492,7 @@
let allMembers = window.lokiPublicChatAPI.getListOfMembers();
allMembers = allMembers.filter(d => !!d);
allMembers = allMembers.filter(d => d.authorProfileName !== 'Anonymous');
allMembers = _.uniq(allMembers, true, d => d.authorPhoneNumber);
const cursorPos = event.target.selectionStart;

@ -9,6 +9,7 @@
Whisper.MemberListView = Whisper.View.extend({
initialize(options) {
this.member_list = [];
this.memberMapping = {};
this.selected_idx = 0;
this.onClicked = options.onClicked;
this.render();
@ -43,6 +44,31 @@
this.render();
}
},
replaceMentions(message) {
let result = message;
// Sort keys from long to short, so we don't have to
// worry about one key being a substring of another
const keys = _.sortBy(_.keys(this.memberMapping), d => -d.length);
keys.forEach(key => {
const pubkey = this.memberMapping[key];
result = result.split(key).join(`@${pubkey}`);
});
return result;
},
pendingMentions() {
return this.memberMapping;
},
deleteMention(mention) {
if (mention) {
delete this.memberMapping[mention];
} else {
// Delete all mentions if no argument is passed
this.memberMapping = {};
}
},
membersShown() {
return this.member_list.length !== 0;
},
@ -60,5 +86,21 @@
selectedMember() {
return this.member_list[this.selected_idx];
},
addPubkeyMapping(name, pubkey) {
let handle = `@${name}`;
let chars = 4;
while (
_.has(this.memberMapping, handle) &&
this.memberMapping[handle] !== pubkey
) {
const shortenedPubkey = pubkey.substr(pubkey.length - chars);
handle = `@${name}(..${shortenedPubkey})`;
chars += 1;
}
this.memberMapping[handle] = pubkey;
return handle;
},
});
})();

@ -1,10 +1,12 @@
/* global Whisper,
/* global
Whisper,
$,
getAccountManager,
textsecure,
i18n,
passwordUtil,
_,
setTimeout
*/
/* eslint-disable more/no-then */
@ -15,6 +17,10 @@
window.Whisper = window.Whisper || {};
const REGISTER_INDEX = 0;
const PROFILE_INDEX = 1;
let currentPageIndex = REGISTER_INDEX;
Whisper.StandaloneRegistrationView = Whisper.View.extend({
templateName: 'standalone',
className: 'full-screen-flow standalone-fullscreen',
@ -67,8 +73,24 @@
this.onSecondaryDeviceRegistered = this.onSecondaryDeviceRegistered.bind(
this
);
const sanitiseNameInput = () => {
const oldVal = this.$('#display-name').val();
this.$('#display-name').val(oldVal.replace(/[^a-zA-Z0-9 ]/g, ''));
};
this.$('#display-name').get(0).oninput = () => {
sanitiseNameInput();
};
this.$('#display-name').get(0).onpaste = () => {
// Sanitise data immediately after paste because it's easier
setTimeout(() => {
sanitiseNameInput();
});
};
},
events: {
keyup: 'onKeyup',
'validation input.number': 'onValidation',
'click #request-voice': 'requestVoice',
'click #request-sms': 'requestSMSVerification',
@ -94,12 +116,13 @@
$(this).hide();
} else {
$(this).show();
currentPageIndex = pageIndex;
}
});
},
async showRegisterPage() {
this.registrationParams = {};
this.showPage(0);
this.showPage(REGISTER_INDEX);
},
async showProfilePage(mnemonic, language) {
this.registrationParams = {
@ -109,7 +132,25 @@
this.$passwordInput.val('');
this.$passwordConfirmationInput.val('');
this.onValidatePassword();
this.showPage(1);
this.showPage(PROFILE_INDEX);
this.$('#display-name').focus();
},
onKeyup(event) {
if (currentPageIndex !== PROFILE_INDEX) {
// Only want enter/escape keys to work on profile page
return;
}
switch (event.key) {
case 'Enter':
this.onSaveProfile();
break;
case 'Escape':
case 'Esc':
this.onBack();
break;
default:
}
},
async register(mnemonic, language) {
// Make sure the password is valid

@ -22,6 +22,8 @@ if (config.appInstance) {
title += ` - ${config.appInstance}`;
}
window.Lodash = require('lodash');
window.platform = process.platform;
window.getDefaultPoWDifficulty = () => config.defaultPoWDifficulty;
window.getTitle = () => title;

@ -67,3 +67,32 @@
}
}
}
.module-conversation-list-item--mentioned-us {
border-left: 4px solid #ffb000 !important;
}
.at-symbol {
background-color: #ffb000;
color: $color-black;
text-align: center;
padding-top: 1px;
padding-left: 3px;
padding-right: 3px;
position: absolute;
right: -6px;
top: 12px;
font-weight: 300;
font-size: 11px;
letter-spacing: 0.25px;
height: 16px;
min-width: 16px;
border-radius: 8px;
box-shadow: 0px 0px 0px 1px $color-dark-85;
}

@ -1895,7 +1895,7 @@
position: absolute;
right: -6px;
top: 6px;
top: -6px;
font-weight: 300;
font-size: 11px;

@ -26,6 +26,7 @@ export type PropsData = {
lastUpdated: number;
unreadCount: number;
mentionedUs: boolean;
isSelected: boolean;
isTyping: boolean;
@ -93,12 +94,17 @@ export class ConversationListItem extends React.PureComponent<Props> {
}
public renderUnread() {
const { unreadCount } = this.props;
const { unreadCount, mentionedUs } = this.props;
if (unreadCount > 0) {
const atSymbol = mentionedUs ? <p className="at-symbol">@</p> : null;
return (
<div className="module-conversation-list-item__unread-count">
{unreadCount}
<div>
<p className="module-conversation-list-item__unread-count">
{unreadCount}
</p>
{atSymbol}
</div>
);
}
@ -285,6 +291,7 @@ export class ConversationListItem extends React.PureComponent<Props> {
showFriendRequestIndicator,
isBlocked,
style,
mentionedUs,
} = this.props;
const triggerId = `${phoneNumber}-ctxmenu-${Date.now()}`;
@ -305,6 +312,9 @@ export class ConversationListItem extends React.PureComponent<Props> {
unreadCount > 0
? 'module-conversation-list-item--has-unread'
: null,
unreadCount > 0 && mentionedUs
? 'module-conversation-list-item--mentioned-us'
: null,
isSelected ? 'module-conversation-list-item--is-selected' : null,
showFriendRequestIndicator
? 'module-conversation-list-item--has-friend-request'

@ -49,6 +49,7 @@ export type ConversationType = {
isClosable?: boolean;
lastUpdated: number;
unreadCount: number;
mentionedUs: boolean;
isSelected: boolean;
isTyping: boolean;
isFriend?: boolean;

@ -24,6 +24,7 @@ describe('state/selectors/conversations', () => {
isMe: false,
lastUpdated: Date.now(),
unreadCount: 1,
mentionedUs: false,
isSelected: false,
isTyping: false,
},
@ -39,6 +40,7 @@ describe('state/selectors/conversations', () => {
isMe: false,
lastUpdated: Date.now(),
unreadCount: 1,
mentionedUs: false,
isSelected: false,
isTyping: false,
},
@ -54,6 +56,7 @@ describe('state/selectors/conversations', () => {
isMe: false,
lastUpdated: Date.now(),
unreadCount: 1,
mentionedUs: false,
isSelected: false,
isTyping: false,
},
@ -69,6 +72,7 @@ describe('state/selectors/conversations', () => {
isMe: false,
lastUpdated: Date.now(),
unreadCount: 1,
mentionedUs: false,
isSelected: false,
isTyping: false,
},
@ -84,6 +88,7 @@ describe('state/selectors/conversations', () => {
isMe: false,
lastUpdated: Date.now(),
unreadCount: 1,
mentionedUs: false,
isSelected: false,
isTyping: false,
},

Loading…
Cancel
Save