153 lines
6.4 KiB
TypeScript
153 lines
6.4 KiB
TypeScript
/*
|
|
Copyright 2018 New Vector Ltd
|
|
Copyright 2020 The Matrix.org Foundation C.I.C.
|
|
|
|
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 { asyncAction } from './actionCreators';
|
|
import Modal from '../Modal';
|
|
import * as Rooms from '../Rooms';
|
|
import { _t } from '../languageHandler';
|
|
import * as sdk from '../index';
|
|
import { MatrixClient } from "matrix-js-sdk/src/client";
|
|
import { Room } from "matrix-js-sdk/src/models/room";
|
|
import { AsyncActionPayload } from "../dispatcher/payloads";
|
|
import RoomListStore from "../stores/room-list/RoomListStore";
|
|
import { SortAlgorithm } from "../stores/room-list/algorithms/models";
|
|
import { DefaultTagID } from "../stores/room-list/models";
|
|
|
|
export default class RoomListActions {
|
|
/**
|
|
* Creates an action thunk that will do an asynchronous request to
|
|
* tag room.
|
|
*
|
|
* @param {MatrixClient} matrixClient the matrix client to set the
|
|
* account data on.
|
|
* @param {Room} room the room to tag.
|
|
* @param {string} oldTag the tag to remove (unless oldTag ==== newTag)
|
|
* @param {string} newTag the tag with which to tag the room.
|
|
* @param {?number} oldIndex the previous position of the room in the
|
|
* list of rooms.
|
|
* @param {?number} newIndex the new position of the room in the list
|
|
* of rooms.
|
|
* @returns {AsyncActionPayload} an async action payload
|
|
* @see asyncAction
|
|
*/
|
|
public static tagRoom(
|
|
matrixClient: MatrixClient, room: Room,
|
|
oldTag: string, newTag: string,
|
|
oldIndex: number | null, newIndex: number | null,
|
|
): AsyncActionPayload {
|
|
let metaData = null;
|
|
|
|
// Is the tag ordered manually?
|
|
const store = RoomListStore.instance;
|
|
if (newTag && store.getTagSorting(newTag) === SortAlgorithm.Manual) {
|
|
const newList = [...store.orderedLists[newTag]];
|
|
|
|
newList.sort((a, b) => a.tags[newTag].order - b.tags[newTag].order);
|
|
|
|
// If the room was moved "down" (increasing index) in the same list we
|
|
// need to use the orders of the tiles with indices shifted by +1
|
|
const offset = (
|
|
newTag === oldTag && oldIndex < newIndex
|
|
) ? 1 : 0;
|
|
|
|
const indexBefore = offset + newIndex - 1;
|
|
const indexAfter = offset + newIndex;
|
|
|
|
const prevOrder = indexBefore <= 0 ?
|
|
0 : newList[indexBefore].tags[newTag].order;
|
|
const nextOrder = indexAfter >= newList.length ?
|
|
1 : newList[indexAfter].tags[newTag].order;
|
|
|
|
metaData = {
|
|
order: (prevOrder + nextOrder) / 2.0,
|
|
};
|
|
}
|
|
|
|
return asyncAction('RoomListActions.tagRoom', () => {
|
|
const promises = [];
|
|
const roomId = room.roomId;
|
|
|
|
// Evil hack to get DMs behaving
|
|
if ((oldTag === undefined && newTag === DefaultTagID.DM) ||
|
|
(oldTag === DefaultTagID.DM && newTag === undefined)
|
|
) {
|
|
return Rooms.guessAndSetDMRoom(
|
|
room, newTag === DefaultTagID.DM,
|
|
).catch((err) => {
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
console.error("Failed to set direct chat tag " + err);
|
|
Modal.createTrackedDialog('Failed to set direct chat tag', '', ErrorDialog, {
|
|
title: _t('Failed to set direct chat tag'),
|
|
description: ((err && err.message) ? err.message : _t('Operation failed')),
|
|
});
|
|
});
|
|
}
|
|
|
|
const hasChangedSubLists = oldTag !== newTag;
|
|
|
|
// More evilness: We will still be dealing with moving to favourites/low prio,
|
|
// but we avoid ever doing a request with TAG_DM.
|
|
//
|
|
// if we moved lists, remove the old tag
|
|
if (oldTag && oldTag !== DefaultTagID.DM &&
|
|
hasChangedSubLists
|
|
) {
|
|
const promiseToDelete = matrixClient.deleteRoomTag(
|
|
roomId, oldTag,
|
|
).catch(function(err) {
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
console.error("Failed to remove tag " + oldTag + " from room: " + err);
|
|
Modal.createTrackedDialog('Failed to remove tag from room', '', ErrorDialog, {
|
|
title: _t('Failed to remove tag %(tagName)s from room', {tagName: oldTag}),
|
|
description: ((err && err.message) ? err.message : _t('Operation failed')),
|
|
});
|
|
});
|
|
|
|
promises.push(promiseToDelete);
|
|
}
|
|
|
|
// if we moved lists or the ordering changed, add the new tag
|
|
if (newTag && newTag !== DefaultTagID.DM &&
|
|
(hasChangedSubLists || metaData)
|
|
) {
|
|
// metaData is the body of the PUT to set the tag, so it must
|
|
// at least be an empty object.
|
|
metaData = metaData || {};
|
|
|
|
const promiseToAdd = matrixClient.setRoomTag(roomId, newTag, metaData).catch(function(err) {
|
|
const ErrorDialog = sdk.getComponent("dialogs.ErrorDialog");
|
|
console.error("Failed to add tag " + newTag + " to room: " + err);
|
|
Modal.createTrackedDialog('Failed to add tag to room', '', ErrorDialog, {
|
|
title: _t('Failed to add tag %(tagName)s to room', {tagName: newTag}),
|
|
description: ((err && err.message) ? err.message : _t('Operation failed')),
|
|
});
|
|
|
|
throw err;
|
|
});
|
|
|
|
promises.push(promiseToAdd);
|
|
}
|
|
|
|
return Promise.all(promises);
|
|
}, () => {
|
|
// For an optimistic update
|
|
return {
|
|
room, oldTag, newTag, metaData,
|
|
};
|
|
});
|
|
}
|
|
}
|