Empowering Web3 UX through Push Protocol's Notification System

A Step-by-Step Guide to Implementing Web3 Notifications

Empowering Web3 UX through Push Protocol's Notification System

Welcome to our step-by-step guide on integrating Push Protocol's notification system into your NextJS web application. In this blog, we'll cover the process of initializing a user, creating a notification channel, and implementing the APIs for sending and receiving notifications.

Initializing the User

Let's start by initializing the user with the following code snippet:

import { PushAPI, CONSTANTS } from "@pushprotocol/restapi";

const userAlice = await PushAPI.initialize(signer, {
  env: CONSTANTS.ENV.STAGING,
});

This code seamlessly initializes the user, creating a Push user profile object. If the profile doesn't exist, it is promptly generated, laying the foundation for subsequent function calls within the Push network.

Create Channel API

Next, let's create a channel using the provided code:

// userAlice.channel.create({options})
const response = await userAlice.channel.create({
  name: "Test Channel",
  description: "Test Description",
  icon: "",
  url: "https://push.org",
});

This API is crucial for establishing a communication channel, allowing the system to send notifications. Pay attention to the parameters such as name, description, icon (base64 encoded), and URL, as they define the channel's characteristics.

NextJS Integration with TypeScript

To seamlessly integrate these functionalities into a NextJS frontend with TypeScript, we have a React component named Channel. This component serves as a unified form for creating a channel. You can follow along with the provided code, adjusting the types based on your application needs.

// @ts-nocheck
import React, { useState, useEffect } from 'react';
import { PushAPI, CONSTANTS } from "@pushprotocol/restapi";


interface ChannelProps {
  signer: any; // Adjust the type of signer based on your application
}

const Channel: React.FC<ChannelProps> = ({ signer }) => {
  // State to hold form data
  const [formData, setFormData] = useState({
    name: '',
    description: '',
    url: '',
    icon: ''
    // delegateAddresses: null,
  });

  // State to    added delegates
  // const [addedDelegates, setAddedDelegates] = useState([]);

  // Function to handle form input changes
  const handleInputChange = (e) => {
    setFormData({
      ...formData,
      [e.target.name]: e.target.value,
    });
  };



  // Function to handle form submission
  const handleSubmit = async () => {
    const push_signer = await PushAPI.initialize(signer, { env: CONSTANTS.ENV.STAGING });
    try {
      if (signer) {
        const response = await push_signer.channel.create({
          name: formData.name,
          description: formData.description,
          url: formData.url,
          icon: formData.icon
        });
        console.log("Channel Created", response);
        // Rest of your code...
      } else {
        console.error('Signer or signer.channel.create is undefined');
      }
    } catch (error) {
      // Handle errors here
      console.error(error);
    }
  };

  // useEffect to log the form data whenever it changes (for debugging purposes)
  useEffect(() => {
    console.log('Form Data:', formData);
  }, [formData]);

  useEffect(() => { handleSubmit }, [signer]);

  // useEffect to log the added delegates whenever it changes (for debugging purposes)
  // useEffect(() => {
  //   console.log('Added Delegates:', addedDelegates);
  // }, [addedDelegates]);

  return (
    <div className="flex flex-col items-center">
      <label className="mb-2" htmlFor="name">
        Channel Name:
      </label>
      <input
        type="text"
        id="name"
        name="name"
        value={formData.name}
        onChange={handleInputChange}
        className="border p-2 mb-4"
      />

      <label className="mb-2" htmlFor="description">
        Channel Description:
      </label>
      <textarea
        id="description"
        name="description"
        value={formData.description}
        onChange={handleInputChange}
        className="border p-2 mb-4"
      />

      <label className="mb-2" htmlFor="url">
        URL
      </label>
      <textarea
        id="url"
        name="url"
        value={formData.url}
        onChange={handleInputChange}
        className="border p-2 mb-4"
      />

      <label className="mb-2" htmlFor="icon">
        ICON
      </label>
      <textarea
        id="icon"
        name="icon"
        value={formData.icon}
        onChange={handleInputChange}
        className="border p-2 mb-4"
      />


      {/* <label className="mb-2" htmlFor="delegateAddresses">
                Delegate Addresses (comma-separated):
              </label>
              <input
                type="text"
                id="delegateAddresses"
                name="delegateAddresses"
                value={formData.delegateAddresses}
                onChange={handleInputChange}
                className="border p-2 mb-4"
              /> */}

      <button
        onClick={handleSubmit}
        className="bg-blue-500 text-white p-2 rounded hover:bg-blue-700"
      >
        Save
      </button>
    </div>
  );
};

export default Channel;

Sending Notifications

Now, let's explore the APIs for sending notifications:

// @ts-nocheck
import React, { useState } from 'react';

const NotificationSender = ({ onNotificationSent }) => {
    const [notificationBody, setNotificationBody] = useState('');

    const handleInputChange = (e) => {
        setNotificationBody(e.target.value);
    };

    const handleSendNotification = async () => {
        try {
            const sendNotifRes = await userAlice.channel.send(["*"], {
                notification: { title: "test", body: notificationBody },
            });

            onNotificationSent(sendNotifRes);

            setNotificationBody('');
        } catch (error) {
            console.error(error);
        }
    };

    return (
        // Render the notification sending form
    );
};

export default NotificationSender;

This component allows users to input a notification body and send it through the Push protocol. The onNotificationSent callback is invoked to notify the parent component.

Viewing Notifications

Finally, let's implement the API for fetching notifications:

import React, { useState } from 'react';

const NotificationDisplay = ({ notification }) => {
    const [shownNotification, setShownNotification] = useState(notification);

    // useEffect to update shownNotification whenever a new notification is received

    return (
        // Render the notification display component
    );
};

export default NotificationDisplay;

Surprise code

The provided code exemplifies a sophisticated notification system integrated seamlessly into a React application. Leveraging the capabilities of Push Protocol's notification system, this solution empowers users to receive real-time notifications in a Web3 environment.

The implementation involves initializing a user through the Ethereum provider, creating a channel, and establishing a connection to the notification stream. As the stream connects, a simulated notification is sent, showcasing the system's capability to capture, process, and display notifications. The progress information dynamically updates, providing users with valuable insights into the notification process.

Moreover, the code incorporates the UIWeb/NotificationItem component to elegantly render notifications. The NotificationInterface function encapsulates the user interface, offering a user-friendly experience by displaying wallet-specific notification items.

This solution serves as a powerful tool for developers seeking to enhance user engagement by integrating a robust and efficient notification system into their Web3 applications. With its comprehensive features, including real-time updates, streamlined UI rendering, and clear progress tracking, this code is poised to elevate the overall user experience in the Web3 ecosystem.


import { useState, useEffect, useRef } from "react";
import { ethers } from "ethers";
import { PushAPI, CONSTANTS } from "@pushprotocol/restapi";
import { NotificationItem, chainNameType } from "@pushprotocol/uiweb";

interface NotificationItemData {
  cta: string;
  title: string;
  message: string;
  app: string;
  icon: string;
  image: string;
  url: string;
  blockchain: string;
  notification: string;
}

interface NotifItem extends NotificationItemData {
  key: number;
}

export function Notification2(props: any) {
  const [wallet, setWallet] = useState<string>(
    "0xD8634C39BBFd4033c0d3289C4515275102423681"
  );
  const [progressTexts, setProgressTexts] = useState<string[]>([]);
  const [notifItems, setNotifItems] = useState<NotifItem[]>([]);

  const walletRef = useRef<HTMLInputElement | null>(null);

  useEffect(() => {
    if (walletRef.current) {
      walletRef.current.value = wallet;
    }
  }, [wallet]);

  const triggerNotification = async () => {
    // Demo only supports MetaMask (or other browser based wallets) and gets provider that injects as window.ethereum into each page
    const provider = new ethers.providers.Web3Provider(window.ethereum);

    // Switch to sepolia
    await provider.send("wallet_switchEthereumChain", [{ chainId: "0xAA36A7" }]);

    // Get provider
    await provider.send("eth_requestAccounts", []);

    // Grabbing signer from provider
    const signer = provider.getSigner();

    // Initialize user for push
    const userAlice = await PushAPI.initialize(signer, { env: CONSTANTS.ENV.STAGING });

    // establish connection to stream
    const stream = await userAlice.initStream([CONSTANTS.STREAM.CONNECT, CONSTANTS.STREAM.NOTIF]);

    // Listen for stream connection
    stream.on(CONSTANTS.STREAM.CONNECT, async (data) => {
      console.log("STREAM CONNECTED");
      let text = ['Stream Connected...', 'Sending Simulated Notification...', 'Wait for few moments for stream to capture notif and display...', 'Waiting for you to sign notification payload...'];
      setProgressTexts(text);
      await userAlice.channel.send([userAlice.account], {
        notification: { 
          title: 'GM Builders!', 
          body: `_Simulated notification_ listened by stream and rendered with **@UIWeb/NotificationItem** with the latest timestamp - ${new Date().valueOf()} [timestamp: ${new Date().valueOf()}]` 
        },
        payload: {
          title: 'GM Builders!', 
          body: `_Simulated notification_ listened by stream and rendered with **@UIWeb/NotificationItem** with the latest timestamp - ${new Date().valueOf()} [timestamp: ${new Date().valueOf()}]`,
          cta: 'https://push.org',
          embed: 'https://push.org/assets/images/cover-image-8485332aa8d3f031e142a1180c71b341.webp',
        }
      });
      text.push('Message generated and sent. Waiting for the stream to pick it up...');
      setProgressTexts(text);
    });

    // Listen for notifications
    stream.on(CONSTANTS.STREAM.NOTIF, (item) => {
      let text = progressTexts;
      console.log(item);
      text.push('Notification Received...');
      text.push(JSON.stringify(item));
      setProgressTexts(text);

      // create notification item compatible with UIWeb/NotificationItem
      const compatibleNotifItem = {
        title: item.message.payload.title,
        message: item.message.payload.body,
        image: item.message.payload.embed,
        cta: item.message.payload.cta,
        icon: item.channel.icon,
        app: item.channel.name,
        url: item.channel.url,
        blockchain: item.source,
        notification: item.message.notification,
      };
      setNotifItems([compatibleNotifItem]);
    });

    // connect stream
    stream.connect();
  };

  function NotificationInterface() {
    const inputStyle = {
      padding: "10px",
      margin: "10px 0",
      width: "100%",
      boxSizing: "border-box",
    };

    const textareaStyle = {
      ...inputStyle,
      height: "100px",
      resize: "vertical",
    };

    const buttonStyle = {
      padding: "10px 20px",
      backgroundColor: "#dd44b9",
      color: "#FFF",
      border: "none",
      borderRadius: "5px",
      cursor: "pointer",
      marginTop: "20px",
    };

    return (
      <div style={{ width: "auto", margin: "20px auto" }}>
        <div style={{ display: "flex", justifyContent: "space-between" }}>
          <div style={{ flex: 1 }}>
            <h2>
              Push based mechanism for displaying notifications on frontend
            </h2>
            <p />
            <label>
              Put any wallet address and click on fetch notifications to see the
              live results. Click to expand <b>Live Editor</b> tab to see the
              code and play with it. For this demo, You will need Metamask (or
              equivalent browser injected wallet), you will also need to sign a
              transaction to see the notifications.
            </label>
            <p />
          </div>
        </div>
        <div>
          <hr />
          <h3>Progress (will show progress information once Trigger Notification is clicked)</h3>

          {progressTexts.map((text, idx) => {
            return (
              <React.Fragment key={idx}>
                <span>{text}</span>
                <br />
              </React.Fragment>
            );
          })}
        </div>
        <hr />
        <button style={buttonStyle} onClick={triggerNotification}>
          Trigger Notification
        </button>

        <p />
        <p />

        {notifItems.length > 0 ? (
          <h3>{`Notification Items for ${wallet}`}</h3>
        ) : (
          <></>
        )}

        {notifItems.map((notifItemSingular, idx) => {
          const {
            cta,
            title,
            message,
            app,
            icon,
            image,
            url,
            blockchain,
            notification,
          } = notifItemSingular;

          return (
            <NotificationItem
              key={idx} // any unique id
              notificationTitle={title}
              notificationBody={message}
              cta={cta}
              app={app}
              icon={icon}
              image={image}
              url={url}
              theme={"light"} // or can be dark
              chainName={blockchain as chainNameType} // if using Typescript
            />
          );
        })}
      </div>
    );
  }

  return (
    <>
      <NotificationInterface />
    </>
  );
}

export default Notification2;

Conclusion

In conclusion, the incorporation of the Notification2 component results in a refined and structured presentation of received notifications, significantly augmenting the overall user experience. This component plays a pivotal role in streamlining the notification handling process within your NextJS web application.

To sum up, by adhering to this guide and seamlessly integrating the provided code snippets, you can empower your NextJS web application with a resilient notification system. This not only enriches your application's functionality but also ensures a seamless and engaging user experience within the dynamic landscape of Web3 environments.


Thank you for exploring the implementation of a powerful notification system in your NextJS web application through this blog. If you found the insights valuable, please consider sharing it with your network. For more tech-related content and updates, follow me on Hashnode. Your support is appreciated!

Cheers

Shaikh Rumman Fardeen

Push Protocol Ambassador

Follow me:

Linkedin : https://www.linkedin.com/in/srummanf

Github : https://github.com/srummanf

Twitter : https://twitter.com/srummanf