Compare commits

...

2 Commits

Author SHA1 Message Date
jmug 1b2a980a47 [chore] Workflow cleanup.
Docker / publish (push) Successful in 13m59s Details
2026-02-23 01:55:29 -08:00
Mariano Uvalle 1ce522579e Mobile reactions and WS reconnection
* [feat] Add reaction menu item for mobile clients.

* [fix] Reconnection to the websocket with exponential backoff.

* [chore] Dev flake.

* [hack] Push to my personal gcr.
2026-02-22 13:47:39 -08:00
12 changed files with 211 additions and 165 deletions

1
.envrc Normal file
View File

@ -0,0 +1 @@
use flake

View File

@ -3,36 +3,38 @@ name: Docker
on: on:
push: push:
branches: branches:
- "master" - "handmade"
tags: tags:
- "*" - "*"
paths-ignore: # TODO: Bring back once gitea is updated past 1.21
- ".github/**" # paths-ignore:
- "!.github/workflows/docker.yml" # - ".github/**"
- "!.github/workflows/preview_*.yml" # - "!.github/workflows/docker.yml"
- ".vscode/**" # - ".vscode/**"
- ".gitignore" # - ".gitignore"
- ".gitlab-ci.yml" # - ".gitlab-ci.yml"
- "LICENSE" # - "LICENSE"
- "README" # - "README"
pull_request: pull_request:
branches: branches:
- "master" - "handmade"
paths-ignore: # TODO: Bring back once gitea is updated past 1.21
- ".github/**" # paths-ignore:
- "!.github/workflows/docker.yml" # - ".github/**"
- "!.github/workflows/preview_*.yml" # - "!.github/workflows/docker.yml"
- ".vscode/**" # - "!.github/workflows/preview_*.yml"
- ".gitignore" # - ".vscode/**"
- ".gitlab-ci.yml" # - ".gitignore"
- "LICENSE" # - ".gitlab-ci.yml"
- "README" # - "LICENSE"
# - "README"
workflow_dispatch: workflow_dispatch:
jobs: jobs:
publish: publish:
runs-on: ubuntu-latest runs-on: ubuntu-latest
if: github.event_name != 'pull_request' # NOTE: Running on pull requests for now, but without pushing.
# if: github.event_name != 'pull_request'
steps: steps:
- name: Checkout - name: Checkout
uses: actions/checkout@v4 uses: actions/checkout@v4
@ -46,7 +48,7 @@ jobs:
id: meta id: meta
uses: docker/metadata-action@v5 uses: docker/metadata-action@v5
with: with:
images: revoltchat/client, ghcr.io/revoltchat/client images: handmadecities/handmade-revolt-web-client
env: env:
DOCKER_METADATA_ANNOTATIONS_LEVELS: manifest,index DOCKER_METADATA_ANNOTATIONS_LEVELS: manifest,index
- name: Login to DockerHub - name: Login to DockerHub
@ -55,13 +57,6 @@ jobs:
with: with:
username: ${{ secrets.DOCKERHUB_USERNAME }} username: ${{ secrets.DOCKERHUB_USERNAME }}
password: ${{ secrets.DOCKERHUB_TOKEN }} password: ${{ secrets.DOCKERHUB_TOKEN }}
- name: Login to Github Container Registry
uses: docker/login-action@v3
if: github.event_name != 'pull_request'
with:
registry: ghcr.io
username: ${{ github.actor }}
password: ${{ secrets.GITHUB_TOKEN }}
- name: Build and publish - name: Build and publish
uses: docker/build-push-action@v6 uses: docker/build-push-action@v6
with: with:

View File

@ -1,54 +0,0 @@
name: Add Issue to Board
on:
issues:
types: [opened]
jobs:
track_issue:
runs-on: ubuntu-latest
steps:
- name: Get project data
env:
GITHUB_TOKEN: ${{ secrets.PAT }}
run: |
gh api graphql -f query='
query {
organization(login: "revoltchat"){
projectV2(number: 3) {
id
fields(first:20) {
nodes {
... on ProjectV2SingleSelectField {
id
name
options {
id
name
}
}
}
}
}
}
}' > project_data.json
echo 'PROJECT_ID='$(jq '.data.organization.projectV2.id' project_data.json) >> $GITHUB_ENV
echo 'STATUS_FIELD_ID='$(jq '.data.organization.projectV2.fields.nodes[] | select(.name== "Status") | .id' project_data.json) >> $GITHUB_ENV
echo 'TODO_OPTION_ID='$(jq '.data.organization.projectV2.fields.nodes[] | select(.name== "Status") | .options[] | select(.name=="Todo") |.id' project_data.json) >> $GITHUB_ENV
- name: Add issue to project
env:
GITHUB_TOKEN: ${{ secrets.PAT }}
ISSUE_ID: ${{ github.event.issue.node_id }}
run: |
item_id="$( gh api graphql -f query='
mutation($project:ID!, $issue:ID!) {
addProjectV2ItemById(input: {projectId: $project, contentId: $issue}) {
item {
id
}
}
}' -f project=$PROJECT_ID -f issue=$ISSUE_ID --jq '.data.addProjectV2ItemById.item.id')"
echo 'ITEM_ID='$item_id >> $GITHUB_ENV

View File

@ -1,79 +0,0 @@
name: Add PR to Board
on:
pull_request_target:
types: [opened, synchronize, ready_for_review, review_requested]
jobs:
track_pr:
runs-on: ubuntu-latest
steps:
- name: Get project data
env:
GITHUB_TOKEN: ${{ secrets.PAT }}
run: |
gh api graphql -f query='
query {
organization(login: "revoltchat"){
projectV2(number: 5) {
id
fields(first:20) {
nodes {
... on ProjectV2SingleSelectField {
id
name
options {
id
name
}
}
}
}
}
}
}' > project_data.json
echo 'PROJECT_ID='$(jq '.data.organization.projectV2.id' project_data.json) >> $GITHUB_ENV
echo 'STATUS_FIELD_ID='$(jq '.data.organization.projectV2.fields.nodes[] | select(.name== "Status") | .id' project_data.json) >> $GITHUB_ENV
echo 'INCOMING_OPTION_ID='$(jq '.data.organization.projectV2.fields.nodes[] | select(.name== "Status") | .options[] | select(.name=="🆕 Untriaged") |.id' project_data.json) >> $GITHUB_ENV
- name: Add PR to project
env:
GITHUB_TOKEN: ${{ secrets.PAT }}
PR_ID: ${{ github.event.pull_request.node_id }}
run: |
item_id="$( gh api graphql -f query='
mutation($project:ID!, $pr:ID!) {
addProjectV2ItemById(input: {projectId: $project, contentId: $pr}) {
item {
id
}
}
}' -f project=$PROJECT_ID -f pr=$PR_ID --jq '.data.addProjectV2ItemById.item.id')"
echo 'ITEM_ID='$item_id >> $GITHUB_ENV
- name: Set fields
env:
GITHUB_TOKEN: ${{ secrets.PAT }}
run: |
gh api graphql -f query='
mutation (
$project: ID!
$item: ID!
$status_field: ID!
$status_value: String!
) {
set_status: updateProjectV2ItemFieldValue(input: {
projectId: $project
itemId: $item
fieldId: $status_field
value: {
singleSelectOptionId: $status_value
}
}) {
projectV2Item {
id
}
}
}' -f project=$PROJECT_ID -f item=$ITEM_ID -f status_field=$STATUS_FIELD_ID -f status_value=${{ env.INCOMING_OPTION_ID }} --silent

2
.gitignore vendored
View File

@ -15,3 +15,5 @@ public/assets_*
!public/assets_default !public/assets_default
.vscode/chrome_data .vscode/chrome_data
.direnv

64
flake.lock Normal file
View File

@ -0,0 +1,64 @@
{
"nodes": {
"flake-utils": {
"inputs": {
"systems": [
"systems"
]
},
"locked": {
"lastModified": 1731533236,
"narHash": "sha256-l0KFg5HjrsfsO/JpG+r7fRrqm12kzFHyUHqHCVpMMbI=",
"owner": "numtide",
"repo": "flake-utils",
"rev": "11707dc2f618dd54ca8739b309ec4fc024de578b",
"type": "github"
},
"original": {
"owner": "numtide",
"repo": "flake-utils",
"type": "github"
}
},
"nixpkgs": {
"locked": {
"lastModified": 1770169770,
"narHash": "sha256-awR8qIwJxJJiOmcEGgP2KUqYmHG4v/z8XpL9z8FnT1A=",
"owner": "NixOS",
"repo": "nixpkgs",
"rev": "aa290c9891fa4ebe88f8889e59633d20cc06a5f2",
"type": "github"
},
"original": {
"owner": "NixOS",
"ref": "nixpkgs-unstable",
"repo": "nixpkgs",
"type": "github"
}
},
"root": {
"inputs": {
"flake-utils": "flake-utils",
"nixpkgs": "nixpkgs",
"systems": "systems"
}
},
"systems": {
"locked": {
"lastModified": 1681028828,
"narHash": "sha256-Vy1rq5AaRuLzOxct8nz4T6wlgyUR7zLU309k9mBC768=",
"owner": "nix-systems",
"repo": "default",
"rev": "da67096a3b9bf56a91d16901293e51ba5b49a27e",
"type": "github"
},
"original": {
"owner": "nix-systems",
"repo": "default",
"type": "github"
}
}
},
"root": "root",
"version": 7
}

28
flake.nix Normal file
View File

@ -0,0 +1,28 @@
{
description = "Node+yarn dev shell flake";
inputs = {
nixpkgs.url = "github:NixOS/nixpkgs/nixpkgs-unstable";
systems.url = "github:nix-systems/default";
flake-utils = {
url = "github:numtide/flake-utils";
inputs.systems.follows = "systems";
};
};
outputs =
{ nixpkgs, flake-utils, ... }:
flake-utils.lib.eachDefaultSystem (
system:
let
pkgs = nixpkgs.legacyPackages.${system};
in
{
devShells.default = pkgs.mkShell {
packages = with pkgs; [
nodejs_24
yarn
];
};
}
);
}

View File

@ -28,6 +28,7 @@ type Transition =
| "SUCCESS" | "SUCCESS"
| "DISCONNECT" | "DISCONNECT"
| "RETRY" | "RETRY"
| "RETRY_FAILED"
| "LOGOUT" | "LOGOUT"
| "ONLINE" | "ONLINE"
| "OFFLINE"; | "OFFLINE";
@ -40,6 +41,7 @@ export default class Session {
state: State = window.navigator.onLine ? "Ready" : "Offline"; state: State = window.navigator.onLine ? "Ready" : "Offline";
user_id: string | null = null; user_id: string | null = null;
client: Client | null = null; client: Client | null = null;
retryAttempts: number = 0;
/** /**
* Create a new Session * Create a new Session
@ -89,9 +91,11 @@ export default class Session {
* Called when the client signals it has disconnected * Called when the client signals it has disconnected
*/ */
private onDropped() { private onDropped() {
this.emit({ if (this.state === "Connecting") {
action: "DISCONNECT", this.emit({ action: "RETRY_FAILED" });
}); } else {
this.emit({ action: "DISCONNECT" });
}
} }
/** /**
@ -211,6 +215,7 @@ export default class Session {
// Ready successfully received // Ready successfully received
case "SUCCESS": { case "SUCCESS": {
this.assert("Connecting"); this.assert("Connecting");
this.retryAttempts = 0;
this.state = "Online"; this.state = "Online";
break; break;
} }
@ -239,6 +244,18 @@ export default class Session {
this.state = "Connecting"; this.state = "Connecting";
break; break;
} }
// Reconnect attempt failed, schedule another with backoff
case "RETRY_FAILED": {
this.assert("Connecting");
this.retryAttempts++;
const delay = Math.min(500 * Math.pow(2, this.retryAttempts), 16000);
setTimeout(() => {
if (this.state === "Connecting") {
this.client!.websocket.connect();
}
}, delay);
break;
}
// User instructed logout // User instructed logout
case "LOGOUT": { case "LOGOUT": {
this.assert("Connecting", "Online", "Disconnected"); this.assert("Connecting", "Online", "Disconnected");

View File

@ -32,6 +32,7 @@ import CreateRole from "./components/CreateRole";
import CreateServer from "./components/CreateServer"; import CreateServer from "./components/CreateServer";
import CustomStatus from "./components/CustomStatus"; import CustomStatus from "./components/CustomStatus";
import DeleteMessage from "./components/DeleteMessage"; import DeleteMessage from "./components/DeleteMessage";
import ReactMessage from "./components/ReactMessage";
import Error from "./components/Error"; import Error from "./components/Error";
import ImageViewer from "./components/ImageViewer"; import ImageViewer from "./components/ImageViewer";
import KickMember from "./components/KickMember"; import KickMember from "./components/KickMember";
@ -275,6 +276,7 @@ export const modalController = new ModalControllerExtended({
create_bot: CreateBot, create_bot: CreateBot,
custom_status: CustomStatus, custom_status: CustomStatus,
delete_message: DeleteMessage, delete_message: DeleteMessage,
react_message: ReactMessage,
error: Error, error: Error,
image_viewer: ImageViewer, image_viewer: ImageViewer,
kick_member: KickMember, kick_member: KickMember,

View File

@ -0,0 +1,46 @@
import styled from "styled-components";
import { Modal } from "@revoltchat/ui";
import { ModalProps } from "../types"
import { Message } from "revolt.js";
import { emojiDictionary } from "../../../assets/emojis"
import { HackAlertThisFileWillBeReplaced } from "../../../components/common/messaging/MessageBox"
const PickerContainer = styled.div`
max-height: 420px;
max-width: 370px;
overflow: hidden;
> div {
position: unset;
}
`
export default function ReactMessage({
target: message,
onClose,
...props
}: ModalProps<"react_message">) {
return (
<Modal
{...props}
padding={false}
maxWidth="370px"
>
<PickerContainer>
<HackAlertThisFileWillBeReplaced
onSelect={(emoji) =>{
message.react(
emojiDictionary[
emoji as keyof typeof emojiDictionary
] ?? emoji,
);
onClose();
}}
onClose={onClose}
/>
</PickerContainer>
</Modal>
)
}

View File

@ -153,6 +153,10 @@ export type Modal = {
type: "delete_message"; type: "delete_message";
target: Message; target: Message;
} }
| {
type: "react_message",
target: Message;
}
| { | {
type: "kick_member"; type: "kick_member";
member: Member; member: Member;

View File

@ -65,6 +65,7 @@ type Action =
| { action: "quote_message"; content: string } | { action: "quote_message"; content: string }
| { action: "edit_message"; id: string } | { action: "edit_message"; id: string }
| { action: "delete_message"; target: Message } | { action: "delete_message"; target: Message }
| { action: "react_message"; target: Message }
| { action: "open_file"; attachment: API.File } | { action: "open_file"; attachment: API.File }
| { action: "save_file"; attachment: API.File } | { action: "save_file"; attachment: API.File }
| { action: "copy_file_link"; attachment: API.File } | { action: "copy_file_link"; attachment: API.File }
@ -402,6 +403,13 @@ export default function ContextMenus() {
}); });
break; break;
case "react_message":
modalController.push({
type: "react_message",
target: data.target,
});
break;
case "leave_group": case "leave_group":
case "close_dm": case "close_dm":
case "delete_channel": case "delete_channel":
@ -508,6 +516,8 @@ export default function ContextMenus() {
"Open in Admin Panel" "Open in Admin Panel"
) : locale === "admin_system" ? ( ) : locale === "admin_system" ? (
"Open User in Admin Panel" "Open User in Admin Panel"
) : locale === "react_message" ? (
"React"
) : ( ) : (
<Text <Text
id={`app.context_menu.${ id={`app.context_menu.${
@ -833,6 +843,16 @@ export default function ContextMenus() {
}); });
} }
if (message.channel?.havePermission("React")) {
generateAction(
{
action: "react_message",
target: message,
},
"react_message",
);
}
if (message.author_id !== userId) { if (message.author_id !== userId) {
generateAction( generateAction(
{ {