Merge pull request #690 from neuroscr/multidevice-publicchat

Add/Remove Moderators interface
pull/693/head
Ryan Tharp 5 years ago committed by GitHub
commit 2cf39cc1ed
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23

@ -2245,6 +2245,18 @@
"inviteFriends": {
"message": "Invite Friends"
},
"manageModerators": {
"message": "Manage Moderators"
},
"addModerators": {
"message": "Add Moderators"
},
"removeModerators": {
"message": "Remove Moderators"
},
"add": {
"message": "Add"
},
"groupInvitation": {
"message": "Group Invitation"
},
@ -2254,6 +2266,9 @@
"noFriendsToAdd": {
"message": "no friends to add"
},
"noModeratorsToRemove": {
"message": "no moderators to remove"
},
"couldNotDecryptMessage": {
"message": "Couldn't decrypt a message"
},

@ -822,6 +822,8 @@
<script type='text/javascript' src='js/views/confirm_session_reset_view.js'></script>
<script type='text/javascript' src='js/views/edit_profile_dialog_view.js'></script>
<script type='text/javascript' src='js/views/invite_friends_dialog_view.js'></script>
<script type='text/javascript' src='js/views/moderators_add_dialog_view.js'></script>
<script type='text/javascript' src='js/views/moderators_remove_dialog_view.js'></script>
<script type='text/javascript' src='js/views/user_details_dialog_view.js'></script>
<script type='text/javascript' src='js/wall_clock_listener.js'></script>

@ -851,6 +851,18 @@
}
});
Whisper.events.on('addModerators', async groupConvo => {
if (appView) {
appView.showAddModeratorsDialog(groupConvo);
}
});
Whisper.events.on('removeModerators', async groupConvo => {
if (appView) {
appView.showRemoveModeratorsDialog(groupConvo);
}
});
Whisper.events.on(
'publicChatInvitationAccepted',
async (serverAddress, channelId) => {

@ -329,7 +329,7 @@ class LokiAppDotNetServerAPI {
// if it's a response style with a meta
if (result.status !== 200) {
if (!forceFreshToken && response.meta.code === 401) {
if (!forceFreshToken && (!response.meta || response.meta.code === 401)) {
// copy options because lint complains if we modify this directly
const updatedOptions = options;
// force it this time
@ -371,6 +371,60 @@ class LokiAppDotNetServerAPI {
return res.response.data.annotations || [];
}
async getModerators(channelId) {
if (!channelId) {
log.warn('No channelId provided to getModerators!');
return [];
}
const res = await this.serverRequest(
`loki/v1/channels/${channelId}/moderators`
);
return (!res.err && res.response && res.response.moderators) || [];
}
async addModerators(pubKeysParam) {
let pubKeys = pubKeysParam;
if (!Array.isArray(pubKeys)) {
pubKeys = [pubKeys];
}
pubKeys = pubKeys.map(key => `@${key}`);
const users = await this.getUsers(pubKeys);
const validUsers = users.filter(user => !!user.id);
const results = await Promise.all(
validUsers.map(async user => {
log.info(`POSTing loki/v1/moderators/${user.id}`);
const res = await this.serverRequest(`loki/v1/moderators/${user.id}`, {
method: 'POST',
});
return !!(!res.err && res.response && res.response.data);
})
);
const anyFailures = results.some(test => !test);
return anyFailures ? results : true; // return failures or total success
}
async removeModerators(pubKeysParam) {
let pubKeys = pubKeysParam;
if (!Array.isArray(pubKeys)) {
pubKeys = [pubKeys];
}
pubKeys = pubKeys.map(key => `@${key}`);
const users = await this.getUsers(pubKeys);
const validUsers = users.filter(user => !!user.id);
const results = await Promise.all(
validUsers.map(async user => {
const res = await this.serverRequest(`loki/v1/moderators/${user.id}`, {
method: 'DELETE',
});
return !!(!res.err && res.response && res.response.data);
})
);
const anyFailures = results.some(test => !test);
return anyFailures ? results : true; // return failures or total success
}
async getSubscribers(channelId, wantObjects) {
if (!channelId) {
log.warn('No channelId provided to getSubscribers!');
@ -645,6 +699,10 @@ class LokiPublicChannelAPI {
return this.serverAPI.getSubscribers(this.channelId, true);
}
getModerators() {
return this.serverAPI.getModerators(this.channelId);
}
// get moderation actions
async pollForModerators() {
try {
@ -1287,6 +1345,14 @@ class LokiPublicChannelAPI {
// look up primary device once
const primaryPubKey = slavePrimaryMap[slaveKey];
if (!Array.isArray(slaveMessages[slaveKey])) {
log.warn(
`messages for ${slaveKey} is not an array`,
slaveMessages[slaveKey]
);
return;
}
// send out remaining messages for this merged identity
slaveMessages[slaveKey].forEach(messageDataP => {
const messageData = messageDataP; // for linter

@ -57,6 +57,14 @@ const {
const {
InviteFriendsDialog,
} = require('../../ts/components/conversation/InviteFriendsDialog');
const {
AddModeratorsDialog,
} = require('../../ts/components/conversation/ModeratorsAddDialog');
const {
RemoveModeratorsDialog,
} = require('../../ts/components/conversation/ModeratorsRemoveDialog');
const {
GroupInvitation,
} = require('../../ts/components/conversation/GroupInvitation');
@ -242,6 +250,8 @@ exports.setup = (options = {}) => {
ConfirmDialog,
UpdateGroupDialog,
InviteFriendsDialog,
AddModeratorsDialog,
RemoveModeratorsDialog,
GroupInvitation,
BulkEdit,
MediaGallery,

@ -266,5 +266,13 @@
const dialog = new Whisper.InviteFriendsDialogView(groupConvo);
this.el.append(dialog.el);
},
showAddModeratorsDialog(groupConvo) {
const dialog = new Whisper.AddModeratorsDialogView(groupConvo);
this.el.append(dialog.el);
},
showRemoveModeratorsDialog(groupConvo) {
const dialog = new Whisper.RemoveModeratorsDialogView(groupConvo);
this.el.append(dialog.el);
},
});
})();

@ -299,6 +299,14 @@
window.Whisper.events.trigger('inviteFriends', this.model);
},
onAddModerators: () => {
window.Whisper.events.trigger('addModerators', this.model);
},
onRemoveModerators: () => {
window.Whisper.events.trigger('removeModerators', this.model);
},
onShowUserDetails: pubkey => {
if (this.model.isPrivate()) {
window.Whisper.events.trigger('onShowUserDetails', {

@ -0,0 +1,66 @@
/* global Whisper, log */
// eslint-disable-next-line func-names
(function() {
'use strict';
window.Whisper = window.Whisper || {};
Whisper.AddModeratorsDialogView = Whisper.View.extend({
className: 'loki-dialog modal',
async initialize(convo) {
this.close = this.close.bind(this);
this.onSubmit = this.onSubmit.bind(this);
this.chatName = convo.get('name');
this.chatServer = convo.get('server');
this.channelId = convo.get('channelId');
// get current list of moderators
this.channelAPI = await convo.getPublicSendData();
const modPubKeys = await this.channelAPI.getModerators();
const convos = window.getConversations().models;
// private friends (not you) that aren't already moderators
const friends = convos.filter(
d =>
!!d &&
d.isFriend() &&
d.isPrivate() &&
!d.isMe() &&
!modPubKeys.includes(d.id)
);
this.friends = friends;
this.$el.focus();
this.render();
},
render() {
const view = new Whisper.ReactWrapperView({
className: 'add-moderators-dialog',
Component: window.Signal.Components.AddModeratorsDialog,
props: {
friendList: this.friends,
chatName: this.chatName,
onSubmit: this.onSubmit,
onClose: this.close,
},
});
this.$el.append(view.el);
return this;
},
close() {
this.remove();
},
async onSubmit(pubKeys) {
log.info(`asked to add ${pubKeys}`);
const res = await this.channelAPI.serverAPI.addModerators(pubKeys);
if (res !== true) {
// we have errors, deal with them...
// how?
}
},
});
})();

@ -0,0 +1,64 @@
/* global Whisper */
// eslint-disable-next-line func-names
(function() {
'use strict';
window.Whisper = window.Whisper || {};
Whisper.RemoveModeratorsDialogView = Whisper.View.extend({
className: 'loki-dialog modal',
async initialize(convo) {
this.close = this.close.bind(this);
this.onSubmit = this.onSubmit.bind(this);
this.chatName = convo.get('name');
this.chatServer = convo.get('server');
this.channelId = convo.get('channelId');
// get current list of moderators
this.channelAPI = await convo.getPublicSendData();
const modPubKeys = await this.channelAPI.getModerators();
const convos = window.getConversations().models;
const moderators = modPubKeys
.map(
pubKey =>
convos.find(c => c.id === pubKey) || {
id: pubKey, // memberList need a key
authorPhoneNumber: pubKey,
}
)
.filter(c => !!c);
this.mods = moderators;
this.$el.focus();
this.render();
},
render() {
const view = new Whisper.ReactWrapperView({
className: 'remove-moderators-dialog',
Component: window.Signal.Components.RemoveModeratorsDialog,
props: {
modList: this.mods,
onSubmit: this.onSubmit,
onClose: this.close,
chatName: this.chatName,
},
});
this.$el.append(view.el);
return this;
},
close() {
this.remove();
},
async onSubmit(pubKeys) {
const res = await this.channelAPI.serverAPI.removeModerators(pubKeys);
if (res !== true) {
// we have errors, deal with them...
// how?
}
},
});
})();

@ -193,11 +193,8 @@ MessageSender.prototype = {
constructor: MessageSender,
// makeAttachmentPointer :: Attachment -> Promise AttachmentPointerProto
async makeAttachmentPointer(
attachment,
publicServer = null,
{ isRaw = false, isAvatar = false }
) {
async makeAttachmentPointer(attachment, publicServer = null, options = {}) {
const { isRaw = false, isAvatar = false } = options;
if (typeof attachment !== 'object' || attachment == null) {
return Promise.resolve(undefined);
}

@ -30,6 +30,8 @@
}
.invite-friends-dialog,
.add-moderators-dialog,
.remove-moderators-dialog,
.create-group-dialog {
.content {
max-width: 100% !important;
@ -50,6 +52,8 @@
}
.create-group-dialog,
.add-moderators-dialog,
.remove-moderators-dialog,
.invite-friends-dialog {
.no-friends {
text-align: center;
@ -61,6 +65,8 @@
}
.create-group-dialog,
.add-moderators-dialog,
.remove-moderators-dialog,
.edit-profile-dialog {
.error-message {
text-align: center;
@ -129,6 +135,8 @@
.member-list-container,
.create-group-dialog,
.add-moderators-dialog,
.remove-moderators-dialog,
.invite-friends-dialog {
.member-item {
padding: 4px;
@ -182,6 +190,8 @@
.dark-theme {
.member-list-container,
.create-group-dialog,
.add-moderators-dialog,
.remove-moderators-dialog,
.invite-friends-dialog {
.member-item {
&:hover:not(.member-selected) {
@ -203,6 +213,12 @@
}
}
.add-moderators-dialog {
.module-main-header__search__input {
color: rgb(32, 32, 32);
}
}
.module-conversation-list-item--mentioned-us {
border-left: 4px solid #ffb000 !important;
}

@ -67,8 +67,10 @@ interface Props {
onUpdateGroup: () => void;
onLeaveGroup: () => void;
onAddModerators: () => void;
onRemoveModerators: () => void;
onInviteFriends: () => void;
onShowUserDetails?: (userPubKey: string) => void;
i18n: LocalizerType;
@ -240,6 +242,8 @@ export class ConversationHeader extends React.Component<Props> {
onCopyPublicKey,
onUpdateGroup,
onLeaveGroup,
onAddModerators,
onRemoveModerators,
onInviteFriends,
} = this.props;
@ -255,6 +259,14 @@ export class ConversationHeader extends React.Component<Props> {
{isPrivateGroup || amMod ? (
<MenuItem onClick={onUpdateGroup}>{i18n('updateGroup')}</MenuItem>
) : null}
{amMod ? (
<MenuItem onClick={onAddModerators}>{i18n('addModerators')}</MenuItem>
) : null}
{amMod ? (
<MenuItem onClick={onRemoveModerators}>
{i18n('removeModerators')}
</MenuItem>
) : null}
{isPrivateGroup ? (
<MenuItem onClick={onLeaveGroup}>{i18n('leaveGroup')}</MenuItem>
) : null}

@ -73,7 +73,7 @@ export class InviteFriendsDialog extends React.Component<Props, State> {
/>
</div>
{hasFriends ? null : (
<p className="no-friends">`(${window.i18n('noFriendsToAdd')})`</p>
<p className="no-friends">{window.i18n('noFriendsToAdd')}</p>
)}
<div className="buttons">
<button className="cancel" tabIndex={0} onClick={this.closeDialog}>

@ -0,0 +1,222 @@
import React from 'react';
import { Contact, MemberList } from './MemberList';
import { cleanSearchTerm } from '../../util/cleanSearchTerm';
interface Props {
friendList: Array<any>;
chatName: string;
onSubmit: any;
onClose: any;
}
declare global {
interface Window {
i18n: any;
}
}
interface State {
friendList: Array<Contact>;
inputBoxValue: string;
}
export class AddModeratorsDialog extends React.Component<Props, State> {
private readonly updateSearchBound: (
event: React.FormEvent<HTMLInputElement>
) => void;
private readonly inputRef: React.RefObject<HTMLInputElement>;
constructor(props: any) {
super(props);
this.updateSearchBound = this.updateSearch.bind(this);
this.onMemberClicked = this.onMemberClicked.bind(this);
this.add = this.add.bind(this);
this.closeDialog = this.closeDialog.bind(this);
this.onClickOK = this.onClickOK.bind(this);
this.onKeyUp = this.onKeyUp.bind(this);
this.inputRef = React.createRef();
let friends = this.props.friendList;
friends = friends.map(d => {
const lokiProfile = d.getLokiProfile();
const name = lokiProfile ? lokiProfile.displayName : 'Anonymous';
// TODO: should take existing members into account
const existingMember = false;
return {
id: d.id,
authorPhoneNumber: d.id,
authorProfileName: name,
selected: false,
authorName: name,
authorColor: d.getColor(),
checkmarked: false,
existingMember,
};
});
this.state = {
friendList: friends,
inputBoxValue: '',
};
window.addEventListener('keyup', this.onKeyUp);
}
public updateSearch(event: React.FormEvent<HTMLInputElement>) {
const searchTerm = event.currentTarget.value;
const cleanedTerm = cleanSearchTerm(searchTerm);
if (!cleanedTerm) {
return;
}
this.setState(state => {
return {
...state,
inputBoxValue: searchTerm,
};
});
}
public add() {
// if we have valid data
if (this.state.inputBoxValue.length > 64) {
const weHave = this.state.friendList.some(
user => user.authorPhoneNumber === this.state.inputBoxValue
);
if (!weHave) {
// lookup to verify it's registered?
// convert pubKey into local object...
const friends = this.state.friendList;
friends.push({
id: this.state.inputBoxValue,
authorPhoneNumber: this.state.inputBoxValue,
authorProfileName: this.state.inputBoxValue,
authorAvatarPath: '',
selected: true,
authorName: this.state.inputBoxValue,
authorColor: '#000000',
checkmarked: true,
existingMember: false,
});
this.setState(state => {
return {
...state,
friendList: friends,
};
});
}
//
}
// clear
if (this.inputRef.current) {
this.inputRef.current.value = '';
}
this.setState(state => {
return {
...state,
inputBoxValue: '',
};
});
}
public render() {
const i18n = window.i18n;
const hasFriends = this.state.friendList.length !== 0;
return (
<div className="content">
<p className="titleText">
${i18n('addModerators')} <span>${this.props.chatName}</span>
</p>
<div className="addModeratorBox">
<p>Add Moderator:</p>
<input
type="text"
ref={this.inputRef}
className="module-main-header__search__input"
placeholder={i18n('search')}
dir="auto"
onChange={this.updateSearchBound}
/>
<button className="add" tabIndex={0} onClick={this.add}>
{i18n('add')}
</button>
</div>
<div className="moderatorList">
<p>From friends:</p>
<div className="friend-selection-list">
<MemberList
members={this.state.friendList}
selected={{}}
i18n={i18n}
onMemberClicked={this.onMemberClicked}
/>
</div>
{hasFriends ? null : (
<p className="no-friends">{i18n('noFriendsToAdd')}</p>
)}
</div>
<div className="buttons">
<button className="cancel" tabIndex={0} onClick={this.closeDialog}>
{i18n('cancel')}
</button>
<button className="ok" tabIndex={0} onClick={this.onClickOK}>
{i18n('ok')}
</button>
</div>
</div>
);
}
private onClickOK() {
this.add(); // process inputBox
const selectedFriends = this.state.friendList
.filter(d => d.checkmarked)
.map(d => d.id);
if (selectedFriends.length > 0) {
this.props.onSubmit(selectedFriends);
}
this.closeDialog();
}
private onKeyUp(event: any) {
switch (event.key) {
case 'Enter':
this.onClickOK();
break;
case 'Esc':
case 'Escape':
this.closeDialog();
break;
default:
}
}
private closeDialog() {
window.removeEventListener('keyup', this.onKeyUp);
this.props.onClose();
}
private onMemberClicked(selected: any) {
const updatedFriends = this.state.friendList.map(member => {
if (member.id === selected.id) {
return { ...member, checkmarked: !member.checkmarked };
} else {
return member;
}
});
this.setState(state => {
return {
...state,
friendList: updatedFriends,
};
});
}
}

@ -0,0 +1,141 @@
import React from 'react';
import { Contact, MemberList } from './MemberList';
interface Props {
modList: Array<any>;
chatName: string;
onSubmit: any;
onClose: any;
}
declare global {
interface Window {
i18n: any;
}
}
interface State {
modList: Array<Contact>;
}
export class RemoveModeratorsDialog extends React.Component<Props, State> {
constructor(props: any) {
super(props);
this.onModClicked = this.onModClicked.bind(this);
this.closeDialog = this.closeDialog.bind(this);
this.onClickOK = this.onClickOK.bind(this);
this.onKeyUp = this.onKeyUp.bind(this);
let mods = this.props.modList;
mods = mods.map(d => {
let name = '';
if (d.getLokiProfile) {
const lokiProfile = d.getLokiProfile();
name = lokiProfile ? lokiProfile.displayName : 'Anonymous';
}
const authorColor = d.getColor ? d.getColor() : '#000000';
// TODO: should take existing members into account
const existingMember = false;
return {
id: d.id,
authorPhoneNumber: d.id,
authorProfileName: name,
selected: false,
authorName: name,
authorColor,
checkmarked: true,
existingMember,
};
});
this.state = {
modList: mods,
};
window.addEventListener('keyup', this.onKeyUp);
}
public render() {
const i18n = window.i18n;
const hasMods = this.state.modList.length !== 0;
return (
<div className="content">
<p className="titleText">
${i18n('removeModerators')} <span>${this.props.chatName}</span>
</p>
<div className="moderatorList">
<p>Existing moderators:</p>
<div className="friend-selection-list">
<MemberList
members={this.state.modList}
selected={{}}
i18n={i18n}
onMemberClicked={this.onModClicked}
/>
</div>
{hasMods ? null : (
<p className="no-friends">{i18n('noModeratorsToRemove')}</p>
)}
</div>
<div className="buttons">
<button className="cancel" tabIndex={0} onClick={this.closeDialog}>
{i18n('cancel')}
</button>
<button className="ok" tabIndex={0} onClick={this.onClickOK}>
{i18n('ok')}
</button>
</div>
</div>
);
}
private onClickOK() {
const removedMods = this.state.modList
.filter(d => !d.checkmarked)
.map(d => d.id);
if (removedMods.length > 0) {
this.props.onSubmit(removedMods);
}
this.closeDialog();
}
private onKeyUp(event: any) {
switch (event.key) {
case 'Enter':
this.onClickOK();
break;
case 'Esc':
case 'Escape':
this.closeDialog();
break;
default:
}
}
private closeDialog() {
window.removeEventListener('keyup', this.onKeyUp);
this.props.onClose();
}
private onModClicked(selected: any) {
const updatedFriends = this.state.modList.map(member => {
if (member.id === selected.id) {
return { ...member, checkmarked: !member.checkmarked };
} else {
return member;
}
});
this.setState(state => {
return {
...state,
modList: updatedFriends,
};
});
}
}
Loading…
Cancel
Save