Glrk UI

Popover

A simplified wrapper for Shadcn Popover with flexible option handling and automatic type conversion

Installation

npx shadcn@latest add @glrk-ui/popover

If you haven't set up the prerequisites yet, check out Prerequest section.

Copy and paste the following code into shadcn popover component.

ui/popover.tsx
type PopoverWrapperProps = {
  trigger: React.ReactNode
  content: React.ReactNode
  triggerCls?: string
  contentCls?: string
  contentProps?: Omit<React.ComponentProps<typeof PopoverPrimitive.Content>, "className">
} & Omit<React.ComponentProps<typeof PopoverPrimitive.Root>, "children">

function PopoverWrapper({
  trigger,
  content,
  triggerCls,
  contentCls,
  contentProps,
  ...props
}: PopoverWrapperProps) {
  return (
    <Popover {...props}>
      <PopoverTrigger
        className={triggerCls}
        asChild={typeof trigger !== "string"}
      >
        {trigger}
      </PopoverTrigger>

      <PopoverContent {...contentProps} className={contentCls}>
        {content}
      </PopoverContent>
    </Popover>
  )
}

Updated PopoverWrapper in the export block.

export {
  Popover,
  PopoverTrigger,
  PopoverContent,
  PopoverProvider,
  PopoverWrapper,
}

Installation

npm install @radix-ui/react-popover

popover.tsx

Copy and paste the following code into your project.

ui/popover.tsx
"use client"

import * as React from "react"
import * as PopoverPrimitive from "@radix-ui/react-popover"

import { cn } from "@/lib/utils"

function Popover(props: React.ComponentProps<typeof PopoverPrimitive.Root>) {
  return <PopoverPrimitive.Root data-slot="popover" {...props} />
}

function PopoverTrigger(props: React.ComponentProps<typeof PopoverPrimitive.Trigger>) {
  return <PopoverPrimitive.Trigger data-slot="popover-trigger" {...props} />
}

function PopoverContent({
  className,
  align = "center",
  sideOffset = 4,
  ...props
}: React.ComponentProps<typeof PopoverPrimitive.Content>) {
  return (
    <PopoverPrimitive.Portal>
      <PopoverPrimitive.Content
        data-slot="popover-content"
        align={align}
        sideOffset={sideOffset}
        className={cn(
          "bg-popover text-popover-foreground data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 data-[side=bottom]:slide-in-from-top-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 z-50 w-72 origin-(--radix-popover-content-transform-origin) rounded-md border p-4 shadow-md outline-hidden",
          className
        )}
        {...props}
      />
    </PopoverPrimitive.Portal>
  )
}

function PopoverAnchor(props: React.ComponentProps<typeof PopoverPrimitive.Anchor>) {
  return <PopoverPrimitive.Anchor data-slot="popover-anchor" {...props} />
}

type PopoverWrapperProps = {
  trigger: React.ReactNode
  content: React.ReactNode
  triggerCls?: string
  contentCls?: string
  contentProps?: Omit<React.ComponentProps<typeof PopoverPrimitive.Content>, "className">
} & Omit<React.ComponentProps<typeof PopoverPrimitive.Root>, "children">

function PopoverWrapper({
  trigger,
  content,
  triggerCls,
  contentCls,
  contentProps,
  ...props
}: PopoverWrapperProps) {
  return (
    <Popover {...props}>
      <PopoverTrigger
        className={triggerCls}
        asChild={typeof trigger !== "string"}
      >
        {trigger}
      </PopoverTrigger>

      <PopoverContent {...contentProps} className={contentCls}>
        {content}
      </PopoverContent>
    </Popover>
  )
}

export {
  Popover,
  PopoverTrigger,
  PopoverContent,
  PopoverAnchor,
  PopoverWrapper,
}

Done

You can now use PopoverWrapper

Usage

Basic

import { PopoverWrapper } from "@/components/ui/popover";

export function Basic() {
  return (
     <PopoverWrapper
      trigger="Hover"
      content="Some popover content"
    />
  )
}

Reference

PopoverWrapper

Prop

Type