const CustomMessageInput = () => (
<div className='message-input'>
<textarea value={text} className='message-input__input' />
<button type='button' className='message-input__button'>
⬆️
</button>
</div>
);
Message Input UI
Message input is used for composing and editing messages. In this sense, it’s a primary component that users interact with in a chat, so it’s important to get it right.
Message input is a bit more complex than it might seem at first glance. Not just a text box with a “send” button, it has a lot of hidden features:
- Updating the typing status
- Uploading and previewing attachments
- Displaying link previews
- Auto-completing mentions, commands, emoji…
We will cover this features step by step. For now, let’s start with the simplest markup possible:
.message-input {
display: flex;
align-items: center;
border: 2px solid #00000029;
border-radius: 8px;
}
.message-input:has(.message-input__input:focus) {
border-color: #005fff;
}
.message-input__input {
flex-grow: 1;
border: 0;
outline: 0;
background: none;
font: inherit;
padding: 8px;
resize: none;
}
.message-input__button {
border: 1px solid transparent;
outline: 0;
background: none;
font: inherit;
border-radius: 4px;
margin: 8px;
padding: 8px;
cursor: pointer;
}
.message-input__button:hover {
background: #fafafa;
border-color: #00000014;
}
.message-input__button:focus {
border-color: #005fff;
}
Note that you should not render your custom message input directly, but instead pass it as a prop to
either Channel
or
MessageInput
component. That way,
you can be sure that your input is wrapped with the necessary context providers, most importantly
the MessageInputContext
.
import {
Chat,
Channel,
ChannelHeader,
ChannelList,
MessageList,
Thread,
Window,
MessageInput,
} from 'stream-chat-react';
export const App = () => (
<Chat client={chatClient}>
<ChannelList filters={filters} sort={sort} options={options} />
<Channel>
<Window>
<ChannelHeader />
<MessageList />
<MessageInput Input={CustomMessageInput} />
</Window>
<Thread />
</Channel>
</Chat>
);
For now, our custom input doesn’t do anything. The
MessageInputContext
handles most of the
input-related state and actions, so instead of handling the state yourself, just use the provided
values and callbacks:
import { useMessageInputContext } from 'stream-chat-react';
const CustomMessageInput = () => {
const { text, handleChange, handleSubmit } = useMessageInputContext();
return (
<div className='message-input'>
<textarea value={text} className='message-input__input' onChange={handleChange} />
<button type='button' className='message-input__button' onClick={handleSubmit}>
⬆️
</button>
</div>
);
};
The great thing is that the handleChange
callback not only updates the text value, but also
detects links (if URL enrichment is enabled), and updates your typing status.
And with the handleSubmit
callback added to the “send” button, our basic implementation is
complete - try it out!
Uploading and Previewing Attachments
To support adding attachments to the message, we should start by adding a file input. And similar to
the “send” button, once a file is selected, we can use a callback provided in the
MessageInputContext
to upload it as an
attachment:
import { useMessageInputContext } from 'stream-chat-react';
const AttachmentUploadButton = () => {
const { uploadNewFiles } = useMessageInputContext();
function handleChange(e) {
const files = e.currentTarget.files;
if (files && files.length > 0) {
uploadNewFiles(files);
e.currentTarget.value = '';
}
}
return (
<label className='message-input__button'>
<input type='file' className='visually-hidden' onChange={handleChange} />
📎
</label>
);
};
There are two cases when uploads should not be allowed:
- Uploads are disabled for the current channel. We should check the
isUploadEnabled
value from theMessageInputContext
to make sure. - The maximum number of message attachments has been reached. For this we should check the
maxFilesLeft
value from theMessageInputContext
.
Let’s add these two checks:
import { useMessageInputContext } from 'stream-chat-react';
const AttachmentUploadButton = () => {
const { uploadNewFiles, isUploadEnabled, maxFilesLeft } = useMessageInputContext();
function handleChange(e) {
const files = e.currentTarget.files;
if (files && files.length > 0) {
uploadNewFiles(files);
}
}
if (!isUploadEnabled || maxFilesLeft === 0) {
return null;
}
return (
<label className='message-input__button'>
<input type='file' className='visually-hidden' onChange={handleChange} />
📎
</label>
);
};
Now we need a way to preview the added attachments. The SDK provides a ready-made component for
this:
AttachmentPreviewList
.
Instead of importing it directly, you can grab it from the
ComponentContext
, which is used throughout the
SDK to provide overridable UI components, and only fall back to the default implementation if it
hasn’t been overridden:
import {
useMessageInputContext,
useComponentContext,
AttachmentPreviewList as DefaultAttachmentPreviewList,
} from 'stream-chat-react';
const CustomMessageInput = () => {
const { text, handleChange, handleSubmit } = useMessageInputContext();
const { AttachmentPreviewList = DefaultAttachmentPreviewList } = useComponentContext();
return (
<div className='message-input'>
<div className='message-input__composer'>
<AttachmentUploadButton />
<textarea className='message-input__input' value={text} onChange={handleChange} />
<button type='button' className='message-input__button' onClick={handleSubmit}>
⬆️
</button>
</div>
<AttachmentPreviewList />
</div>
);
};
.message-input__composer {
display: flex;
align-items: center;
border: 2px solid #00000029;
border-radius: 8px;
}
.message-input__composer:has(.message-input__input:focus) {
border-color: #005fff;
}
.message-input__input {
flex-grow: 1;
border: 0;
outline: 0;
background: none;
font: inherit;
padding: 8px;
resize: none;
}
.message-input__button {
border: 1px solid transparent;
outline: 0;
background: none;
font: inherit;
border-radius: 4px;
margin: 8px;
padding: 8px;
cursor: pointer;
}
.message-input__button:hover {
background: #fafafa;
border-color: #00000014;
}
.message-input__button:focus,
.message-input__button:focus-within {
border-color: #005fff;
}
.visually-hidden {
width: 0;
height: 0;
pointer-events: none;
}
The nice thing about this approach is that our custom message input is not tied to a particular
implementation of the attachment preview component, and the preview component can be overridden at
the Channel
level.
However, if you need deep customization of the attachment preview list, and the out-of-the-box component doesn’t meet your needs, it’s fairly easy to implement it from scratch.
The two relevant values from the MessageInputContext
are
fileUploads
and
imageUploads
, which map upload
ids to the objects that describe an attachment. Image attachments are usually treated differently
from generic files, so it is convenient to keep the two attachment types are separate.
Since object keys come in no particular order, you can use
fileOrder
and
imageOrder
arrays, which specify
the order of attachments by their ids.
import { useMessageInputContext } from 'stream-chat-react';
const CustomAttachmentPreviewList = () => {
const { fileOrder, imageOrder } = useMessageInputContext();
if (fileOrder.length === 0 && imageOrder.length === 0) {
return null;
}
return (
<ul className='message-input__attachments'>
{imageOrder.map((id) => (
<li className='message-input__attachment' key={id}>
<ImageAttachmentPreview id={id} />
</li>
))}
{fileOrder.map((id) => (
<li key={id}>
<FileAttachmentPreview id={id} />
</li>
))}
</ul>
);
};
Note that an attachment can be in three different states:
'uploading'
means the attachment is still uploading; you may want to display a spinner in this case.'failed'
means that something went wrong while uploading the attachment; you may want to show an option to retry the upload.'finished'
means the attachment had been successfully uploaded.
For image attachments, the previewUri
property is available, which is a temporary URL that can be
used to display an image preview before the image is uploaded. Once the image is uploaded, this
temporary URL is no longer available, and url
should be used instead.
import { useMessageInputContext } from 'stream-chat-react';
const ImageAttachmentPreview = ({ id }) => {
const { imageUploads } = useMessageInputContext();
const image = imageUploads[id];
const url = image.previewUri ?? image.url ?? 'fallback.webm';
return (
<div
className='message-input__attachment-preview message-input__attachment-preview_image'
style={{ backgroundImage: `url(${url})` }}
aria-label={image.file.name}
>
<AttachmentActions attachment={image} type='image' />
</div>
);
};
const AttachmentActions = ({ attachment }) => {
const { removeImage, uploadImage, removeFile, uploadFile } = useMessageInputContext();
let [remove, upload] = type === 'image' ? [removeImage, uploadImage] : [removeFile, uploadFile];
let children = null;
if (attachment.state === 'uploading') {
children = <div className='message-input__attachment-action'>Loading...</div>;
}
if (attachment.state === 'finished') {
children = (
<>
<a
className='message-input__attachment-action'
href={attachment.url}
target='_blank'
rel='noreferrer'
>
📥
</a>
<button className='message-input__attachment-action' onClick={() => remove(attachment.id)}>
❌
</button>
</>
);
}
if (attachment.state === 'failed') {
<button className='message-input__attachment-action' onClick={() => upload(attachment.id)}>
Failed. Retry?
</button>;
}
return <div className='message-input__attachment-actions'>{children}</div>;
};
For generic file attachments, the file name (file.name
) and MIME type (file.type
) can be used to
display a preview:
import { useMessageInputContext } from 'stream-chat-react';
const FileAttachmentPreview = ({ id }) => {
const { fileUploads } = useMessageInputContext();
const attachment = fileUploads[id];
return (
<div className='message-input__attachment-preview message-input__attachment-preview_file'>
📄 {attachment.file.name} <br />({attachment.file.size} bytes)
<AttachmentActions attachment={attachment} type='file' />
</div>
);
};
Putting it all together, this is a complete implementation of the attachment preview component:
import { useMessageInputContext } from 'stream-chat-react';
const CustomAttachmentPreviewList = () => {
const { fileOrder, imageOrder } = useMessageInputContext();
if (fileOrder.length === 0 && imageOrder.length === 0) {
return null;
}
return (
<ul className='message-input__attachments'>
{imageOrder.map((id) => (
<li className='message-input__attachment' key={id}>
<ImageAttachmentPreview id={id} />
</li>
))}
{fileOrder.map((id) => (
<li key={id}>
<FileAttachmentPreview id={id} />
</li>
))}
</ul>
);
};
const ImageAttachmentPreview = ({ id }) => {
const { imageUploads } = useMessageInputContext();
const image = imageUploads[id];
const url = image.previewUri ?? image.url ?? 'fallback.webm';
return (
<div
className='message-input__attachment-preview message-input__attachment-preview_image'
style={{ backgroundImage: `url(${url})` }}
aria-label={image.file.name}
>
<AttachmentActions attachment={image} type='image' />
</div>
);
};
const FileAttachmentPreview = ({ id }) => {
const { fileUploads } = useMessageInputContext();
const attachment = fileUploads[id];
return (
<div className='message-input__attachment-preview message-input__attachment-preview_file'>
📄 {attachment.file.name} <br />({attachment.file.size} bytes)
<AttachmentActions attachment={attachment} type='file' />
</div>
);
};
const AttachmentActions = ({ type, attachment }) => {
const { removeImage, uploadImage, removeFile, uploadFile } = useMessageInputContext();
let [remove, upload] = type === 'image' ? [removeImage, uploadImage] : [removeFile, uploadFile];
let children = null;
if (attachment.state === 'uploading') {
children = <div className='message-input__attachment-action'>Loading...</div>;
}
if (attachment.state === 'finished') {
children = (
<>
<a
className='message-input__attachment-action'
href={attachment.url}
target='_blank'
rel='noreferrer'
>
📥
</a>
<button className='message-input__attachment-action' onClick={() => remove(attachment.id)}>
❌
</button>
</>
);
}
if (attachment.state === 'failed') {
<button className='message-input__attachment-action' onClick={() => upload(attachment.id)}>
Failed. Retry?
</button>;
}
return <div className='message-input__attachment-actions'>{children}</div>;
};
.message-input__attachments {
display: flex;
flex-wrap: wrap;
gap: 16px;
list-style: none;
padding: 0;
}
.message-input__attachment-preview {
position: relative;
border-radius: 8px;
width: 160px;
height: 160px;
padding: 8px;
border: 1px solid #00000014;
}
.message-input__attachment-preview_image {
background-size: cover;
background-position: center;
background-clip: border-box;
background-repeat: no-repeat;
}
.message-input__attachment-actions {
display: flex;
gap: 4px;
position: absolute;
left: 0;
bottom: 0;
padding: 8px;
}
.message-input__attachment-action {
font: inherit;
border-radius: 4px;
padding: 4px;
background: #fff;
border: 1px solid #00000014;
}
Displaying Link Previews
If URL enrichment is enabled both in channel settings (enabled by default) and in the
urlEnrichmentConfig
of the MessageInput
component (disabled by default), the SDK will automatically detect links in
the message text (as long as it’s set properly in the
MessageInputContext
) and create previews
for them.
To display link previews, you can use a pre-built
LinkPreviewList
component
available in the ComponentContext
. Using the
ComponentContext
allows you to hook into the component override mechanism used throughout the SDK.
So the rough idea is:
- Grab the
LinkPreviewList
from theComponentContext
(fall back to the default implementation if the component wasn’t overridden). - Grab the
linkPreviews
from theMessageInputContext
and pass them to theLinkPreviewList
.
The only thing to note here is that
linkPreviews
is a Map with
URLs as keys and enriched data as values. Before passing it to the LinkPreviewList
, we should
convert it to an array:
import {
useMessageInputContext,
useComponentContext,
AttachmentPreviewList as DefaultAttachmentPreviewList,
LinkPreviewList as DefaultLinkPreviewList,
} from 'stream-chat-react';
const CustomMessageInput = () => {
const { text, linkPreviews, handleChange, handleSubmit } = useMessageInputContext();
const {
LinkPreviewList = DefaultLinkPreviewList,
AttachmentPreviewList = DefaultAttachmentPreviewList,
} = useComponentContext();
return (
<div className='message-input'>
<LinkPreviewList linkPreviews={Array.from(linkPreviews.values())} />
<div className='message-input__composer'>
<AttachmentUploadButton />
<textarea className='message-input__input' value={text} onChange={handleChange} />
<button type='button' className='message-input__button' onClick={handleSubmit}>
⬆️
</button>
</div>
<AttachmentPreviewList />
</div>
);
};
As always, if you need deeper customization, you can implement the link preview component from
scratch. Since all the necessary data can be found in the
linkPreviews
value of the
MessageInputContext
, the implementation itself is quite simple.
The link preview itself goes through several lifecycle states:
'queued'
means that the URL enrichment process hasn’t started yet. This process is debounced, so to avoid flashing UI we should ignore queued previews.'loading'
means that URL enrichment is in progress. Depending on the desired UX, you can either ignore loading previews, or show a spinner or other loading state.'loaded'
means that the preview is ready.'dismissed'
means the preview has been dismissed by the user.
So the only bit of interactivity we need to add is an option to dismiss a link preview:
import { useMessageInputContext } from 'stream-chat-react';
const CustomLinkPreviewList = () => {
const { linkPreviews: linkPreviewMap, dismissLinkPreview } = useMessageInputContext();
const linkPreviews = Array.from(linkPreviewMap.values());
if (linkPreviews.length === 0) {
return null;
}
return (
<ul className='message-input__link-previews'>
{linkPreviews.map(
(preview) =>
preview.state === 'loaded' && (
<li key={preview.og_scrape_url} className='message-input__link-preview'>
<span className='message-input__link-preview-icon'>🔗</span>
<div className='message-input__link-preview-og'>
<strong>{preview.title}</strong>
<br />
{preview.text}
</div>
<button
type='button'
className='message-input__button'
onClick={() => dismissLinkPreview(preview)}
>
❌
</button>
</li>
),
)}
</ul>
);
};
.message-input__link-previews {
list-style: none;
padding: 0;
}
.message-input__link-preview {
display: flex;
align-items: center;
}
.message-input__link-preview-icon {
margin: 0 25px 0 19px;
}
.message-input__link-preview-og {
flex-grow: 1;
}
Handling Slow Mode
If the Slow Mode is configured for the current channel, we should prevent messages from being sent before the cooldown period has passed.
The first thing we should do is disable or hide the “send” button while the cooldown period is in
effect. We can use the
cooldownRemaining
value
from the MessageInputContext
: it’s guaranteed to have a non-zero numeric value while the user is
within a cooldown period, and to reset to a falsy value when the cooldown period ends. Note that to
avoid excessive rendering, the value itself does not “tick down” every second.
To provide visual feedback with a countdown, we can use the
CooldownTimer
component from the
ComponentContext
:
import {
useMessageInputContext,
useComponentContext,
AttachmentPreviewList as DefaultAttachmentPreviewList,
CooldownTimer as DefaultCooldownTimer,
} from 'stream-chat-react';
const CustomMessageInput = () => {
const {
text,
cooldownRemaining,
setCooldownRemaining,
cooldownInterval,
handleChange,
handleSubmit,
} = useMessageInputContext();
const {
CooldownTimer = DefaultCooldownTimer,
AttachmentPreviewList = DefaultAttachmentPreviewList,
} = useComponentContext();
return (
<div className='message-input'>
<div className='message-input__composer'>
<AttachmentUploadButton />
<textarea className='message-input__input' value={text} onChange={handleChange} />
<button
type='button'
className='message-input__button'
disabled={!!cooldownRemaining}
onClick={handleSubmit}
>
{cooldownRemaining ? (
<>
⏳{' '}
<CooldownTimer
cooldownInterval={cooldownInterval}
setCooldownRemaining={setCooldownRemaining}
/>
</>
) : (
<>⬆️</>
)}
</button>
</div>
<AttachmentPreviewList />
</div>
);
};
The component itself is very simple, it’s just a single <div>
with a timer that counts down
seconds. Most customizations can be done with CSS, but if you need to implement the whole component
from scratch, the only trick is to add a timer that counts down from the
cooldownInterval
:
const CustomCooldownTimer = () => {
const { cooldownRemaining, cooldownInterval } = useMessageInputContext();
const enabled = !!cooldownRemaining;
const [secondsPassed, setSecondsPassed] = useState(0);
useEffect(() => {
const startedAt = Date.now();
const interval = setInterval(() => {
setSecondsPassed(Math.floor((Date.now() - startedAt) / 1000));
}, 500);
return () => clearInterval(interval);
}, [enabled]);
return <strong>⏳ {cooldownInterval - secondsPassed}</strong>;
};