- 🎯 ContentEditable Architecture - Modern implementation with rich text support
- 🧠 Smart Mention Detection - DOM-aware detection that distinguishes mentions from regular text
- ⚡ Zero Dependencies - Lightweight with no external dependencies
- ♿ Fully Accessible - Complete ARIA roles and keyboard navigation
- 🎨 Highly Customizable - Slot-based customization system
- 🔧 TypeScript Support - Full type safety out of the box
- 📱 Flexible Triggers - Customizable trigger characters or strings
- 🎪 Rich Text Support - Display mentions as styled chips
- 🚀 Function Values - Support for executable functions as option values
- 📋 Advanced Paste Handling - Intelligent mention parsing from pasted content
- 🔄 Auto-Conversion - Optional automatic conversion of text mentions to chips
- ⌨️ Custom Keyboard Handling - Support for custom keyboard events and form submission
- 💾 Data Value Support - Programmatic mention reconstruction from data values
npm install mentisimport { MentionInput } from "mentis";
import "mentis/dist/index.css";
function App() {
const [dataValue, setDataValue] = useState("");
return (
<MentionInput
dataValue={dataValue}
onChange={(mentionData) => {
setDataValue(mentionData.dataValue);
}}
options={[
{ label: "Alice Johnson", value: "alice" },
{ label: "Bob Smith", value: "bob" },
{ label: "Charlie Brown", value: "charlie" },
]}
/>
);
}import { MentionInput } from "mentis";
function BasicExample() {
return (
<MentionInput
options={[
{ label: "Alice Johnson", value: "alice" },
{ label: "Bob Smith", value: "bob" },
{ label: "Charlie Brown", value: "charlie" },
]}
onChange={(mentionData) => console.log(mentionData)}
/>
);
}import { MentionInput } from "mentis";
function FunctionValueExample() {
return (
<MentionInput
options={[
{ label: "Send Message", value: () => console.log("Message sent!") },
{ label: "Clear Input", value: () => setValue("") },
{ label: "Alice Johnson", value: "alice" },
]}
onChange={(mentionData) => console.log(mentionData)}
/>
);
}import { MentionInput } from "mentis";
function StyledExample() {
return (
<MentionInput
options={options}
slotsProps={{
container: {
className: "w-full max-w-lg relative",
},
contentEditable: {
className:
"w-full rounded-xl border border-gray-300 bg-white px-4 py-3 text-base shadow-sm focus:border-blue-500 focus:ring-2 focus:ring-blue-200 outline-none transition placeholder-gray-400",
},
modal: {
className:
"absolute z-10 mt-2 w-full bg-white border border-gray-200 rounded-xl shadow-lg max-h-60 overflow-auto",
},
option: {
className:
"px-4 py-2 cursor-pointer text-base text-gray-800 hover:bg-gray-100 hover:text-black rounded-lg transition",
},
highlightedClassName: "bg-blue-500 text-white hover:bg-blue-500",
chipClassName: "bg-blue-500 text-white hover:bg-blue-500",
}}
/>
);
}import { MentionInput } from "mentis";
function CustomTriggerExample() {
return (
<MentionInput
trigger="#"
options={[
{ label: "React", value: "react" },
{ label: "TypeScript", value: "typescript" },
{ label: "JavaScript", value: "javascript" },
]}
/>
);
}import { MentionInput } from "mentis";
function AutoConvertExample() {
return (
<MentionInput
autoConvertMentions={true}
keepTriggerOnSelect={false}
options={[
{ label: "Alice Johnson", value: "alice" },
{ label: "Bob Smith", value: "bob" },
]}
/>
);
}import { MentionInput } from "mentis";
function FormSubmissionExample() {
const [value, setValue] = useState("");
const handleSubmit = () => {
console.log("Submitting:", value);
setValue("");
};
return (
<MentionInput
value={value}
onChange={(mentionData) => setValue(mentionData.value)}
onKeyDown={(event) => {
// Handle Enter key for form submission
if (event.key === "Enter") {
event.preventDefault();
handleSubmit();
}
}}
options={[
{ label: "Alice Johnson", value: "alice" },
{ label: "Bob Smith", value: "bob" },
]}
/>
);
}import { MentionInput } from "mentis";
function KeyboardShortcutsExample() {
return (
<MentionInput
onKeyDown={(event) => {
// Ctrl/Cmd + Enter to submit
if ((event.ctrlKey || event.metaKey) && event.key === "Enter") {
event.preventDefault();
handleSubmit();
}
// Ctrl/Cmd + S to save
if ((event.ctrlKey || event.metaKey) && event.key === "s") {
event.preventDefault();
saveContent();
}
}}
options={options}
/>
);
}| Prop | Type | Default | Description |
|---|---|---|---|
options |
MentionOption[] |
- | Array of mention options |
displayValue |
string |
"" |
Current display value of the input (what user sees) |
dataValue |
string |
- | Data value for programmatic control (mention IDs) |
onChange |
(value: MentionData) => void |
- | Callback when value changes with mention data |
trigger |
string |
"@" |
Character(s) that trigger the mention dropdown |
keepTriggerOnSelect |
boolean |
true |
Whether to keep the trigger character after selection |
autoConvertMentions |
boolean |
false |
Automatically convert mentions to chips |
onKeyDown |
(event: KeyboardEvent) => void |
- | Custom keyboard event handler |
slotsProps |
SlotProps |
- | Customization props for different parts |
type MentionOption = {
label: string; // Display text
value: string | Function; // Unique identifier or executable function
};type MentionData = {
displayValue: string; // Text as displayed to user (with mention labels)
dataValue: string; // Text with mention values (actual data)
mentions: Array<{
label: string; // Display text of the mention
value: string; // Unique identifier of the mention
startIndex: number; // Start position in the text
endIndex: number; // End position in the text
}>;
};type SlotProps = {
container?: React.HTMLAttributes<HTMLDivElement>;
contentEditable?: ContentEditableInputCustomProps;
modal?: ModalProps;
option?: OptionProps;
noOptions?: React.HTMLAttributes<HTMLDivElement>;
highlightedClassName?: string;
chipClassName?: string;
};- Arrow Keys: Navigate through mention options
- Enter: Select highlighted option
- Escape: Close mention dropdown
- Tab: Navigate through options and select
- Backspace: Navigate into mention chips
The onKeyDown prop allows you to handle custom keyboard events:
- Form Submission: Handle Enter key for form submission when the modal is closed
- Keyboard Shortcuts: Implement custom shortcuts like Ctrl+S for save
- Event Handling: The component's internal handling (navigation, selection) takes precedence over custom handlers
Note: When the mention modal is open, Enter, Tab, Escape, and arrow keys are handled internally for navigation and selection.
The dataValue prop enables powerful programmatic control over mention content:
- Setting Content: Pass
dataValue="alice bob"to programmatically load mentions - Clean Data Extraction:
onChangeprovides clean data values (IDs) separate from display text - Mention Reconstruction: Automatically converts data values back to visual mentions
- AI Integration: Perfect for sending clean data to APIs while showing rich UI to users
- Editing Support: Load existing content with mentions for editing workflows
// Set mentions programmatically
setDataValue("user-123 user-456"); // Shows "@John Doe @Jane Smith"
// Get clean data for APIs
onChange={(data) => {
console.log(data.displayValue); // "@John Doe hello @Jane Smith"
console.log(data.dataValue); // "user-123 hello user-456"
sendToAPI(data.dataValue); // Send clean data to backend
}}Options can have function values that execute when selected, useful for actions like sending messages or clearing input.
When autoConvertMentions is enabled, the component automatically converts text mentions to chips when users type space or press Enter.
The component intelligently parses mentions from pasted content, converting them to chips automatically.
Mentions are displayed as styled chips within the contentEditable interface, providing a rich text experience.
Explore complete examples in the following directories:
MIT © Alexander Dunlop
