diff --git a/src/ZulipMobile.js b/src/ZulipMobile.js
index cfbdc532da6..34c0d662b13 100644
--- a/src/ZulipMobile.js
+++ b/src/ZulipMobile.js
@@ -17,6 +17,7 @@ import AppEventHandlers from './boot/AppEventHandlers';
import { initializeSentry } from './sentry';
import ZulipSafeAreaProvider from './boot/ZulipSafeAreaProvider';
import { OfflineNoticeProvider } from './boot/OfflineNoticeProvider';
+import TopicEditModalProvider from './boot/TopicEditModalProvider';
initializeSentry();
@@ -79,9 +80,11 @@ export default function ZulipMobile(): Node {
-
-
-
+
+
+
+
+
diff --git a/src/action-sheets/index.js b/src/action-sheets/index.js
index e697c7cb1c3..7b714309872 100644
--- a/src/action-sheets/index.js
+++ b/src/action-sheets/index.js
@@ -79,6 +79,7 @@ type TopicArgs = {
zulipFeatureLevel: number,
dispatch: Dispatch,
_: GetText,
+ startEditTopic: (streamId: number, topic: string) => void,
...
};
@@ -171,6 +172,14 @@ const deleteMessage = {
},
};
+const editTopic = {
+ title: 'Edit topic',
+ errorMessage: 'Failed to edit topic',
+ action: ({ streamId, topic, startEditTopic }) => {
+ startEditTopic(streamId, topic);
+ },
+};
+
const markTopicAsRead = {
title: 'Mark topic as read',
errorMessage: 'Failed to mark topic as read',
@@ -532,9 +541,18 @@ export const constructTopicActionButtons = (args: {|
const buttons = [];
const unreadCount = getUnreadCountForTopic(unread, streamId, topic);
+ const isAdmin = roleIsAtLeast(ownUserRole, Role.Admin);
if (unreadCount > 0) {
buttons.push(markTopicAsRead);
}
+ // At present, the permissions for editing the topic of a message are highly complex.
+ // Until we move to a better set of policy options, we'll only display the edit topic
+ // button to admins.
+ // Issue: https://github.com/zulip/zulip/issues/21739
+ // Relevant comment: https://github.com/zulip/zulip-mobile/issues/5365#issuecomment-1197093294
+ if (isAdmin) {
+ buttons.push(editTopic);
+ }
if (isTopicMuted(streamId, topic, mute)) {
buttons.push(unmuteTopic);
} else {
@@ -545,7 +563,7 @@ export const constructTopicActionButtons = (args: {|
} else {
buttons.push(unresolveTopic);
}
- if (roleIsAtLeast(ownUserRole, Role.Admin)) {
+ if (isAdmin) {
buttons.push(deleteTopic);
}
const sub = subscriptions.get(streamId);
@@ -705,6 +723,7 @@ export const showTopicActionSheet = (args: {|
showActionSheetWithOptions: ShowActionSheetWithOptions,
callbacks: {|
dispatch: Dispatch,
+ startEditTopic: (streamId: number, topic: string) => void,
_: GetText,
|},
backgroundData: $ReadOnly<{
diff --git a/src/boot/TopicEditModalProvider.js b/src/boot/TopicEditModalProvider.js
new file mode 100644
index 00000000000..f082e6c5b93
--- /dev/null
+++ b/src/boot/TopicEditModalProvider.js
@@ -0,0 +1,57 @@
+/* @flow strict-local */
+import React, { createContext, useState, useCallback, useContext } from 'react';
+import type { Context, Node } from 'react';
+
+import TopicEditModal from '../topics/TopicEditModal';
+
+type Props = $ReadOnly<{|
+ children: Node,
+|}>;
+
+export type TopicEditProviderStateType = {
+ streamId: number,
+ oldTopic: string,
+};
+
+type StartEditTopicContext = (streamId: number, oldTopic: string) => void;
+
+const TopicEditModalContext: Context = createContext(() => {
+ throw new Error(
+ 'Tried to open the edit-topic UI from a component without TopicEditModalProvider above it in the tree.',
+ );
+});
+
+export const useStartEditTopic = (): StartEditTopicContext => useContext(TopicEditModalContext);
+
+export default function TopicEditModalProvider(props: Props): Node {
+ const { children } = props;
+
+ const [topicModalProviderState, setTopicModalProviderState] =
+ useState(null);
+
+ const startEditTopic = useCallback(
+ (streamIdArg, oldTopicArg) => {
+ if (!topicModalProviderState) {
+ setTopicModalProviderState({
+ streamId: streamIdArg,
+ oldTopic: oldTopicArg,
+ });
+ }
+ },
+ [topicModalProviderState],
+ );
+
+ const closeEditTopicModal = () => {
+ setTopicModalProviderState(null);
+ };
+
+ return (
+
+
+ {children}
+
+ );
+}
diff --git a/src/chat/ChatScreen.js b/src/chat/ChatScreen.js
index c4b16876b2a..7cc1f3c3cf8 100644
--- a/src/chat/ChatScreen.js
+++ b/src/chat/ChatScreen.js
@@ -29,6 +29,7 @@ import { showErrorAlert } from '../utils/info';
import { TranslationContext } from '../boot/TranslationProvider';
import * as api from '../api';
import { useConditionalEffect } from '../reactUtils';
+import { useStartEditTopic } from '../boot/TopicEditModalProvider';
type Props = $ReadOnly<{|
navigation: AppNavigationProp<'chat'>,
@@ -132,6 +133,7 @@ export default function ChatScreen(props: Props): Node {
(value: EditMessage | null) => navigation.setParams({ editMessage: value }),
[navigation],
);
+ const startEditTopic = useStartEditTopic();
const isNarrowValid = useSelector(state => getIsNarrowValid(state, narrow));
const draft = useSelector(state => getDraftForNarrow(state, narrow));
@@ -219,6 +221,7 @@ export default function ChatScreen(props: Props): Node {
}
showMessagePlaceholders={showMessagePlaceholders}
startEditMessage={setEditMessage}
+ startEditTopic={startEditTopic}
/>
);
}
diff --git a/src/search/SearchMessagesCard.js b/src/search/SearchMessagesCard.js
index da94b7d4386..ede07efd0d4 100644
--- a/src/search/SearchMessagesCard.js
+++ b/src/search/SearchMessagesCard.js
@@ -9,6 +9,7 @@ import { createStyleSheet } from '../styles';
import LoadingIndicator from '../common/LoadingIndicator';
import SearchEmptyState from '../common/SearchEmptyState';
import MessageList from '../webview/MessageList';
+import { useStartEditTopic } from '../boot/TopicEditModalProvider';
const styles = createStyleSheet({
results: {
@@ -24,6 +25,7 @@ type Props = $ReadOnly<{|
export default function SearchMessagesCard(props: Props): Node {
const { narrow, isFetching, messages } = props;
+ const startEditTopic = useStartEditTopic();
if (isFetching) {
// Display loading indicator only if there are no messages to
@@ -55,6 +57,7 @@ export default function SearchMessagesCard(props: Props): Node {
// TODO: handle editing a message from the search results,
// or make this prop optional
startEditMessage={() => undefined}
+ startEditTopic={startEditTopic}
/>
);
diff --git a/src/streams/TopicItem.js b/src/streams/TopicItem.js
index d60e6b4f5fd..2d58dc2e3b6 100644
--- a/src/streams/TopicItem.js
+++ b/src/streams/TopicItem.js
@@ -25,6 +25,7 @@ import {
import { getMute } from '../mute/muteModel';
import { getUnread } from '../unread/unreadModel';
import { getOwnUserRole } from '../permissionSelectors';
+import { useStartEditTopic } from '../boot/TopicEditModalProvider';
const componentStyles = createStyleSheet({
selectedRow: {
@@ -70,6 +71,7 @@ export default function TopicItem(props: Props): Node {
useActionSheet().showActionSheetWithOptions;
const _ = useContext(TranslationContext);
const dispatch = useDispatch();
+ const startEditTopic = useStartEditTopic();
const backgroundData = useSelector(state => ({
auth: getAuth(state),
mute: getMute(state),
@@ -88,7 +90,7 @@ export default function TopicItem(props: Props): Node {
onLongPress={() => {
showTopicActionSheet({
showActionSheetWithOptions,
- callbacks: { dispatch, _ },
+ callbacks: { dispatch, startEditTopic, _ },
backgroundData,
streamId,
topic: name,
diff --git a/src/title/TitleStream.js b/src/title/TitleStream.js
index a6f4190dbfc..db6ec4997c6 100644
--- a/src/title/TitleStream.js
+++ b/src/title/TitleStream.js
@@ -27,6 +27,7 @@ import { showStreamActionSheet, showTopicActionSheet } from '../action-sheets';
import type { ShowActionSheetWithOptions } from '../action-sheets';
import { getUnread } from '../unread/unreadModel';
import { getOwnUserRole } from '../permissionSelectors';
+import { useStartEditTopic } from '../boot/TopicEditModalProvider';
type Props = $ReadOnly<{|
narrow: Narrow,
@@ -51,6 +52,7 @@ export default function TitleStream(props: Props): Node {
const { narrow, color } = props;
const dispatch = useDispatch();
const stream = useSelector(state => getStreamInNarrow(state, narrow));
+ const startEditTopic = useStartEditTopic();
const backgroundData = useSelector(state => ({
auth: getAuth(state),
mute: getMute(state),
@@ -75,7 +77,7 @@ export default function TitleStream(props: Props): Node {
? () => {
showTopicActionSheet({
showActionSheetWithOptions,
- callbacks: { dispatch, _ },
+ callbacks: { dispatch, startEditTopic, _ },
backgroundData,
streamId: stream.stream_id,
topic: topicOfNarrow(narrow),
diff --git a/src/topics/TopicEditModal.js b/src/topics/TopicEditModal.js
new file mode 100644
index 00000000000..f40ab4c987d
--- /dev/null
+++ b/src/topics/TopicEditModal.js
@@ -0,0 +1,184 @@
+/* @flow strict-local */
+import React, { useState, useContext, useEffect, useMemo } from 'react';
+import { Modal, View } from 'react-native';
+import type { Node } from 'react';
+
+import { useSelector } from '../react-redux';
+import styles, { ThemeContext, createStyleSheet } from '../styles';
+import * as api from '../api';
+import { fetchSomeMessageIdForConversation } from '../message/fetchActions';
+import ZulipTextIntl from '../common/ZulipTextIntl';
+import ZulipTextButton from '../common/ZulipTextButton';
+import Input from '../common/Input';
+import { getAuth, getZulipFeatureLevel, getStreamsById, getRealm } from '../selectors';
+import { TranslationContext } from '../boot/TranslationProvider';
+import { showErrorAlert } from '../utils/info';
+import { ensureUnreachable } from '../generics';
+
+type Props = $ReadOnly<{|
+ topicModalProviderState: {
+ oldTopic: string,
+ streamId: number,
+ } | null,
+ closeEditTopicModal: () => void,
+|}>;
+
+export default function TopicEditModal(props: Props): Node {
+ const { topicModalProviderState, closeEditTopicModal } = props;
+
+ const auth = useSelector(getAuth);
+ const zulipFeatureLevel = useSelector(getZulipFeatureLevel);
+ const streamsById = useSelector(getStreamsById);
+ const mandatoryTopics = useSelector(state => getRealm(state).mandatoryTopics);
+ const _ = useContext(TranslationContext);
+ const { backgroundColor } = useContext(ThemeContext);
+
+ const { oldTopic, streamId } = topicModalProviderState || {};
+
+ const [newTopicInputValue, setNewTopicInputValue] = useState(oldTopic);
+
+ // This resets the input value when we enter a new topic-editing session.
+ useEffect(() => {
+ setNewTopicInputValue(oldTopic);
+ }, [oldTopic]);
+
+ const validationErrors = useMemo(() => {
+ const result = [];
+ if (typeof newTopicInputValue === 'string') {
+ const trimmedInput = newTopicInputValue.trim();
+ if (mandatoryTopics && trimmedInput === '') {
+ result.push('mandatory-topic-empty');
+ }
+ if (trimmedInput === oldTopic) {
+ result.push('user-did-not-edit');
+ }
+ // Max topic length:
+ // https://zulip.com/api/update-message#parameter-topic
+ if (trimmedInput.length > 60) {
+ result.push('topic-too-long');
+ }
+ }
+ return result;
+ }, [mandatoryTopics, newTopicInputValue, oldTopic]);
+
+ const handleSubmit = async () => {
+ if (!topicModalProviderState) {
+ throw new Error(_('Topic, streamId, or input value is null.'));
+ }
+ if (validationErrors.length > 0) {
+ const errorMessages = validationErrors
+ .map(error => {
+ switch (error) {
+ case 'mandatory-topic-empty':
+ return _('Topic is required in this stream.');
+ case 'user-did-not-edit':
+ return _("You haven't made any changes.");
+ case 'topic-too-long':
+ return _('Topic too long (max 60 characters).');
+ default:
+ ensureUnreachable(error);
+ throw new Error();
+ }
+ })
+ .join('\n\n');
+ showErrorAlert(errorMessages);
+ return;
+ }
+ try {
+ const messageId = await fetchSomeMessageIdForConversation(
+ auth,
+ streamId,
+ oldTopic,
+ streamsById,
+ zulipFeatureLevel,
+ );
+ if (messageId == null) {
+ throw new Error(
+ _('No messages in topic: {streamAndTopic}', {
+ streamAndTopic: `#${streamsById.get(streamId)?.name ?? 'unknown'} > ${oldTopic}`,
+ }),
+ );
+ }
+ await api.updateMessage(auth, messageId, {
+ propagate_mode: 'change_all',
+ subject: newTopicInputValue.trim(),
+ ...(zulipFeatureLevel >= 9 && {
+ send_notification_to_old_thread: true,
+ send_notification_to_new_thread: true,
+ }),
+ });
+ } catch (error) {
+ showErrorAlert('Failed to edit topic.');
+ } finally {
+ closeEditTopicModal();
+ }
+ };
+
+ const modalStyles = createStyleSheet({
+ backdrop: {
+ position: 'absolute',
+ // backgroundColor and opacity aims to match how much our
+ // action sheets darken the background when they are toggled.
+ backgroundColor: 'black',
+ opacity: 0.25,
+ top: 0,
+ left: 0,
+ right: 0,
+ bottom: 0,
+ },
+ wrapper: {
+ flex: 1,
+ justifyContent: 'center',
+ alignItems: 'center',
+ },
+ modal: {
+ // these values are borrowed from Popup.js
+ justifyContent: 'flex-start',
+ backgroundColor,
+ padding: 15,
+ shadowOpacity: 0.5,
+ elevation: 8,
+ shadowRadius: 16,
+ borderRadius: 5,
+ width: 280,
+ },
+ buttonContainer: {
+ flexDirection: 'row',
+ justifyContent: 'flex-end',
+ },
+ titleText: {
+ fontSize: 18,
+ lineHeight: 21,
+ marginBottom: 10,
+ fontWeight: 'bold',
+ },
+ });
+
+ return (
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/src/webview/MessageList.js b/src/webview/MessageList.js
index 70a9b8eedd5..405e5663ea2 100644
--- a/src/webview/MessageList.js
+++ b/src/webview/MessageList.js
@@ -46,6 +46,7 @@ type OuterProps = $ReadOnly<{|
initialScrollMessageId: number | null,
showMessagePlaceholders: boolean,
startEditMessage: (editMessage: EditMessage) => void,
+ startEditTopic: (streamId: number, topic: string) => void,
|}>;
type SelectorProps = {|
diff --git a/src/webview/__tests__/generateInboundEvents-test.js b/src/webview/__tests__/generateInboundEvents-test.js
index e870d420522..f6db1fb1b9e 100644
--- a/src/webview/__tests__/generateInboundEvents-test.js
+++ b/src/webview/__tests__/generateInboundEvents-test.js
@@ -29,6 +29,7 @@ describe('generateInboundEvents', () => {
narrow: HOME_NARROW,
showMessagePlaceholders: false,
startEditMessage: jest.fn(),
+ startEditTopic: jest.fn(),
dispatch: jest.fn(),
...baseSelectorProps,
showActionSheetWithOptions: jest.fn(),
diff --git a/src/webview/handleOutboundEvents.js b/src/webview/handleOutboundEvents.js
index c74a1592b8d..8c798f416c8 100644
--- a/src/webview/handleOutboundEvents.js
+++ b/src/webview/handleOutboundEvents.js
@@ -169,6 +169,7 @@ type Props = $ReadOnly<{
doNotMarkMessagesAsRead: boolean,
showActionSheetWithOptions: ShowActionSheetWithOptions,
startEditMessage: (editMessage: EditMessage) => void,
+ startEditTopic: (streamId: number, topic: string) => void,
...
}>;
@@ -222,12 +223,19 @@ const handleLongPress = (
if (!message) {
return;
}
- const { dispatch, showActionSheetWithOptions, backgroundData, narrow, startEditMessage } = props;
+ const {
+ dispatch,
+ showActionSheetWithOptions,
+ backgroundData,
+ narrow,
+ startEditMessage,
+ startEditTopic,
+ } = props;
if (target === 'header') {
if (message.type === 'stream') {
showTopicActionSheet({
showActionSheetWithOptions,
- callbacks: { dispatch, _ },
+ callbacks: { dispatch, startEditTopic, _ },
backgroundData,
streamId: message.stream_id,
topic: message.subject,
diff --git a/static/translations/messages_en.json b/static/translations/messages_en.json
index 074764a61cc..1e41ec749e8 100644
--- a/static/translations/messages_en.json
+++ b/static/translations/messages_en.json
@@ -350,5 +350,10 @@
"Copy link to stream": "Copy link to stream",
"Failed to copy stream link": "Failed to copy stream link",
"A stream with this name already exists.": "A stream with this name already exists.",
- "Streams": "Streams"
+ "Streams": "Streams",
+ "Edit topic": "Edit topic",
+ "Submit": "Submit",
+ "Topic is required in this stream.": "Topic is required in this stream.",
+ "You haven't made any changes.": "You haven't made any changes.",
+ "Topic too long (max 60 characters).": "Topic too long (max 60 characters)."
}