Merge pull request #520 from msgmaxim/mentions2

Custom message rendering of mentions
pull/527/head
sachaaaaa 6 years ago committed by GitHub
commit b13a4f3e56
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -56,6 +56,16 @@ class LokiPublicChatAPI extends EventEmitter {
thisServer.unregisterChannel(channelId);
this.servers.splice(i, 1);
}
getListOfMembers() {
return this.allMembers;
}
// TODO: make this private (or remove altogether) when
// we switch to polling the server for group members
setListOfMembers(members) {
this.allMembers = members;
}
}
class LokiPublicServerAPI {
@ -226,6 +236,8 @@ class LokiPublicChannelAPI {
this.pollForDeletions();
this.pollForChannel();
this.pollForModerators();
// TODO: poll for group members here?
}
stop() {

@ -310,6 +310,26 @@
this.$emojiPanelContainer = this.$('.emoji-panel-container');
this.model.updateTextInputState();
this.selectMember = this.selectMember.bind(this);
const updateMemberList = async () => {
const allMessages = await window.Signal.Data.getMessagesByConversation(
this.model.id,
{
limit: Number.MAX_SAFE_INTEGER,
MessageCollection: Whisper.MessageCollection,
}
);
const allMembers = allMessages.models.map(d => d.propsForMessage);
window.lokiPublicChatAPI.setListOfMembers(allMembers);
};
if (this.model.isPublic()) {
updateMemberList();
setInterval(updateMemberList, 10000);
}
},
events: {
@ -1563,22 +1583,41 @@
dialog.focusCancel();
},
selectMember(member) {
const stripQuery = input => {
const pos = input.lastIndexOf('@');
stripQuery(text, cursorPos) {
const end = text.slice(cursorPos).search(/[^a-fA-F0-9]/);
const mentionEnd = end === -1 ? text.length : cursorPos + end;
// This should never happen, but we check just in case
if (pos === -1) {
return input;
}
const stripped = text.substr(0, mentionEnd);
return input.substr(0, pos);
};
const mentionStart = stripped.lastIndexOf('@');
const query = stripped.substr(mentionStart, mentionEnd - mentionStart);
const prev = stripQuery(this.$messageField.val());
const result = `${prev}@${member.authorPhoneNumber} `;
return [stripped.substr(0, mentionStart), query, text.substr(mentionEnd)];
},
selectMember(member) {
const cursorPos = this.$messageField[0].selectionStart;
// Note: skipping the middle value here
const [prev, , end] = this.stripQuery(
this.$messageField.val(),
cursorPos
);
let firstHalf = `${prev}@${member.authorPhoneNumber}`;
let newCursorPos = firstHalf.length;
const needExtraWhitespace =
end.length === 0 || /[a-fA-F0-9@]/.test(end[0]);
if (needExtraWhitespace) {
firstHalf += ' ';
newCursorPos += 1;
}
const result = firstHalf + end;
this.$messageField.val(result);
this.$messageField[0].selectionStart = newCursorPos;
this.$messageField[0].selectionEnd = newCursorPos;
this.$messageField.trigger('input');
},
@ -2149,7 +2188,10 @@
// Note: not only input, but keypresses too (rename?)
handleInputEvent(event) {
this.maybeShowMembers(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));
const keyCode = event.which || event.keyCode;
@ -2238,13 +2280,19 @@
return false;
};
const getQuery = input => {
// This is not quite the same as stripQuery
// as this one searches until the current
// cursor position
const getQuery = (srcLine, cursorPos) => {
const input = srcLine.substr(0, cursorPos);
const atPos = input.lastIndexOf('@');
if (atPos === -1) {
return null;
}
// Whitespace is required right before @
// Whitespace is required right before @ unless
// the beginning of line
if (atPos > 0 && /\w/.test(input.substr(atPos - 1, 1))) {
return null;
}
@ -2259,24 +2307,28 @@
return query;
};
const query = getQuery(event.target.value);
// TODO: for now, extract members from the conversation,
// but change to use a server endpoint in the future
let allMembers = this.model.messageCollection.models.map(
d => d.propsForMessage
);
let allMembers = window.lokiPublicChatAPI.getListOfMembers();
allMembers = allMembers.filter(d => !!d);
allMembers = _.uniq(allMembers, true, d => d.authorPhoneNumber);
const cursorPos = event.target.selectionStart;
// can't use pubkeyPattern here, as we are matching incomplete
// pubkeys (including the single @)
const query = getQuery(event.target.value, cursorPos);
let membersToShow = [];
if (query) {
if (query !== null) {
membersToShow =
query !== ''
? allMembers.filter(m => filterMembers(query, m))
: allMembers;
}
membersToShow = membersToShow.map(m =>
_.pick(m, ['authorPhoneNumber', 'authorProfileName', 'id'])
);
this.memberView.updateMembers(membersToShow);
},

@ -447,3 +447,7 @@ if (config.environment === 'test') {
};
/* eslint-enable global-require, import/no-extraneous-dependencies */
}
window.shortenPubkey = pubkey => `(...${pubkey.substring(pubkey.length - 6)})`;
window.pubkeyPattern = /@[a-fA-F0-9]{64,66}\b/g;

@ -30,6 +30,26 @@
}
}
.mention-profile-name {
color: rgb(194, 244, 255);
background-color: rgb(66, 121, 150);
text-decoration: underline;
border-radius: 4px;
margin: 2px;
padding: 2px;
user-select: none;
}
.mention-profile-name-us {
background-color: rgba(255, 197, 50, 1);
color: black;
}
.message-highlighted {
border-radius: $message-container-border-radius;
background-color: rgba(255, 197, 50, 0.2);
}
.dark-theme {
.member-list-container {
.member-item {

@ -167,7 +167,7 @@
.module-message__container {
position: relative;
display: inline-block;
border-radius: 16px;
border-radius: $message-container-border-radius;
padding-right: 12px;
padding-left: 12px;
padding-top: 10px;

@ -199,6 +199,7 @@ $header-height: 55px;
$button-height: 24px;
$border-radius: 5px;
$message-container-border-radius: 16px;
$font-size: 14px;
$font-size-small: (13/14) + em;

@ -0,0 +1,142 @@
import React from 'react';
import { RenderTextCallbackType } from '../../types/Util';
import classNames from 'classnames';
declare global {
interface Window {
lokiPublicChatAPI: any;
shortenPubkey: any;
pubkeyPattern: any;
}
}
interface MentionProps {
key: number;
text: string;
}
interface MentionState {
found: any;
}
class Mention extends React.Component<MentionProps, MentionState> {
private intervalHandle: any = null;
constructor(props: any) {
super(props);
this.tryRenameMention = this.tryRenameMention.bind(this);
}
public componentWillMount() {
const found = this.findMember(this.props.text.slice(1));
this.setState({ found });
this.tryRenameMention();
// TODO: give up after some period of time?
this.intervalHandle = setInterval(this.tryRenameMention, 30000);
}
public componentWillUnmount() {
this.clearOurInterval();
}
public render() {
if (this.state.found) {
// TODO: We don't have to search the database of message just to know that the message is for us!
const us =
this.state.found.authorPhoneNumber === window.lokiPublicChatAPI.ourKey;
const className = classNames(
'mention-profile-name',
us && 'mention-profile-name-us'
);
const profileName = this.state.found.authorProfileName;
const displayedName =
profileName && profileName.length > 0 ? profileName : 'Anonymous';
return <span className={className}>{displayedName}</span>;
} else {
return (
<span className="mention-profile-name">
{window.shortenPubkey(this.props.text)}
</span>
);
}
}
private clearOurInterval() {
clearInterval(this.intervalHandle);
}
private tryRenameMention() {
const found = this.findMember(this.props.text.slice(1));
if (found) {
this.setState({ found });
this.clearOurInterval();
}
}
private findMember(pubkey: String) {
const members = window.lokiPublicChatAPI.getListOfMembers();
if (!members) {
return null;
}
const filtered = members.filter((m: any) => !!m);
return filtered.find(
({ authorPhoneNumber: pn }: any) => pn && pn === pubkey
);
}
}
interface Props {
text: string;
renderOther?: RenderTextCallbackType;
}
export class AddMentions extends React.Component<Props> {
public static defaultProps: Partial<Props> = {
renderOther: ({ text }) => text,
};
public render() {
const { text, renderOther } = this.props;
const results: Array<any> = [];
const FIND_MENTIONS = window.pubkeyPattern;
// We have to do this, because renderNonNewLine is not required in our Props object,
// but it is always provided via defaultProps.
if (!renderOther) {
return;
}
let match = FIND_MENTIONS.exec(text);
let last = 0;
let count = 1000;
if (!match) {
return renderOther({ text, key: 0 });
}
while (match) {
if (last < match.index) {
const otherText = text.slice(last, match.index);
results.push(renderOther({ text: otherText, key: count++ }));
}
const pubkey = text.slice(match.index, FIND_MENTIONS.lastIndex);
results.push(<Mention text={pubkey} key={count++} />);
// @ts-ignore
last = FIND_MENTIONS.lastIndex;
match = FIND_MENTIONS.exec(text);
}
if (last < text.length) {
results.push(renderOther({ text: text.slice(last), key: count++ }));
}
return results;
}
}

@ -56,15 +56,17 @@ interface Props {
/** Allows you to customize now non-newlines are rendered. Simplest is just a <span>. */
renderNonEmoji?: RenderTextCallbackType;
i18n: LocalizerType;
isPublic?: boolean;
}
export class Emojify extends React.Component<Props> {
public static defaultProps: Partial<Props> = {
renderNonEmoji: ({ text }) => text || '',
isPublic: false,
};
public render() {
const { text, sizeClass, renderNonEmoji, i18n } = this.props;
const { text, sizeClass, renderNonEmoji, i18n, isPublic } = this.props;
const results: Array<any> = [];
const regex = getRegex();
@ -79,13 +81,15 @@ export class Emojify extends React.Component<Props> {
let count = 1;
if (!match) {
return renderNonEmoji({ text, key: 0 });
return renderNonEmoji({ text, key: 0, isPublic });
}
while (match) {
if (last < match.index) {
const textWithNoEmoji = text.slice(last, match.index);
results.push(renderNonEmoji({ text: textWithNoEmoji, key: count++ }));
results.push(
renderNonEmoji({ text: textWithNoEmoji, key: count++, isPublic })
);
}
results.push(getImageTag({ match, sizeClass, key: count++, i18n }));
@ -95,7 +99,9 @@ export class Emojify extends React.Component<Props> {
}
if (last < text.length) {
results.push(renderNonEmoji({ text: text.slice(last), key: count++ }));
results.push(
renderNonEmoji({ text: text.slice(last), key: count++, isPublic })
);
}
return results;

@ -32,6 +32,12 @@ import { isFileDangerous } from '../../util/isFileDangerous';
import { ColorType, LocalizerType } from '../../types/Util';
import { ContextMenu, ContextMenuTrigger, MenuItem } from 'react-contextmenu';
declare global {
interface Window {
shortenPubkey: any;
}
}
interface Trigger {
handleContextClick: (event: React.MouseEvent<HTMLDivElement>) => void;
}
@ -322,9 +328,7 @@ export class Message extends React.PureComponent<Props, State> {
return null;
}
const shortenedPubkey = `(...${authorPhoneNumber.substring(
authorPhoneNumber.length - 6
)})`;
const shortenedPubkey = window.shortenPubkey(authorPhoneNumber);
const displayedPubkey = authorProfileName
? shortenedPubkey
@ -585,6 +589,7 @@ export class Message extends React.PureComponent<Props, State> {
direction,
i18n,
quote,
isPublic,
} = this.props;
if (!quote) {
@ -596,9 +601,7 @@ export class Message extends React.PureComponent<Props, State> {
const quoteColor =
direction === 'incoming' ? authorColor : quote.authorColor;
const shortenedPubkey = `(...${quote.authorPhoneNumber.substring(
quote.authorPhoneNumber.length - 6
)})`;
const shortenedPubkey = window.shortenPubkey(quote.authorPhoneNumber);
const displayedPubkey = quote.authorProfileName
? shortenedPubkey
@ -611,6 +614,7 @@ export class Message extends React.PureComponent<Props, State> {
text={quote.text}
attachment={quote.attachment}
isIncoming={direction === 'incoming'}
isPublic={isPublic}
authorPhoneNumber={displayedPubkey}
authorProfileName={quote.authorProfileName}
authorName={quote.authorName}
@ -741,6 +745,7 @@ export class Message extends React.PureComponent<Props, State> {
isRss={isRss}
i18n={i18n}
textPending={textPending}
isPublic={this.props.isPublic}
/>
</div>
);
@ -1020,8 +1025,21 @@ export class Message extends React.PureComponent<Props, State> {
const width = this.getWidth();
const isShowingImage = this.isShowingImage();
// We parse the message later, but we still need to do an early check
// to see if the message mentions us, so we can display the entire
// message differently
const mentions = this.props.text
? this.props.text.match(window.pubkeyPattern)
: [];
const mentionMe =
mentions &&
mentions.some(m => m.slice(1) === window.lokiPublicChatAPI.ourKey);
const shouldHightlight =
mentionMe && direction === 'incoming' && this.props.isPublic;
const divClass = shouldHightlight ? 'message-highlighted' : '';
return (
<div>
<div className={divClass}>
<ContextMenuTrigger id={rightClickTriggerId}>
<div
className={classNames(

@ -3,6 +3,7 @@ import React from 'react';
import { getSizeClass, SizeClassType } from '../../util/emoji';
import { Emojify } from './Emojify';
import { AddNewLines } from './AddNewLines';
import { AddMentions } from './AddMentions';
import { Linkify } from './Linkify';
import { LocalizerType, RenderTextCallbackType } from '../../types/Util';
@ -15,13 +16,31 @@ interface Props {
disableJumbomoji?: boolean;
/** If set, links will be left alone instead of turned into clickable `<a>` tags. */
disableLinks?: boolean;
isPublic?: boolean;
i18n: LocalizerType;
}
const renderMentions: RenderTextCallbackType = ({ text, key }) => (
<AddMentions key={key} text={text} />
);
const renderDefault: RenderTextCallbackType = ({ text }) => text;
const renderNewLines: RenderTextCallbackType = ({
text: textWithNewLines,
key,
}) => <AddNewLines key={key} text={textWithNewLines} />;
isPublic,
}) => {
const renderOther = isPublic ? renderMentions : renderDefault;
return (
<AddNewLines
key={key}
text={textWithNewLines}
renderNonNewLine={renderOther}
/>
);
};
const renderEmoji = ({
i18n,
@ -29,12 +48,14 @@ const renderEmoji = ({
key,
sizeClass,
renderNonEmoji,
isPublic,
}: {
i18n: LocalizerType;
text: string;
key: number;
sizeClass?: SizeClassType;
renderNonEmoji: RenderTextCallbackType;
isPublic?: boolean;
}) => (
<Emojify
i18n={i18n}
@ -42,6 +63,7 @@ const renderEmoji = ({
text={text}
sizeClass={sizeClass}
renderNonEmoji={renderNonEmoji}
isPublic={isPublic}
/>
);
@ -52,6 +74,10 @@ const renderEmoji = ({
* them for you.
*/
export class MessageBody extends React.Component<Props> {
public static defaultProps: Partial<Props> = {
isPublic: false,
};
public addDownloading(jsx: JSX.Element): JSX.Element {
const { i18n, textPending } = this.props;
@ -76,6 +102,7 @@ export class MessageBody extends React.Component<Props> {
disableLinks,
isRss,
i18n,
isPublic,
} = this.props;
const sizeClass = disableJumbomoji ? undefined : getSizeClass(text);
const textWithPending = textPending ? `${text}...` : text;
@ -88,6 +115,7 @@ export class MessageBody extends React.Component<Props> {
sizeClass,
key: 0,
renderNonEmoji: renderNewLines,
isPublic,
})
);
}
@ -103,6 +131,7 @@ export class MessageBody extends React.Component<Props> {
sizeClass,
key,
renderNonEmoji: renderNewLines,
isPublic,
});
}}
/>

@ -19,6 +19,7 @@ interface Props {
i18n: LocalizerType;
isFromMe: boolean;
isIncoming: boolean;
isPublic?: boolean;
withContentAbove: boolean;
onClick?: () => void;
onClose?: () => void;
@ -214,7 +215,7 @@ export class Quote extends React.Component<Props, State> {
}
public renderText() {
const { i18n, text, attachment, isIncoming } = this.props;
const { i18n, text, attachment, isIncoming, isPublic } = this.props;
if (text) {
return (
@ -225,7 +226,12 @@ export class Quote extends React.Component<Props, State> {
isIncoming ? 'module-quote__primary__text--incoming' : null
)}
>
<MessageBody text={text} disableLinks={true} i18n={i18n} />
<MessageBody
isPublic={isPublic}
text={text}
disableLinks={true}
i18n={i18n}
/>
</div>
);
}

@ -2,6 +2,7 @@ export type RenderTextCallbackType = (
options: {
text: string;
key: number;
isPublic?: boolean;
}
) => JSX.Element | string;

@ -137,6 +137,7 @@
"prefer-type-cast": false,
// We use || and && shortcutting because we're javascript programmers
"strict-boolean-expressions": false,
"no-suspicious-comment": false,
"react-no-dangerous-html": [
true,
{

Loading…
Cancel
Save