React hooks for developing Nostr clients. It's simple yet intelligent.
Nostr-Hooks is a stateful wrapper library of React hooks around NDK, designed to simplify the process of interacting with the Nostr protocol in real-time web applications. It provides an easy-to-use interface with low bandwidth consumption and high performance, allowing developers to quickly integrate Nostr into their projects and build responsive, real-time applications.
I knew that replacing Zustand with React Context API was a bad idea. So, I decided to revert it back to Zustand. This means that you no longer need to wrap your application with the NostrHooksContextProvider
component.
Now you just need to initialize NostrHooks in your root component with just a simple hook.
import { useNostrHooks } from 'nostr-hooks';
const App = () => {
useNostrHooks();
return <YourApp />;
};
Nostr-Hooks v2 is a major release.
-
It replaces the
Zustand
store with theReact Context API
. This means that now you need to wrap your application with theNostrHooksContextProvider
component. -
It replaces
nostr-tools
withnostr-dev-kit (NDK)
. This means that most of the functionalities like caching, batching, and merging filters are now handled by NDK and Nostr-Hooks is only responsible for managing the component state and subscriptions.
npm install nostr-hooks
- Provides a single instance of Nostr pool for the entire application, which is reused by all components.
- Creates a single connection to each Nostr relay at a time and reuses it for all subscriptions, reducing network overhead.
- Automatically manages subscriptions from multiple components and delivers only the events that each component needs.
- Automatically batches multiple subscriptions from different components into a single subscription request, further reducing network overhead.
- Intelligently merges filters into a unique set of filters, reducing the load on the Nostr relays.
- Provides a built-in cache mechanism since version 1.1.
- Minimizes re-renders by updating only the events that have changed, improving application performance.
- Automatically cleans up subscriptions and garbage events when a component unmounts, preventing memory leaks.
Nostr-Hooks is not a replacement for NDK. You may still need to install NDK and use it in your application.
NDK is a powerful library (shout-out to pablo) with a lot of out-of-the-box features, like caching, batching, and merging filters. However, it's a stateless library and doesn't understand the React component lifecycle. This means that it's up to the developer to update the component state when new events arrive, and to unsubscribe from the subscription when the component unmounts. This can be a tedious and error-prone process, especially when scaling the application. Nostr-Hooks on the other hand, is a stateful wrapper library that manages the component state and subscriptions automatically, allowing the developer to focus on building and scaling the application.
You need to initialize NostrHooks in your root component in order to execute ndk.connect()
automatically and create a single instance of Nostr pool for the entire application.
import { useNostrHooks } from 'nostr-hooks';
const App = () => {
useNostrHooks();
return <YourApp />;
};
You can also pass a custom NDK instance to the
useNostrHooks
hook. This is useful when you want to initiate your app with a custom NDK instance with your own configuration. You can also use other provided hooks likeuseNdk
to interact with the NDK instance later.
Here are some examples of how to use the useSubscribe
hook:
import { useSubscribe } from 'nostr-hooks';
const MyComponent = () => {
const { events } = useSubscribe({
filters: [{ authors: ['pubkey1'], kinds: [1] }],
});
if (!events) return <p>Loading...</p>;
return (
<ul>
{events.map((event) => (
<li key={event.id}>
<p>{event.pubkey}</p>
<p>{event.kind}</p>
</li>
))}
</ul>
);
};
The useSubscribe
hook takes an object with one mandatory and two optional parameters:
-
filters
: A mandatory array of filters that the subscription should be created for. -
enabled
: An optional boolean flag indicating whether the subscription is enabled. If set tofalse
, the subscription will not be created automatically. -
opts
: An optional "NDK Subscription Options" object. -
relays
: An optional array of relay urls to use for the subscription. If not provided, the default relays will be used. -
fetchProfiles
: An optional boolean flag indicating whether to fetch profiles for the events in the subscription. If set totrue
, the profiles will be fetched automatically.
There are lots of options available for creating a subscription. Read more about the NDK subscription options here
The useSubscribe
hook returns an object with four properties:
-
events
: An array of events that match the filters. -
eose
: A boolean flag indicating whether the subscription has reached the end of the stream. -
unSubscribe
: A function that can be used to unsubscribe from the subscription. -
isSubscribed
: A boolean flag indicating whether the subscription is active.
import { useSubscribe } from 'nostr-hooks';
const MyComponent = () => {
const { events: articles } = useSubscribe({
filters: [{ authors: ['pubkey'], kinds: [30023] }],
});
const { events: notes } = useSubscribe({
filters: [{ authors: ['pubkey'], kinds: [1] }],
});
return (
<>
<ul>
{articles.map((article) => (
<li key={article.id}>
<p>{article.pubkey}</p>
<p>{article.content}</p>
</li>
))}
</ul>
<ul>
{notes.map((note) => (
<li key={note.id}>
<p>{note.pubkey}</p>
<p>{note.content}</p>
</li>
))}
</ul>
</>
);
};
The useSubscribe
hook can be used multiple times in a single component. Nostr-Hooks batches all subscriptions into a single subscription request, and delivers only the events that each hook needs.
import { useSubscribe } from 'nostr-hooks';
const App = () => {
return (
<>
<ComponentA />
<ComponentB />
</>
);
};
const ComponentA = () => {
const { events } = useSubscribe({
filters: [{ authors: ['pubkey'], kinds: [1] }],
});
return (
<ul>
{events.map((event) => (
<li key={event.id}>
<p>{event.pubkey}</p>
<p>{event.kind}</p>
</li>
))}
</ul>
);
};
const ComponentB = () => {
const { events } = useSubscribe({
filters: [{ authors: ['pubkey'], kinds: [30023] }],
});
return (
<ul>
{events.map((event) => (
<li key={event.id}>
<p>{event.pubkey}</p>
<p>{event.content}</p>
</li>
))}
</ul>
);
};
The useSubscribe
hook can be used in multiple components. Nostr-Hooks batches all subscriptions from all components into a single subscription request, and delivers only the events that each component needs.
import { useSubscribe } from 'nostr-hooks';
const MyComponent = ({ noteId }: Params) => {
const { events } = useSubscribe({
filters: [{ ids: [noteId] }],
enabled: !!noteId,
});
return (
<>
<ul>
{events.map((event) => (
<li key={event.id}>
<p>{event.pubkey}</p>
<p>{event.content}</p>
</li>
))}
</ul>
</>
);
};
The useSubscribe
hook can be used in a component that depends on a prop or state. In this example, the subscription waits for the noteId
prop to be set before creating the subscription.
The useNewEvent
hook is used to create a new NDK event, which can then be published using the internal publish
method.
import { useNewEvent } from 'nostr-hooks';
const MyComponent = () => {
const [content, setContent] = useState('');
const { createNewEvent } = useNewEvent();
const handlePublish = () => {
const event = createNewEvent();
event.content = content;
event.kind = 1;
event.publish();
};
return (
<>
<input type="text" value={content} onChange={(e) => setContent(e.target.value)} />
<button onClick={() => handlePublish()}>Publish Note</button>
</>
);
};
There is also a
usePublish
hook that can be used to publish an existing NDK event.
The useProfiles
hook is used to fetch profiles for a given set of events or users.
The default behavior is to mutate the original events or users with the fetched profiles. To prevent this, you can use the
mutateOrignal
option and set it tofalse
. In this case, the updated events or users will be returned from theuseProfiles
hook, and you can use them to render the UI.
Consider a scenario where you have a list of events, and you want to fetch the profiles of the authors of those events.
const MyComponent = () => {
const { events } = useSubscribe({ filters });
useProfiles({ events });
return (
<ul>
{events.map((event) => (
<li key={event.id}>
<p>{event.author.profile?.name}</p>
<p>{event.author.profile?.bio}</p>
</li>
))}
</ul>
);
};
The useProfiles
hook will automatically fetch the profiles of the authors of the events, and mutate the original events with the fetched profiles. This means that the author
property of each event will be updated with the fetched profile.
You can leverage useNdk
hook to interact with the NDK instance. it returns the NDK instance itself, and two setter functions for updating the NDK instance and the NDK signer.
import { useNdk } from 'nostr-hooks';
const MyComponent = () => {
const { ndk, setNdk, setSigner } = useNdk();
const handleUpdateNdk = ({ ndk }: NDK) => {
setNdk(new NDK({ /* ... */ })); // this will replace the existing NDK instance with the new one
};
const handleUpdateNdkSigner = ({ signer }: NDKSigner) => {
setSigner(new NDKNip07Signer()); // this will keep the existing NDK instance and update its signer
};
};
You can use the useActiveUser
hook to get the active user's profile based on the current NDK instance and its signer.
import { useActiveUser } from 'nostr-hooks';
const MyComponent = () => {
const { activeUser } = useActiveUser();
return (
<div>
<p>{activeUser?.profile?.name}</p>
</div>
);
};
You can use the useNip07
hook to update the current NDK instance with the NIP-07 browser extension's signer.
This hook will automatically update the existing NDK instance
with the signer from the NIP-07 browser extension, and will prompt the user to connect to the NIP-07 browser extension if they haven't already.
import { useNip07 } from 'nostr-hooks';
const MyComponent = () => {
useNip07();
// ...
};
You can use this hook in the root component of your application for the entire application, or you can use it in a specific component where you need the user pubkey. This will update the NDK instance in the entire application.
We welcome contributions from the community! If you'd like to contribute to Nostr-Hooks, please refer to the CONTRIBUTING.md file in the project's GitHub repository.
You can also consider contributing to NDK.
If you'd like to support the development of Nostr-Hooks, please consider donating to the developer.
- ⚡ Zap sats to sepehr@getalby.com
You can also consider supporting the NDK.
Nostr-Hooks is licensed under the MIT License. For more information, see the LICENSE.md file in the project's GitHub repository.
If you have any questions or concerns about Nostr-Hooks, please contact the developer at npub18c556t7n8xa3df2q82rwxejfglw5przds7sqvefylzjh8tjne28qld0we7.