diff --git a/CHANGELOG.md b/CHANGELOG.md
index c27429be7f..c6b8e22f01 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -1,3 +1,35 @@
+Changes in [3.52.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.52.0) (2022-08-16)
+=====================================================================================================
+
+## ✨ Features
+ * Device manager - New device tile info design ([\#9122](https://github.com/matrix-org/matrix-react-sdk/pull/9122)). Contributed by @kerryarchibald.
+ * Device manager generic settings subsection component ([\#9147](https://github.com/matrix-org/matrix-react-sdk/pull/9147)). Contributed by @kerryarchibald.
+ * Migrate the hidden read receipts flag to new "send read receipts" option ([\#9141](https://github.com/matrix-org/matrix-react-sdk/pull/9141)).
+ * Live location sharing - share location at most every 5 seconds ([\#9148](https://github.com/matrix-org/matrix-react-sdk/pull/9148)). Contributed by @kerryarchibald.
+ * Increase max length of voice messages to 15m ([\#9133](https://github.com/matrix-org/matrix-react-sdk/pull/9133)). Fixes vector-im/element-web#18620.
+ * Move pin drop out of labs ([\#9135](https://github.com/matrix-org/matrix-react-sdk/pull/9135)).
+ * Start DM on first message ([\#8612](https://github.com/matrix-org/matrix-react-sdk/pull/8612)). Fixes vector-im/element-web#14736.
+ * Remove "Add Space" button from RoomListHeader when user cannot create spaces ([\#9129](https://github.com/matrix-org/matrix-react-sdk/pull/9129)).
+ * The Welcome Home Screen: Dedicated Download Apps Dialog ([\#9120](https://github.com/matrix-org/matrix-react-sdk/pull/9120)). Fixes vector-im/element-web#22921. Contributed by @justjanne.
+ * The Welcome Home Screen: "Submit Feedback" pane ([\#9090](https://github.com/matrix-org/matrix-react-sdk/pull/9090)). Fixes vector-im/element-web#22918. Contributed by @justjanne.
+ * New User Onboarding Task List ([\#9083](https://github.com/matrix-org/matrix-react-sdk/pull/9083)). Fixes vector-im/element-web#22919. Contributed by @justjanne.
+ * Add support for disabling spell checking ([\#8604](https://github.com/matrix-org/matrix-react-sdk/pull/8604)). Fixes vector-im/element-web#21901.
+ * Live location share - leave maximised map open when beacons expire ([\#9098](https://github.com/matrix-org/matrix-react-sdk/pull/9098)). Contributed by @kerryarchibald.
+
+## 🐛 Bug Fixes
+ * Some slash-commands (`/myroomnick`) have temporarily been disabled before the first message in a DM is sent. ([\#9193](https://github.com/matrix-org/matrix-react-sdk/pull/9193)).
+ * Use stable reference for active tab in tabbedView ([\#9145](https://github.com/matrix-org/matrix-react-sdk/pull/9145)). Contributed by @kerryarchibald.
+ * Fix pillification sometimes doubling up ([\#9152](https://github.com/matrix-org/matrix-react-sdk/pull/9152)). Fixes vector-im/element-web#23036.
+ * Fix composer padding ([\#9137](https://github.com/matrix-org/matrix-react-sdk/pull/9137)). Fixes vector-im/element-web#22992.
+ * Fix highlights not being applied to plaintext messages ([\#9126](https://github.com/matrix-org/matrix-react-sdk/pull/9126)). Fixes vector-im/element-web#22787.
+ * Fix dismissing edit composer when change was undone ([\#9109](https://github.com/matrix-org/matrix-react-sdk/pull/9109)). Fixes vector-im/element-web#22932.
+ * 1-to-1 DM rooms with bots now act like DM rooms instead of multi-user-rooms before ([\#9124](https://github.com/matrix-org/matrix-react-sdk/pull/9124)). Fixes vector-im/element-web#22894.
+ * Apply inline start padding to selected lines on modern layout only ([\#9006](https://github.com/matrix-org/matrix-react-sdk/pull/9006)). Fixes vector-im/element-web#22768. Contributed by @luixxiul.
+ * Peek into world-readable rooms from spotlight ([\#9115](https://github.com/matrix-org/matrix-react-sdk/pull/9115)). Fixes vector-im/element-web#22862.
+ * Use default styling on nested numbered lists due to MD being sensitive ([\#9110](https://github.com/matrix-org/matrix-react-sdk/pull/9110)). Fixes vector-im/element-web#22935.
+ * Fix replying using chat effect commands ([\#9101](https://github.com/matrix-org/matrix-react-sdk/pull/9101)). Fixes vector-im/element-web#22824.
+ * The first message in a DM can no longer be a sticker. This has been changed to avoid issues with the integration manager. ([\#9180](https://github.com/matrix-org/matrix-react-sdk/pull/9180)).
+
Changes in [3.51.0](https://github.com/matrix-org/matrix-react-sdk/releases/tag/v3.51.0) (2022-08-02)
=====================================================================================================
diff --git a/cypress/e2e/spaces/spaces.spec.ts b/cypress/e2e/spaces/spaces.spec.ts
index 0a8212ab8d..e7767de942 100644
--- a/cypress/e2e/spaces/spaces.spec.ts
+++ b/cypress/e2e/spaces/spaces.spec.ts
@@ -237,4 +237,42 @@ describe("Spaces", () => {
cy.contains(".mx_SpaceHierarchy_roomTile", "Gaming").should("exist");
});
});
+
+ it("should render subspaces in the space panel only when expanded", () => {
+ cy.injectAxe();
+
+ cy.createSpace({
+ name: "Child Space",
+ initial_state: [],
+ }).then(spaceId => {
+ cy.createSpace({
+ name: "Root Space",
+ initial_state: [
+ spaceChildInitialState(spaceId),
+ ],
+ }).as("spaceId");
+ });
+ cy.get('.mx_SpacePanel .mx_SpaceButton[aria-label="Root Space"]').should("exist");
+ cy.get('.mx_SpacePanel .mx_SpaceButton[aria-label="Child Space"]').should("not.exist");
+
+ const axeOptions = {
+ rules: {
+ // Disable this check as it triggers on nested roving tab index elements which are in practice fine
+ 'nested-interactive': {
+ enabled: false,
+ },
+ },
+ };
+ cy.checkA11y(undefined, axeOptions);
+ cy.get(".mx_SpacePanel").percySnapshotElement("Space panel collapsed", { widths: [68] });
+
+ cy.get(".mx_SpaceButton_toggleCollapse").click({ force: true });
+ cy.get(".mx_SpacePanel:not(.collapsed)").should("exist");
+
+ cy.contains(".mx_SpaceItem", "Root Space").should("exist")
+ .contains(".mx_SpaceItem", "Child Space").should("exist");
+
+ cy.checkA11y(undefined, axeOptions);
+ cy.get(".mx_SpacePanel").percySnapshotElement("Space panel expanded", { widths: [258] });
+ });
});
diff --git a/cypress/e2e/spotlight/spotlight.spec.ts b/cypress/e2e/spotlight/spotlight.spec.ts
index fee1e39071..d4b2d2cf9b 100644
--- a/cypress/e2e/spotlight/spotlight.spec.ts
+++ b/cypress/e2e/spotlight/spotlight.spec.ts
@@ -114,6 +114,7 @@ Cypress.Commands.add("startDM", (name: string) => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.get(".mx_Spinner").should("not.exist");
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", name);
@@ -216,6 +217,7 @@ describe("Spotlight", () => {
it("should find joined rooms", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightSearch().clear().type(room1Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", room1Name);
cy.spotlightResults().eq(0).click();
@@ -229,6 +231,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.PublicRooms);
cy.spotlightSearch().clear().type(room1Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", room1Name);
cy.spotlightResults().eq(0).should("contain", "View");
@@ -243,6 +246,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.PublicRooms);
cy.spotlightSearch().clear().type(room2Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", room2Name);
cy.spotlightResults().eq(0).should("contain", "Join");
@@ -258,6 +262,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.PublicRooms);
cy.spotlightSearch().clear().type(room3Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", room3Name);
cy.spotlightResults().eq(0).should("contain", "View");
@@ -296,6 +301,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(bot1Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", bot1Name);
cy.spotlightResults().eq(0).click();
@@ -308,6 +314,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(bot2Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", bot2Name);
cy.spotlightResults().eq(0).click();
@@ -324,6 +331,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(bot2Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", bot2Name);
cy.spotlightResults().eq(0).click();
@@ -341,27 +349,53 @@ describe("Spotlight", () => {
cy.get(".mx_RoomSublist[aria-label=People]").should("contain", bot2Name);
// Invite BotBob into existing DM with ByteBot
- cy.getDmRooms(bot2.getUserId()).then(dmRooms => dmRooms[0])
- .then(groupDmId => cy.inviteUser(groupDmId, bot1.getUserId()))
- .then(() => {
- cy.roomHeaderName().should("contain", `${bot1Name} and ${bot2Name}`);
- cy.get(".mx_RoomSublist[aria-label=People]").should("contain", `${bot1Name} and ${bot2Name}`);
+ cy.getDmRooms(bot2.getUserId())
+ .should("have.length", 1)
+ .then(dmRooms => cy.getClient().then(client => client.getRoom(dmRooms[0])))
+ .then(groupDm => {
+ cy.inviteUser(groupDm.roomId, bot1.getUserId());
+ cy.roomHeaderName().should(($element) =>
+ expect($element.get(0).innerText).contains(groupDm.name));
+ cy.get(".mx_RoomSublist[aria-label=People]").should(($element) =>
+ expect($element.get(0).innerText).contains(groupDm.name));
+
+ // Search for BotBob by id, should return group DM and user
+ cy.openSpotlightDialog().within(() => {
+ cy.spotlightFilter(Filter.People);
+ cy.spotlightSearch().clear().type(bot1.getUserId());
+ cy.wait(1000); // wait for the dialog code to settle
+ cy.spotlightResults().should("have.length", 2);
+ cy.spotlightResults().eq(0).should("contain", groupDm.name);
+ });
+
+ // Search for ByteBot by id, should return group DM and user
+ cy.openSpotlightDialog().within(() => {
+ cy.spotlightFilter(Filter.People);
+ cy.spotlightSearch().clear().type(bot2.getUserId());
+ cy.wait(1000); // wait for the dialog code to settle
+ cy.spotlightResults().should("have.length", 2);
+ cy.spotlightResults().eq(0).should("contain", groupDm.name);
+ });
});
+ });
- // Search for BotBob by id, should return group DM and user
+ // Test against https://github.com/vector-im/element-web/issues/22851
+ it("should show each person result only once", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
- cy.spotlightSearch().clear().type(bot1.getUserId());
- cy.spotlightResults().should("have.length", 2);
- cy.spotlightResults().eq(0).should("contain", `${bot1Name} and ${bot2Name}`);
- });
- // Search for ByteBot by id, should return group DM and user
- cy.openSpotlightDialog().within(() => {
- cy.spotlightFilter(Filter.People);
- cy.spotlightSearch().clear().type(bot2.getUserId());
- cy.spotlightResults().should("have.length", 2);
- cy.spotlightResults().eq(0).should("contain", `${bot1Name} and ${bot2Name}`);
+ // 2 rounds of search to simulate the bug conditions. Specifically, the first search
+ // should have 1 result (not 2) and the second search should also have 1 result (instead
+ // of the super buggy 3 described by https://github.com/vector-im/element-web/issues/22851)
+ //
+ // We search for user ID to trigger the profile lookup within the dialog.
+ for (let i = 0; i < 2; i++) {
+ cy.log("Iteration: " + i);
+ cy.spotlightSearch().clear().type(bot1.getUserId());
+ cy.wait(1000); // wait for the dialog code to settle
+ cy.spotlightResults().should("have.length", 1);
+ cy.spotlightResults().eq(0).should("contain", bot1.getUserId());
+ }
});
});
@@ -369,6 +403,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(bot2Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.spotlightResults().should("have.length", 1);
cy.spotlightResults().eq(0).should("contain", bot2Name);
cy.get(".mx_SpotlightDialog_startGroupChat").should("contain", "Start a group chat");
@@ -390,6 +425,7 @@ describe("Spotlight", () => {
cy.openSpotlightDialog().within(() => {
cy.spotlightFilter(Filter.People);
cy.spotlightSearch().clear().type(bot1Name);
+ cy.wait(1000); // wait for the dialog code to settle
cy.get(".mx_Spinner").should("not.exist");
cy.spotlightResults().should("have.length", 1);
});
diff --git a/cypress/e2e/timeline/timeline.spec.ts b/cypress/e2e/timeline/timeline.spec.ts
index 6eacacfed2..94b6ffaa42 100644
--- a/cypress/e2e/timeline/timeline.spec.ts
+++ b/cypress/e2e/timeline/timeline.spec.ts
@@ -155,7 +155,7 @@ describe("Timeline", () => {
cy.visit("/#/room/" + roomId);
cy.setSettingValue("layout", null, SettingLevel.DEVICE, Layout.IRC);
cy.contains(".mx_RoomView_body .mx_GenericEventListSummary[data-layout=irc] " +
- ".mx_GenericEventListSummary_summary", "created and configured the room.");
+ ".mx_GenericEventListSummary_summary", "created and configured the room.").should("exist");
cy.get(".mx_Spinner").should("not.exist");
cy.percySnapshot("Configured room on IRC layout");
});
@@ -166,7 +166,7 @@ describe("Timeline", () => {
// Wait until configuration is finished
cy.contains(".mx_RoomView_body .mx_GenericEventListSummary " +
- ".mx_GenericEventListSummary_summary", "created and configured the room.");
+ ".mx_GenericEventListSummary_summary", "created and configured the room.").should("exist");
// Click "expand" link button
cy.get(".mx_GenericEventListSummary_toggle[aria-expanded=false]").click();
@@ -193,14 +193,14 @@ describe("Timeline", () => {
cy.visit("/#/room/" + roomId);
cy.setSettingValue("showHiddenEventsInTimeline", null, SettingLevel.DEVICE, true);
cy.contains(".mx_RoomView_body .mx_GenericEventListSummary .mx_GenericEventListSummary_summary",
- "created and configured the room.");
+ "created and configured the room.").should("exist");
// Edit message
cy.contains(".mx_RoomView_body .mx_EventTile .mx_EventTile_line", "Message").within(() => {
cy.get('[aria-label="Edit"]').click({ force: true }); // Cypress has no ability to hover
cy.get(".mx_BasicMessageComposer_input").type("Edit{enter}");
});
- cy.get(".mx_RoomView_body .mx_EventTile").contains(".mx_EventTile[data-scroll-tokens]", "MessageEdit");
+ cy.contains(".mx_EventTile[data-scroll-tokens]", "MessageEdit").should("exist");
// Click timestamp to highlight hidden event line
cy.get(".mx_RoomView_body .mx_EventTile_info .mx_MessageTimestamp").click();
@@ -228,18 +228,19 @@ describe("Timeline", () => {
cy.visit("/#/room/" + roomId);
cy.setSettingValue("showHiddenEventsInTimeline", null, SettingLevel.DEVICE, true);
cy.contains(".mx_RoomView_body .mx_GenericEventListSummary " +
- ".mx_GenericEventListSummary_summary", "created and configured the room.");
+ ".mx_GenericEventListSummary_summary", "created and configured the room.").should("exist");
// Edit message
cy.contains(".mx_RoomView_body .mx_EventTile .mx_EventTile_line", "Message").within(() => {
cy.get('[aria-label="Edit"]').click({ force: true }); // Cypress has no ability to hover
cy.get(".mx_BasicMessageComposer_input").type("Edit{enter}");
});
- cy.contains(".mx_RoomView_body .mx_EventTile[data-scroll-tokens]", "MessageEdit");
+ cy.contains(".mx_RoomView_body .mx_EventTile[data-scroll-tokens]", "MessageEdit").should("exist");
// Click top left of the event toggle, which should not be covered by MessageActionBar's safe area
- cy.get(".mx_EventTile .mx_ViewSourceEvent").realHover()
- .get(".mx_EventTile .mx_ViewSourceEvent .mx_ViewSourceEvent_toggle").click('topLeft', { force: false });
+ cy.get(".mx_EventTile .mx_ViewSourceEvent").should("exist").realHover().within(() => {
+ cy.get(".mx_ViewSourceEvent_toggle").click('topLeft', { force: false });
+ });
// Make sure the expand toggle worked
cy.get(".mx_EventTile .mx_ViewSourceEvent_expanded .mx_ViewSourceEvent_toggle").should("be.visible");
@@ -249,17 +250,17 @@ describe("Timeline", () => {
cy.visit("/#/room/" + roomId);
cy.setSettingValue("layout", null, SettingLevel.DEVICE, Layout.Bubble);
cy.contains(".mx_RoomView_body .mx_GenericEventListSummary[data-layout=bubble] " +
- ".mx_GenericEventListSummary_summary", "created and configured the room.");
+ ".mx_GenericEventListSummary_summary", "created and configured the room.").should("exist");
// Click "expand" link button
cy.get(".mx_GenericEventListSummary_toggle[aria-expanded=false]").click();
// Click "collapse" link button on the first hovered info event line
- cy.get(".mx_GenericEventListSummary_unstyledList .mx_EventTile_info:first-of-type").realHover()
- .get(".mx_GenericEventListSummary_toggle[aria-expanded=true]").click({ force: false });
+ cy.get(".mx_GenericEventListSummary_unstyledList .mx_EventTile_info:first-of-type").realHover();
+ cy.get(".mx_GenericEventListSummary_toggle[aria-expanded=true]").click({ force: false });
// Make sure "collapse" link button worked
- cy.get(".mx_GenericEventListSummary_toggle[aria-expanded=false]");
+ cy.get(".mx_GenericEventListSummary_toggle[aria-expanded=false]").should("exist");
});
it("should highlight search result words regardless of formatting", () => {
@@ -273,6 +274,49 @@ describe("Timeline", () => {
cy.get(".mx_EventTile:not(.mx_EventTile_contextual)").find(".mx_EventTile_searchHighlight").should("exist");
cy.get(".mx_RoomView_searchResultsPanel").percySnapshotElement("Highlighted search results");
});
+
+ it("should render url previews", () => {
+ cy.intercept("**/_matrix/media/r0/thumbnail/matrix.org/2022-08-16_yaiSVSRIsNFfxDnV?*", {
+ statusCode: 200,
+ fixture: "riot.png",
+ headers: {
+ "Content-Type": "image/png",
+ },
+ }).as("mxc");
+ cy.intercept("**/_matrix/media/r0/preview_url?url=https%3A%2F%2Fcall.element.io%2F&ts=*", {
+ statusCode: 200,
+ body: {
+ "og:title": "Element Call",
+ "og:description": null,
+ "og:image:width": 48,
+ "og:image:height": 48,
+ "og:image": "mxc://matrix.org/2022-08-16_yaiSVSRIsNFfxDnV",
+ "og:image:type": "image/png",
+ "matrix:image:size": 2121,
+ },
+ headers: {
+ "Content-Type": "application/json",
+ },
+ }).as("preview_url");
+
+ cy.sendEvent(
+ roomId,
+ null,
+ "m.room.message" as EventType,
+ MessageEvent.from("https://call.element.io/").serialize().content,
+ );
+ cy.visit("/#/room/" + roomId);
+
+ cy.get(".mx_LinkPreviewWidget").should("exist").should("contain.text", "Element Call");
+
+ cy.wait("@preview_url");
+ cy.wait("@mxc");
+
+ cy.checkA11y();
+ cy.get(".mx_EventTile_last").percySnapshotElement("URL Preview", {
+ widths: [800, 400],
+ });
+ });
});
describe("message sending", () => {
@@ -285,7 +329,7 @@ describe("Timeline", () => {
cy.getComposer().type(`${MESSAGE}{enter}`);
// Reply to the message
- cy.get(".mx_RoomView_body .mx_EventTile").contains(".mx_EventTile_line", "Hello world").within(() => {
+ cy.get(".mx_RoomView_body").contains(".mx_EventTile_line", "Hello world").within(() => {
cy.get('[aria-label="Reply"]').click({ force: true }); // Cypress has no ability to hover
});
};
@@ -296,20 +340,22 @@ describe("Timeline", () => {
cy.getComposer().type(`${reply}{enter}`);
- cy.get(".mx_RoomView_body .mx_EventTile .mx_EventTile_line").find(".mx_ReplyTile .mx_MTextBody")
+ cy.get(".mx_RoomView_body .mx_EventTile .mx_EventTile_line .mx_ReplyTile .mx_MTextBody")
.should("contain", MESSAGE);
- cy.get(".mx_RoomView_body .mx_EventTile > .mx_EventTile_line > .mx_MTextBody").contains(reply)
+ cy.contains(".mx_RoomView_body .mx_EventTile > .mx_EventTile_line > .mx_MTextBody", reply)
.should("have.length", 1);
});
- xit("can reply with a voice message", () => {
+ it("can reply with a voice message", () => {
viewRoomSendMessageAndSetupReply();
- cy.openMessageComposerOptions().find(`[aria-label="Voice Message"]`).click();
+ cy.openMessageComposerOptions().within(() => {
+ cy.get(`[aria-label="Voice Message"]`).click();
+ });
cy.wait(3000);
- cy.getComposer().find(".mx_MessageComposer_sendMessage").click();
+ cy.get(".mx_RoomView_body .mx_MessageComposer .mx_MessageComposer_sendMessage").click();
- cy.get(".mx_RoomView_body .mx_EventTile .mx_EventTile_line").find(".mx_ReplyTile .mx_MTextBody")
+ cy.get(".mx_RoomView_body .mx_EventTile .mx_EventTile_line .mx_ReplyTile .mx_MTextBody")
.should("contain", MESSAGE);
cy.get(".mx_RoomView_body .mx_EventTile > .mx_EventTile_line > .mx_MVoiceMessageBody")
.should("have.length", 1);
diff --git a/cypress/e2e/user-onboarding/user-onboarding-new.ts b/cypress/e2e/user-onboarding/user-onboarding-new.ts
index 44787ee61e..c6eac8ce27 100644
--- a/cypress/e2e/user-onboarding/user-onboarding-new.ts
+++ b/cypress/e2e/user-onboarding/user-onboarding-new.ts
@@ -40,6 +40,13 @@ describe("User Onboarding (new user)", () => {
bot1 = _bot1;
});
cy.get('.mx_UserOnboardingPage').should('exist');
+ cy.get('.mx_UserOnboardingButton').should('exist');
+ cy.get('.mx_UserOnboardingList')
+ .should('exist')
+ .should(($list) => {
+ const list = $list.get(0);
+ expect(getComputedStyle(list).opacity).to.be.eq("1");
+ });
});
});
@@ -47,20 +54,14 @@ describe("User Onboarding (new user)", () => {
cy.stopSynapse(synapse);
});
- it("page is shown", () => {
- cy.get('.mx_UserOnboardingPage').should('exist');
- cy.get('.mx_UserOnboardingList')
- .should('exist')
- .should(($list) => {
- const list = $list.get(0);
- expect(getComputedStyle(list).opacity).to.be.eq("1");
- });
+ it("page is shown and preference exists", () => {
cy.get('.mx_UserOnboardingPage')
.percySnapshotElement("User onboarding page");
+ cy.openUserSettings("Preferences");
+ cy.contains("Show shortcut to welcome checklist above the room list").should("exist");
});
it("app download dialog", () => {
- cy.get('.mx_UserOnboardingPage').should('exist');
cy.contains(".mx_UserOnboardingTask_action", "Download apps").click();
cy.get('[role=dialog]')
.contains("#mx_BaseDialog_title", "Download Element")
@@ -79,8 +80,18 @@ describe("User Onboarding (new user)", () => {
cy.get(".mx_InviteDialog_editor input").type(bot1.getUserId());
cy.get(".mx_InviteDialog_buttonAndSpinner").click();
cy.get(".mx_InviteDialog_buttonAndSpinner").should("not.exist");
+ const message = "Hi!";
+ cy.get(".mx_SendMessageComposer").type(`${message}!{enter}`);
+ cy.contains(".mx_MTextBody.mx_EventTile_content", message);
cy.visit("/#/home");
-
+ cy.get('.mx_UserOnboardingPage').should('exist');
+ cy.get('.mx_UserOnboardingButton').should('exist');
+ cy.get('.mx_UserOnboardingList')
+ .should('exist')
+ .should(($list) => {
+ const list = $list.get(0);
+ expect(getComputedStyle(list).opacity).to.be.eq("1");
+ });
cy.get(".mx_ProgressBar").invoke("val").should("be.greaterThan", oldProgress);
});
});
diff --git a/cypress/e2e/user-onboarding/user-onboarding-old.ts b/cypress/e2e/user-onboarding/user-onboarding-old.ts
index 2be066e0a1..f079ed9a4c 100644
--- a/cypress/e2e/user-onboarding/user-onboarding-old.ts
+++ b/cypress/e2e/user-onboarding/user-onboarding-old.ts
@@ -40,7 +40,10 @@ describe("User Onboarding (old user)", () => {
cy.stopSynapse(synapse);
});
- it("page is hidden", () => {
+ it("page and preference are hidden", () => {
cy.get('.mx_UserOnboardingPage').should('not.exist');
+ cy.get('.mx_UserOnboardingButton').should('not.exist');
+ cy.openUserSettings("Preferences");
+ cy.contains("Show shortcut to welcome page above the room list").should("not.exist");
});
});
diff --git a/cypress/e2e/widgets/widget-pip-close.spec.ts b/cypress/e2e/widgets/widget-pip-close.spec.ts
new file mode 100644
index 0000000000..7689e38ed0
--- /dev/null
+++ b/cypress/e2e/widgets/widget-pip-close.spec.ts
@@ -0,0 +1,201 @@
+/*
+Copyright 2022 Mikhail Aheichyk
+Copyright 2022 Nordeck IT + Consulting GmbH.
+
+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 { IWidget } from "matrix-widget-api/src/interfaces/IWidget";
+
+import type { MatrixClient, MatrixEvent } from "matrix-js-sdk/src/matrix";
+import { SynapseInstance } from "../../plugins/synapsedocker";
+import { UserCredentials } from "../../support/login";
+
+const DEMO_WIDGET_ID = "demo-widget-id";
+const DEMO_WIDGET_NAME = "Demo Widget";
+const DEMO_WIDGET_TYPE = "demo";
+const ROOM_NAME = "Demo";
+
+const DEMO_WIDGET_HTML = `
+
+
+ Demo Widget
+
+
+
+
+
+
+`;
+
+// mostly copied from src/utils/WidgetUtils.waitForRoomWidget with small modifications
+function waitForRoomWidget(win: Cypress.AUTWindow, widgetId: string, roomId: string, add: boolean): Promise {
+ const matrixClient = win.mxMatrixClientPeg.get();
+
+ return new Promise((resolve, reject) => {
+ function eventsInIntendedState(evList) {
+ const widgetPresent = evList.some((ev) => {
+ return ev.getContent() && ev.getContent()['id'] === widgetId;
+ });
+ if (add) {
+ return widgetPresent;
+ } else {
+ return !widgetPresent;
+ }
+ }
+
+ const room = matrixClient.getRoom(roomId);
+
+ const startingWidgetEvents = room.currentState.getStateEvents('im.vector.modular.widgets');
+ if (eventsInIntendedState(startingWidgetEvents)) {
+ resolve();
+ return;
+ }
+
+ function onRoomStateEvents(ev: MatrixEvent) {
+ if (ev.getRoomId() !== roomId || ev.getType() !== "im.vector.modular.widgets") return;
+
+ const currentWidgetEvents = room.currentState.getStateEvents('im.vector.modular.widgets');
+
+ if (eventsInIntendedState(currentWidgetEvents)) {
+ matrixClient.removeListener(win.matrixcs.RoomStateEvent.Events, onRoomStateEvents);
+ resolve();
+ }
+ }
+
+ matrixClient.on(win.matrixcs.RoomStateEvent.Events, onRoomStateEvents);
+ });
+}
+
+describe("Widget PIP", () => {
+ let synapse: SynapseInstance;
+ let user: UserCredentials;
+ let bot: MatrixClient;
+ let demoWidgetUrl: string;
+
+ function roomCreateAddWidgetPip(userRemove: 'leave' | 'kick' | 'ban') {
+ cy.createRoom({
+ name: ROOM_NAME,
+ invite: [bot.getUserId()],
+ }).then(roomId => {
+ // sets bot to Admin and user to Moderator
+ cy.getClient().then(matrixClient => {
+ return matrixClient.sendStateEvent(roomId, 'm.room.power_levels', {
+ users: {
+ [user.userId]: 50,
+ [bot.getUserId()]: 100,
+ },
+ });
+ }).as('powerLevelsChanged');
+
+ // bot joins the room
+ cy.botJoinRoom(bot, roomId).as('botJoined');
+
+ // setup widget via state event
+ cy.getClient().then(async matrixClient => {
+ const content: IWidget = {
+ id: DEMO_WIDGET_ID,
+ creatorUserId: 'somebody',
+ type: DEMO_WIDGET_TYPE,
+ name: DEMO_WIDGET_NAME,
+ url: demoWidgetUrl,
+ };
+ await matrixClient.sendStateEvent(roomId, 'im.vector.modular.widgets', content, DEMO_WIDGET_ID);
+ }).as('widgetEventSent');
+
+ // open the room
+ cy.viewRoomByName(ROOM_NAME);
+
+ cy.all([
+ cy.get("@powerLevelsChanged"),
+ cy.get("@botJoined"),
+ cy.get("@widgetEventSent"),
+ ]).then(() => {
+ cy.window().then(async win => {
+ // wait for widget state event
+ await waitForRoomWidget(win, DEMO_WIDGET_ID, roomId, true);
+
+ // activate widget in pip mode
+ win.mxActiveWidgetStore.setWidgetPersistence(DEMO_WIDGET_ID, roomId, true);
+
+ // checks that pip window is opened
+ cy.get(".mx_CallView_pip").should("exist");
+
+ // checks that widget is opened in pip
+ cy.accessIframe(`iframe[title="${DEMO_WIDGET_NAME}"]`).within({}, () => {
+ cy.get("#demo").should('exist').then(async () => {
+ const userId = user.userId;
+ if (userRemove == 'leave') {
+ cy.getClient().then(async matrixClient => {
+ await matrixClient.leave(roomId);
+ });
+ } else if (userRemove == 'kick') {
+ await bot.kick(roomId, userId);
+ } else if (userRemove == 'ban') {
+ await bot.ban(roomId, userId);
+ }
+
+ // checks that pip window is closed
+ cy.get(".mx_CallView_pip").should("not.exist");
+ });
+ });
+ });
+ });
+ });
+ }
+
+ beforeEach(() => {
+ cy.startSynapse("default").then(data => {
+ synapse = data;
+
+ cy.initTestUser(synapse, "Mike").then(_user => {
+ user = _user;
+ });
+ cy.getBot(synapse, { displayName: "Bot", autoAcceptInvites: false }).then(_bot => {
+ bot = _bot;
+ });
+ });
+ cy.serveHtmlFile(DEMO_WIDGET_HTML).then(url => {
+ demoWidgetUrl = url;
+ });
+ });
+
+ afterEach(() => {
+ cy.stopSynapse(synapse);
+ cy.stopWebServers();
+ });
+
+ it('should be closed on leave', () => {
+ roomCreateAddWidgetPip('leave');
+ });
+
+ it('should be closed on kick', () => {
+ roomCreateAddWidgetPip('kick');
+ });
+
+ it('should be closed on ban', () => {
+ roomCreateAddWidgetPip('ban');
+ });
+});
diff --git a/cypress/plugins/synapsedocker/templates/consent/log.config b/cypress/plugins/synapsedocker/templates/consent/log.config
index ac232762da..b9123d0f5b 100644
--- a/cypress/plugins/synapsedocker/templates/consent/log.config
+++ b/cypress/plugins/synapsedocker/templates/consent/log.config
@@ -26,7 +26,7 @@ loggers:
synapse.storage.SQL:
# beware: increasing this to DEBUG will make synapse log sensitive
# information such as access tokens.
- level: INFO
+ level: DEBUG
twisted:
# We send the twisted logging directly to the file handler,
@@ -36,7 +36,7 @@ loggers:
propagate: false
root:
- level: INFO
+ level: DEBUG
# Write logs to the `buffer` handler, which will buffer them together in memory,
# then write them to a file.
diff --git a/cypress/plugins/synapsedocker/templates/default/homeserver.yaml b/cypress/plugins/synapsedocker/templates/default/homeserver.yaml
index 842009bcae..347dadc88f 100644
--- a/cypress/plugins/synapsedocker/templates/default/homeserver.yaml
+++ b/cypress/plugins/synapsedocker/templates/default/homeserver.yaml
@@ -22,8 +22,29 @@ log_config: "/data/log.config"
rc_messages_per_second: 10000
rc_message_burst_count: 10000
rc_registration:
- per_second: 10000
- burst_count: 10000
+ per_second: 10000
+ burst_count: 10000
+rc_joins:
+ local:
+ per_second: 9999
+ burst_count: 9999
+ remote:
+ per_second: 9999
+ burst_count: 9999
+rc_joins_per_room:
+ per_second: 9999
+ burst_count: 9999
+rc_3pid_validation:
+ per_second: 1000
+ burst_count: 1000
+
+rc_invites:
+ per_room:
+ per_second: 1000
+ burst_count: 1000
+ per_user:
+ per_second: 1000
+ burst_count: 1000
rc_login:
address:
diff --git a/cypress/plugins/synapsedocker/templates/default/log.config b/cypress/plugins/synapsedocker/templates/default/log.config
index ac232762da..b9123d0f5b 100644
--- a/cypress/plugins/synapsedocker/templates/default/log.config
+++ b/cypress/plugins/synapsedocker/templates/default/log.config
@@ -26,7 +26,7 @@ loggers:
synapse.storage.SQL:
# beware: increasing this to DEBUG will make synapse log sensitive
# information such as access tokens.
- level: INFO
+ level: DEBUG
twisted:
# We send the twisted logging directly to the file handler,
@@ -36,7 +36,7 @@ loggers:
propagate: false
root:
- level: INFO
+ level: DEBUG
# Write logs to the `buffer` handler, which will buffer them together in memory,
# then write them to a file.
diff --git a/cypress/support/settings.ts b/cypress/support/settings.ts
index 06ec815364..ec07df93aa 100644
--- a/cypress/support/settings.ts
+++ b/cypress/support/settings.ts
@@ -82,7 +82,7 @@ declare global {
* @param {*} value The new value of the setting, may be null.
* @return {Promise} Resolves when the setting has been changed.
*/
- setSettingValue(name: string, roomId: string, level: SettingLevel, value: any): Chainable;
+ setSettingValue(settingName: string, roomId: string, level: SettingLevel, value: any): Chainable;
/**
* Gets the value of a setting. The room ID is optional if the
@@ -96,7 +96,7 @@ declare global {
* value.
* @return {*} The value, or null if not found
*/
- getSettingValue(name: string, roomId?: string, excludeDefault?: boolean): Chainable;
+ getSettingValue(settingName: string, roomId?: string, excludeDefault?: boolean): Chainable;
}
}
}
diff --git a/package.json b/package.json
index 933ccfd7ad..a34cc60ad8 100644
--- a/package.json
+++ b/package.json
@@ -1,6 +1,6 @@
{
"name": "matrix-react-sdk",
- "version": "3.51.0",
+ "version": "3.52.0",
"description": "SDK for matrix.org using React",
"author": "matrix.org",
"repository": {
@@ -94,7 +94,7 @@
"matrix-encrypt-attachment": "^1.0.3",
"matrix-events-sdk": "^0.0.1-beta.7",
"matrix-js-sdk": "github:matrix-org/matrix-js-sdk#develop",
- "matrix-widget-api": "^0.1.0-beta.18",
+ "matrix-widget-api": "^1.0.0",
"minimist": "^1.2.5",
"opus-recorder": "^8.0.3",
"pako": "^2.0.3",
diff --git a/res/css/_components.pcss b/res/css/_components.pcss
index d6445f0143..3f23e6de57 100644
--- a/res/css/_components.pcss
+++ b/res/css/_components.pcss
@@ -27,7 +27,12 @@
@import "./components/views/location/_ZoomButtons.pcss";
@import "./components/views/messages/_MBeaconBody.pcss";
@import "./components/views/messages/shared/_MediaProcessingError.pcss";
+@import "./components/views/settings/devices/_DeviceDetails.pcss";
+@import "./components/views/settings/devices/_DeviceExpandDetailsButton.pcss";
+@import "./components/views/settings/devices/_DeviceSecurityCard.pcss";
@import "./components/views/settings/devices/_DeviceTile.pcss";
+@import "./components/views/settings/devices/_FilteredDeviceList.pcss";
+@import "./components/views/settings/devices/_SecurityRecommendations.pcss";
@import "./components/views/settings/devices/_SelectableDeviceTile.pcss";
@import "./components/views/settings/shared/_SettingsSubsection.pcss";
@import "./components/views/spaces/_QuickThemeSwitcher.pcss";
@@ -329,6 +334,7 @@
@import "./views/toasts/_IncomingCallToast.pcss";
@import "./views/toasts/_NonUrgentEchoFailureToast.pcss";
@import "./views/typography/_Heading.pcss";
+@import "./views/user-onboarding/_UserOnboardingButton.pcss";
@import "./views/user-onboarding/_UserOnboardingFeedback.pcss";
@import "./views/user-onboarding/_UserOnboardingHeader.pcss";
@import "./views/user-onboarding/_UserOnboardingList.pcss";
diff --git a/res/css/components/views/settings/devices/_DeviceDetails.pcss b/res/css/components/views/settings/devices/_DeviceDetails.pcss
new file mode 100644
index 0000000000..3017935bb7
--- /dev/null
+++ b/res/css/components/views/settings/devices/_DeviceDetails.pcss
@@ -0,0 +1,74 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_DeviceDetails {
+ display: flex;
+ flex-direction: column;
+ box-sizing: border-box;
+
+ width: 100%;
+
+ margin-top: $spacing-16;
+ padding: $spacing-16;
+ border-radius: 8px;
+ border: 1px solid $quinary-content;
+}
+
+.mx_DeviceDetails_section {
+ padding-bottom: $spacing-16;
+ margin-bottom: $spacing-16;
+ border-bottom: 1px solid $quinary-content;
+
+ display: grid;
+ grid-gap: $spacing-16;
+
+ &:last-child {
+ padding-bottom: 0;
+ border-bottom: 0;
+ margin-bottom: 0;
+ }
+}
+
+.mx_DeviceDetails_sectionHeading {
+ margin: 0;
+}
+
+.mxDeviceDetails_metadataTable {
+ font-size: $font-12px;
+ color: $secondary-content;
+
+ width: 100%;
+
+ border-spacing: 0;
+
+ th {
+ text-transform: uppercase;
+ font-weight: normal;
+ text-align: left;
+ }
+
+ td {
+ padding-top: $spacing-8;
+ }
+
+ .mxDeviceDetails_metadataLabel {
+ width: 160px;
+ }
+
+ .mxDeviceDetails_metadataValue {
+ color: $primary-content;
+ }
+}
diff --git a/res/css/components/views/settings/devices/_DeviceExpandDetailsButton.pcss b/res/css/components/views/settings/devices/_DeviceExpandDetailsButton.pcss
new file mode 100644
index 0000000000..4c9d787fdb
--- /dev/null
+++ b/res/css/components/views/settings/devices/_DeviceExpandDetailsButton.pcss
@@ -0,0 +1,41 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_DeviceExpandDetailsButton {
+ height: 32px;
+ width: 32px;
+ background: transparent;
+
+ border-radius: 4px;
+ color: $secondary-content;
+
+ --icon-transform: rotate(-90deg);
+}
+
+.mx_DeviceExpandDetailsButton.mx_DeviceExpandDetailsButton_expanded {
+ --icon-transform: rotate(0deg);
+
+ background: $system;
+}
+
+.mx_DeviceExpandDetailsButton_icon {
+ height: 12px;
+ width: 12px;
+
+ transition: all 0.3s;
+ transform: var(--icon-transform);
+ transform-origin: center;
+}
diff --git a/res/css/components/views/settings/devices/_DeviceSecurityCard.pcss b/res/css/components/views/settings/devices/_DeviceSecurityCard.pcss
new file mode 100644
index 0000000000..2c267b4314
--- /dev/null
+++ b/res/css/components/views/settings/devices/_DeviceSecurityCard.pcss
@@ -0,0 +1,70 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_DeviceSecurityCard {
+ width: 100%;
+ display: flex;
+ flex-direction: row;
+ align-items: flex-start;
+ box-sizing: border-box;
+
+ padding: $spacing-16;
+
+ border: 1px solid $quinary-content;
+ border-radius: 8px;
+}
+
+.mx_DeviceSecurityCard_icon {
+ flex: 0 0 40px;
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ margin-right: $spacing-16;
+ border-radius: 8px;
+
+ height: 40px;
+ width: 40px;
+
+ color: var(--icon-color);
+ background-color: var(--background-color);
+
+ &.Verified {
+ --icon-color: $e2e-verified-color;
+ --background-color: $e2e-verified-color-light;
+ }
+
+ &.Unverified {
+ --icon-color: $e2e-warning-color;
+ --background-color: $e2e-warning-color-light;
+ }
+
+ &.Inactive {
+ --icon-color: $secondary-content;
+ --background-color: $system;
+ }
+}
+
+.mx_DeviceSecurityCard_content {
+ flex: 1 1;
+}
+.mx_DeviceSecurityCard_heading {
+ margin: 0 0 $spacing-4 0;
+}
+.mx_DeviceSecurityCard_description {
+ margin: 0;
+ font-size: $font-12px;
+ color: $secondary-content;
+}
diff --git a/res/css/components/views/settings/devices/_DeviceTile.pcss b/res/css/components/views/settings/devices/_DeviceTile.pcss
index 159cace6ac..d89fd9c76e 100644
--- a/res/css/components/views/settings/devices/_DeviceTile.pcss
+++ b/res/css/components/views/settings/devices/_DeviceTile.pcss
@@ -18,7 +18,6 @@ limitations under the License.
display: flex;
flex-direction: row;
align-items: center;
-
width: 100%;
}
@@ -27,15 +26,21 @@ limitations under the License.
}
.mx_DeviceTile_metadata {
- margin-top: 2px;
+ margin-top: $spacing-4;
font-size: $font-12px;
color: $secondary-content;
+ line-height: $font-14px;
+}
+
+.mx_DeviceTile_inactiveIcon {
+ height: 14px;
+ margin-right: $spacing-8;
+ vertical-align: middle;
}
.mx_DeviceTile_actions {
display: grid;
grid-gap: $spacing-8;
grid-auto-flow: column;
-
margin-left: $spacing-8;
}
diff --git a/res/css/components/views/settings/devices/_FilteredDeviceList.pcss b/res/css/components/views/settings/devices/_FilteredDeviceList.pcss
new file mode 100644
index 0000000000..01c8df787e
--- /dev/null
+++ b/res/css/components/views/settings/devices/_FilteredDeviceList.pcss
@@ -0,0 +1,64 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_FilteredDeviceList {
+ .mx_Dropdown {
+ flex: 1 0 80px;
+ }
+}
+
+.mx_FilteredDeviceList_header {
+ display: flex;
+ flex-direction: row;
+ align-items: center;
+ box-sizing: border-box;
+
+ width: 100%;
+ height: 48px;
+ padding: 0 $spacing-16;
+ margin-bottom: $spacing-32;
+
+ background-color: $system;
+ border-radius: 8px;
+ color: $secondary-content;
+}
+
+.mx_FilteredDeviceList_headerLabel {
+ flex: 1 1 100%;
+}
+
+.mx_FilteredDeviceList_list {
+ list-style-type: none;
+ display: grid;
+ grid-gap: $spacing-16;
+ margin: 0;
+ padding: 0 $spacing-8;
+}
+
+.mx_FilteredDeviceList_listItem {
+ display: flex;
+ flex-direction: column;
+}
+
+.mx_FilteredDeviceList_securityCard {
+ margin-bottom: $spacing-32;
+}
+
+.mx_FilteredDeviceList_noResults {
+ width: 100%;
+ text-align: center;
+ margin-bottom: $spacing-32;
+}
diff --git a/res/css/components/views/settings/devices/_SecurityRecommendations.pcss b/res/css/components/views/settings/devices/_SecurityRecommendations.pcss
new file mode 100644
index 0000000000..d0a5333559
--- /dev/null
+++ b/res/css/components/views/settings/devices/_SecurityRecommendations.pcss
@@ -0,0 +1,19 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_SecurityRecommendations_spacing {
+ height: $spacing-16;
+}
diff --git a/res/css/structures/_HomePage.pcss b/res/css/structures/_HomePage.pcss
index 6bfabd9c87..f35de9919c 100644
--- a/res/css/structures/_HomePage.pcss
+++ b/res/css/structures/_HomePage.pcss
@@ -37,15 +37,15 @@ limitations under the License.
}
h1 {
- font-weight: 600;
+ font-weight: $font-semi-bold;
font-size: $font-32px;
line-height: $font-44px;
margin-bottom: 4px;
}
- h4 {
+ h2 {
margin-top: 4px;
- font-weight: 600;
+ font-weight: $font-semi-bold;
font-size: $font-18px;
line-height: $font-25px;
color: $muted-fg-color;
diff --git a/res/css/structures/_SpacePanel.pcss b/res/css/structures/_SpacePanel.pcss
index 7fdb2500b4..e62a6c90a1 100644
--- a/res/css/structures/_SpacePanel.pcss
+++ b/res/css/structures/_SpacePanel.pcss
@@ -78,10 +78,6 @@ $activeBorderColor: $primary-content;
margin: 0;
list-style: none;
padding: 0;
-
- > .mx_SpaceItem {
- padding-left: 16px;
- }
}
.mx_SpaceButton_toggleCollapse {
@@ -290,6 +286,11 @@ $activeBorderColor: $primary-content;
visibility: hidden;
}
}
+
+ .mx_SpaceTreeLevel {
+ // Indent subspaces
+ padding-left: 16px;
+ }
}
.mx_SpaceButton_avatarWrapper {
@@ -378,11 +379,16 @@ $activeBorderColor: $primary-content;
}
.mx_SpacePanel_contextMenu {
+ max-width: 360px;
+
.mx_SpacePanel_contextMenu_header {
margin: 12px 16px 12px;
font-weight: $font-semi-bold;
font-size: $font-15px;
line-height: $font-18px;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ white-space: nowrap;
}
.mx_SpacePanel_iconHome::before {
diff --git a/res/css/structures/_SpaceRoomView.pcss b/res/css/structures/_SpaceRoomView.pcss
index b39f57cbb0..2664549b17 100644
--- a/res/css/structures/_SpaceRoomView.pcss
+++ b/res/css/structures/_SpaceRoomView.pcss
@@ -177,6 +177,10 @@ $SpaceRoomViewInnerWidth: 428px;
h1 {
display: inline-block;
+ overflow: hidden;
+ text-overflow: ellipsis;
+ white-space: nowrap;
+ width: 100%;
}
}
diff --git a/res/css/views/elements/_AccessibleButton.pcss b/res/css/views/elements/_AccessibleButton.pcss
index 7d01c17e12..8718d86233 100644
--- a/res/css/views/elements/_AccessibleButton.pcss
+++ b/res/css/views/elements/_AccessibleButton.pcss
@@ -76,6 +76,12 @@ limitations under the License.
mask-image: url('$(res)/img/feather-customised/x.svg');
}
}
+
+ &.mx_AccessibleButton_kind_icon {
+ padding: 0;
+ height: 32px;
+ width: 32px;
+ }
}
&.mx_AccessibleButton_kind_primary,
diff --git a/res/css/views/elements/_LabelledCheckbox.pcss b/res/css/views/elements/_LabelledCheckbox.pcss
index d280d27eba..8545c6747b 100644
--- a/res/css/views/elements/_LabelledCheckbox.pcss
+++ b/res/css/views/elements/_LabelledCheckbox.pcss
@@ -16,6 +16,7 @@ limitations under the License.
.mx_LabelledCheckbox {
display: flex;
+ gap: 8px;
flex-direction: row;
.mx_Checkbox {
diff --git a/res/css/views/messages/_DisambiguatedProfile.pcss b/res/css/views/messages/_DisambiguatedProfile.pcss
index ef4bc7cfb3..4863dc3518 100644
--- a/res/css/views/messages/_DisambiguatedProfile.pcss
+++ b/res/css/views/messages/_DisambiguatedProfile.pcss
@@ -34,3 +34,10 @@ limitations under the License.
color: $primary-content;
}
}
+
+@media only percy {
+ .mx_DisambiguatedProfile_displayName {
+ /* Override the colour in percy tests for screenshot consistency */
+ color: $username-variant1-color !important;
+ }
+}
diff --git a/res/css/views/rooms/_LinkPreviewWidget.pcss b/res/css/views/rooms/_LinkPreviewWidget.pcss
index ceb715275d..7949233a9f 100644
--- a/res/css/views/rooms/_LinkPreviewWidget.pcss
+++ b/res/css/views/rooms/_LinkPreviewWidget.pcss
@@ -32,6 +32,7 @@ limitations under the License.
display: flex;
flex-wrap: wrap;
row-gap: $spacing-8;
+ flex: 1;
.mx_LinkPreviewWidget_image,
.mx_LinkPreviewWidget_caption {
diff --git a/res/css/views/user-onboarding/_UserOnboardingButton.pcss b/res/css/views/user-onboarding/_UserOnboardingButton.pcss
new file mode 100644
index 0000000000..3eba86045a
--- /dev/null
+++ b/res/css/views/user-onboarding/_UserOnboardingButton.pcss
@@ -0,0 +1,84 @@
+/*
+Copyright 2022 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.
+*/
+
+.mx_UserOnboardingButton {
+ display: flex;
+ flex-direction: column;
+ align-content: stretch;
+ align-items: stretch;
+ border-radius: 8px;
+ margin: $spacing-8 $spacing-8 0;
+ padding: $spacing-12;
+
+ &.mx_UserOnboardingButton_selected,
+ &:hover,
+ &:focus-within {
+ background-color: $panel-actions;
+ }
+
+ .mx_UserOnboardingButton_content {
+ display: flex;
+ flex-direction: row;
+ gap: 5px;
+ align-items: center;
+
+ .mx_Heading_h4 {
+ margin-right: auto;
+ font-size: $font-14px;
+ color: $primary-content;
+ }
+
+ .mx_UserOnboardingButton_percentage {
+ font-size: $font-12px;
+ color: $secondary-content;
+ }
+
+ .mx_UserOnboardingButton_close {
+ position: relative;
+ box-sizing: border-box;
+ width: 14px;
+ height: 14px;
+ border-radius: 7px;
+ border: 1px solid $secondary-content;
+ flex-shrink: 0;
+
+ &::before {
+ background-color: $secondary-content;
+ content: "";
+ mask-repeat: no-repeat;
+ mask-position: center;
+ mask-size: contain;
+ width: 7px;
+ height: 7px;
+ position: absolute;
+ left: 50%;
+ top: 50%;
+ transform: translate(-50%, -50%);
+ mask-image: url("$(res)/img/element-icons/cancel-rounded.svg");
+ }
+ }
+ }
+
+ .mx_ProgressBar {
+ width: auto;
+ margin-top: $spacing-8;
+ background: $background;
+ }
+
+ &.mx_UserOnboardingButton_completed .mx_ProgressBar {
+ display: none;
+ }
+}
diff --git a/res/img/e2e/verified-deprecated.svg b/res/img/e2e/verified-deprecated.svg
new file mode 100644
index 0000000000..f90d9db554
--- /dev/null
+++ b/res/img/e2e/verified-deprecated.svg
@@ -0,0 +1,3 @@
+
diff --git a/res/img/e2e/verified.svg b/res/img/e2e/verified.svg
index f90d9db554..9213d2b05d 100644
--- a/res/img/e2e/verified.svg
+++ b/res/img/e2e/verified.svg
@@ -1,3 +1,3 @@
diff --git a/res/img/e2e/warning-deprecated.svg b/res/img/e2e/warning-deprecated.svg
new file mode 100644
index 0000000000..58f5c3b7d1
--- /dev/null
+++ b/res/img/e2e/warning-deprecated.svg
@@ -0,0 +1,3 @@
+
diff --git a/res/img/e2e/warning.svg b/res/img/e2e/warning.svg
index 58f5c3b7d1..1acbb53bb7 100644
--- a/res/img/e2e/warning.svg
+++ b/res/img/e2e/warning.svg
@@ -1,3 +1,3 @@
diff --git a/res/img/element-icons/settings/inactive.svg b/res/img/element-icons/settings/inactive.svg
new file mode 100644
index 0000000000..63b6b97bd5
--- /dev/null
+++ b/res/img/element-icons/settings/inactive.svg
@@ -0,0 +1,3 @@
+
diff --git a/res/img/feather-customised/dropdown-arrow.svg b/res/img/feather-customised/dropdown-arrow.svg
index a1d46fa61a..24645d2bba 100644
--- a/res/img/feather-customised/dropdown-arrow.svg
+++ b/res/img/feather-customised/dropdown-arrow.svg
@@ -1,5 +1,5 @@