2017-09-22 17:52:06 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 New Vector Ltd
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
import EventEmitter from 'events';
|
2017-10-23 15:04:26 +00:00
|
|
|
import { groupMemberFromApiObject, groupRoomFromApiObject } from '../groups';
|
2017-10-23 14:28:38 +00:00
|
|
|
import FlairStore from './FlairStore';
|
2017-11-28 11:54:05 +00:00
|
|
|
import MatrixClientPeg from '../MatrixClientPeg';
|
2017-09-22 17:52:06 +00:00
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
function parseMembersResponse(response) {
|
|
|
|
return response.chunk.map((apiMember) => groupMemberFromApiObject(apiMember));
|
|
|
|
}
|
|
|
|
|
|
|
|
function parseRoomsResponse(response) {
|
|
|
|
return response.chunk.map((apiRoom) => groupRoomFromApiObject(apiRoom));
|
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
/**
|
2017-10-04 15:56:35 +00:00
|
|
|
* Stores the group summary for a room and provides an API to change it and
|
2017-10-04 16:01:44 +00:00
|
|
|
* other useful group APIs that may have an effect on the group summary.
|
2017-09-22 17:52:06 +00:00
|
|
|
*/
|
2017-10-04 15:56:35 +00:00
|
|
|
export default class GroupStore extends EventEmitter {
|
2017-10-31 16:13:13 +00:00
|
|
|
|
|
|
|
static STATE_KEY = {
|
|
|
|
GroupMembers: 'GroupMembers',
|
|
|
|
GroupInvitedMembers: 'GroupInvitedMembers',
|
|
|
|
Summary: 'Summary',
|
|
|
|
GroupRooms: 'GroupRooms',
|
|
|
|
};
|
|
|
|
|
2017-11-28 11:54:05 +00:00
|
|
|
constructor(groupId) {
|
2017-09-22 17:52:06 +00:00
|
|
|
super();
|
2017-11-06 10:18:10 +00:00
|
|
|
if (!groupId) {
|
|
|
|
throw new Error('GroupStore needs a valid groupId to be created');
|
|
|
|
}
|
2017-10-04 16:51:38 +00:00
|
|
|
this.groupId = groupId;
|
2018-01-17 16:59:13 +00:00
|
|
|
this._state = {};
|
|
|
|
this._state[GroupStore.STATE_KEY.Summary] = {};
|
|
|
|
this._state[GroupStore.STATE_KEY.GroupRooms] = [];
|
|
|
|
this._state[GroupStore.STATE_KEY.GroupMembers] = [];
|
|
|
|
this._state[GroupStore.STATE_KEY.GroupInvitedMembers] = [];
|
2017-10-31 11:42:09 +00:00
|
|
|
this._ready = {};
|
2017-10-27 10:36:32 +00:00
|
|
|
|
2018-01-17 17:01:42 +00:00
|
|
|
this._fetchResourcePromise = {};
|
2018-01-17 16:59:13 +00:00
|
|
|
this._resourceFetcher = {
|
|
|
|
[GroupStore.STATE_KEY.Summary]: () => {
|
|
|
|
return MatrixClientPeg.get()
|
|
|
|
.getGroupSummary(this.groupId);
|
|
|
|
},
|
|
|
|
[GroupStore.STATE_KEY.GroupRooms]: () => {
|
|
|
|
return MatrixClientPeg.get()
|
|
|
|
.getGroupRooms(this.groupId)
|
|
|
|
.then(parseRoomsResponse);
|
|
|
|
},
|
|
|
|
[GroupStore.STATE_KEY.GroupMembers]: () => {
|
|
|
|
return MatrixClientPeg.get()
|
|
|
|
.getGroupUsers(this.groupId)
|
|
|
|
.then(parseMembersResponse);
|
|
|
|
},
|
|
|
|
[GroupStore.STATE_KEY.GroupInvitedMembers]: () => {
|
|
|
|
return MatrixClientPeg.get()
|
|
|
|
.getGroupInvitedUsers(this.groupId)
|
|
|
|
.then(parseMembersResponse);
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2017-10-27 10:36:32 +00:00
|
|
|
this.on('error', (err) => {
|
|
|
|
console.error(`GroupStore for ${this.groupId} encountered error`, err);
|
|
|
|
});
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
_fetchResource(stateKey) {
|
2018-01-17 17:01:42 +00:00
|
|
|
// Ongoing request, ignore
|
|
|
|
if (this._fetchResourcePromise[stateKey]) return;
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
const clientPromise = this._resourceFetcher[stateKey]();
|
2018-01-17 17:01:42 +00:00
|
|
|
|
|
|
|
// Indicate ongoing request
|
|
|
|
this._fetchResourcePromise[stateKey] = clientPromise;
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
clientPromise.then((result) => {
|
|
|
|
this._state[stateKey] = result;
|
|
|
|
this._ready[stateKey] = true;
|
2017-10-23 15:04:26 +00:00
|
|
|
this._notifyListeners();
|
|
|
|
}).catch((err) => {
|
2017-10-27 10:37:45 +00:00
|
|
|
// Invited users not visible to non-members
|
2018-01-17 16:59:13 +00:00
|
|
|
if (stateKey === GroupStore.STATE_KEY.GroupInvitedMembers && err.httpStatus === 403) {
|
2017-10-27 10:37:45 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-09-22 17:52:06 +00:00
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
console.error("Failed to get resource " + stateKey + ":" + err);
|
2017-09-22 17:52:06 +00:00
|
|
|
this.emit('error', err);
|
2018-01-17 17:01:42 +00:00
|
|
|
}).finally(() => {
|
|
|
|
// Indicate finished request, allow for future fetches
|
|
|
|
delete this._fetchResourcePromise[stateKey];
|
2017-09-22 17:52:06 +00:00
|
|
|
});
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
return clientPromise;
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
_notifyListeners() {
|
|
|
|
this.emit('update');
|
|
|
|
}
|
|
|
|
|
2018-01-02 18:12:08 +00:00
|
|
|
/**
|
|
|
|
* Register a listener to recieve updates from the store. This also
|
|
|
|
* immediately triggers an update to send the current state of the
|
|
|
|
* store (which could be the initial state).
|
|
|
|
*
|
2018-01-17 16:59:13 +00:00
|
|
|
* This also causes a fetch of all group data, which might cause
|
|
|
|
* 4 separate HTTP requests, but only said requests aren't already
|
|
|
|
* ongoing.
|
2018-01-02 18:12:08 +00:00
|
|
|
*
|
|
|
|
* @param {function} fn the function to call when the store updates.
|
|
|
|
* @return {Object} tok a registration "token" with a single
|
|
|
|
* property `unregister`, a function that can
|
|
|
|
* be called to unregister the listener such
|
|
|
|
* that it won't be called any more.
|
|
|
|
*/
|
2017-10-27 10:36:32 +00:00
|
|
|
registerListener(fn) {
|
|
|
|
this.on('update', fn);
|
2017-10-31 11:42:09 +00:00
|
|
|
// Call to set initial state (before fetching starts)
|
|
|
|
this.emit('update');
|
2018-01-17 16:59:13 +00:00
|
|
|
|
|
|
|
this._fetchResource(GroupStore.STATE_KEY.Summary);
|
|
|
|
this._fetchResource(GroupStore.STATE_KEY.GroupRooms);
|
|
|
|
this._fetchResource(GroupStore.STATE_KEY.GroupMembers);
|
|
|
|
this._fetchResource(GroupStore.STATE_KEY.GroupInvitedMembers);
|
2017-12-15 14:12:21 +00:00
|
|
|
|
2018-01-02 18:12:08 +00:00
|
|
|
// Similar to the Store of flux/utils, we return a "token" that
|
|
|
|
// can be used to unregister the listener.
|
2017-12-15 14:12:21 +00:00
|
|
|
return {
|
|
|
|
unregister: () => {
|
|
|
|
this.unregisterListener(fn);
|
|
|
|
},
|
|
|
|
};
|
2017-10-27 10:36:32 +00:00
|
|
|
}
|
|
|
|
|
2017-10-27 13:33:10 +00:00
|
|
|
unregisterListener(fn) {
|
|
|
|
this.removeListener('update', fn);
|
|
|
|
}
|
|
|
|
|
2017-10-31 11:42:09 +00:00
|
|
|
isStateReady(id) {
|
|
|
|
return this._ready[id];
|
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
getSummary() {
|
2018-01-17 16:59:13 +00:00
|
|
|
return this._state[GroupStore.STATE_KEY.Summary];
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
2017-10-05 13:30:04 +00:00
|
|
|
getGroupRooms() {
|
2018-01-17 16:59:13 +00:00
|
|
|
return this._state[GroupStore.STATE_KEY.GroupRooms];
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
getGroupMembers() {
|
|
|
|
return this._state[GroupStore.STATE_KEY.GroupMembers];
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2018-01-17 16:59:13 +00:00
|
|
|
getGroupInvitedMembers() {
|
|
|
|
return this._state[GroupStore.STATE_KEY.GroupInvitedMembers];
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2017-10-17 15:08:19 +00:00
|
|
|
getGroupPublicity() {
|
2018-01-17 16:59:13 +00:00
|
|
|
return this._state[GroupStore.STATE_KEY.Summary].user ?
|
|
|
|
this._state[GroupStore.STATE_KEY.Summary].user.is_publicised : null;
|
2017-10-17 15:08:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
isUserPrivileged() {
|
2018-01-17 16:59:13 +00:00
|
|
|
return this._state[GroupStore.STATE_KEY.Summary].user ?
|
|
|
|
this._state[GroupStore.STATE_KEY.Summary].user.is_privileged : null;
|
2017-10-17 15:08:19 +00:00
|
|
|
}
|
|
|
|
|
2017-11-02 13:25:55 +00:00
|
|
|
addRoomToGroup(roomId, isPublic) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-11-02 13:25:55 +00:00
|
|
|
.addRoomToGroup(this.groupId, roomId, isPublic)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupRooms));
|
2017-11-02 13:25:55 +00:00
|
|
|
}
|
|
|
|
|
2017-11-09 16:28:21 +00:00
|
|
|
updateGroupRoomVisibility(roomId, isPublic) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-11-09 16:28:21 +00:00
|
|
|
.updateGroupRoomVisibility(this.groupId, roomId, isPublic)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupRooms));
|
2017-10-05 13:30:04 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
removeRoomFromGroup(roomId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-05 13:30:04 +00:00
|
|
|
.removeRoomFromGroup(this.groupId, roomId)
|
|
|
|
// Room might be in the summary, refresh just in case
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary))
|
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupRooms));
|
2017-10-04 15:56:35 +00:00
|
|
|
}
|
|
|
|
|
2017-10-23 15:04:26 +00:00
|
|
|
inviteUserToGroup(userId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get().inviteUserToGroup(this.groupId, userId)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupInvitedMembers));
|
2017-10-23 15:04:26 +00:00
|
|
|
}
|
|
|
|
|
2017-11-07 18:51:41 +00:00
|
|
|
acceptGroupInvite() {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get().acceptGroupInvite(this.groupId)
|
2017-11-07 18:51:41 +00:00
|
|
|
// The user might be able to see more rooms now
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupRooms))
|
2017-11-08 11:52:52 +00:00
|
|
|
// The user should now appear as a member
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupMembers))
|
|
|
|
// The user should now not appear as an invited member
|
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.GroupInvitedMembers));
|
2017-11-07 18:51:41 +00:00
|
|
|
}
|
|
|
|
|
2017-09-22 17:52:06 +00:00
|
|
|
addRoomToGroupSummary(roomId, categoryId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-04 16:51:38 +00:00
|
|
|
.addRoomToGroupSummary(this.groupId, roomId, categoryId)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
addUserToGroupSummary(userId, roleId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-04 16:51:38 +00:00
|
|
|
.addUserToGroupSummary(this.groupId, userId, roleId)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
removeRoomFromGroupSummary(roomId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-04 16:51:38 +00:00
|
|
|
.removeRoomFromGroupSummary(this.groupId, roomId)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
removeUserFromGroupSummary(userId) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-04 16:51:38 +00:00
|
|
|
.removeUserFromGroupSummary(this.groupId, userId)
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary));
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|
2017-09-26 13:46:57 +00:00
|
|
|
|
2017-09-26 13:58:49 +00:00
|
|
|
setGroupPublicity(isPublished) {
|
2017-11-28 11:54:05 +00:00
|
|
|
return MatrixClientPeg.get()
|
2017-10-04 16:51:38 +00:00
|
|
|
.setGroupPublicity(this.groupId, isPublished)
|
2017-11-28 11:54:05 +00:00
|
|
|
.then(() => { FlairStore.invalidatePublicisedGroups(MatrixClientPeg.get().credentials.userId); })
|
2018-01-17 16:59:13 +00:00
|
|
|
.then(this._fetchResource.bind(this, GroupStore.STATE_KEY.Summary));
|
2017-09-26 13:46:57 +00:00
|
|
|
}
|
2017-09-22 17:52:06 +00:00
|
|
|
}
|