* update deps + image codemod (studio) * update next links (studio) * update deps * update links (ui) * remove next-transpile-modules * move next-themes dependency * chore: update ConfirmDialog * chore: remove old ConfirmModal js file. migrated to TS * dependency wrangling * remove empty page * update next links (www) * First run bump react-data-grid-v7 beta 4 * fix package-lock.json * more deps wrangling * update recharts * update sentry options * fix some broken things in www * studio fixes * fix graphiql * fix studio build * fix menu hydration * small build error * update turbo * fix www typescript errors * docs image codemod * links codemod docs * fix docs typescript errors * move useConsent to ui to prevent circular deps * Fix links * Fix homepage * Fix links * move studio/ to apps/ * Revert "move studio/ to apps/" This reverts commit 1b0a985fcb7569f29c8a6dd05b9c3063152547b9. * disable outputFileTracingRoot * remove outputFileTracingRoot * fix homepage product cards * fix PrivacySettings links * Fix links * Fix the build for www. * Minor fixes for JWTGenerator. * Fix the docs and ui tests. * Revert codehike back to 0.8.3 * remove ConfirmAlert() * reenable babel because mobx hates me * fix blog image and comparison page avatar * Fix svg errors * update image synthax * Fix code hike * Move the button in a div so that it doesn't inherit its parent height and make the button look weird. * When components are defined in a component, they get recreated on each render. This makes them unstable in certain cases and causes infinite rerenders. * Replace the next/head usage with next/script. * Chore/upgrade next 13 fix table editor (#18431) * fix table editor styling and fix row deletion logic * Fix deleting selected rows from header, and fix checkboxes not clearing up * Fix deleting all rows when filter applied, and fix deleting all rows * Fix grid size styling issue * Fix TS error * Hydration errors * studio org pages fixes * fix more studio links * audit logs fixes * dropdown icon styling fixes * fix some images in www * upgrade to next 14 * try new sentry wrapper for api * see if this is even invoked * Revert "see if this is even invoked" This reverts commit 86c3973ffa7f8ef5e1eb6d95a5809156cebf217b. * Revert "try new sentry wrapper for api" This reverts commit f67623ebad0f241d7e9fe275d50f7707bf64c474. * Revert "upgrade to next 14" This reverts commit a24dd6131eaff475a90ef991c2f832a64e4aaa2b. * chore: allow node version 19/20 * Try to fix the LogTable so that it renders with the newer "react-data-grid" version. * Fix type errors in the log renderer code. * Fix the replication screen. * Add the CSS for the GraphiQL. * Fix SQL editor results rendering * Lint * Fix SQL editor results height issue * Fix auth RLS not invalidating RQ when toggling RLS * Fix database tables new/edit column regressed * Fix migrations page empty state if migrations schema not yet created * Fix API side panel docs temp remove postgrest text for column description PK and FK * Fix + improve timeout handling in SQL editor --------- Co-authored-by: Jonathan Summers-Muir <MildTomato@users.noreply.github.com> Co-authored-by: Joshen Lim <joshenlimek@gmail.com> Co-authored-by: Francesco Sansalvadore <f.sansalvadore@gmail.com> Co-authored-by: Terry Sutton <saltcod@gmail.com> Co-authored-by: Ivan Vasilov <vasilov.ivan@gmail.com> Co-authored-by: Kevin Grüneberg <k.grueneberg1994@gmail.com>
148 lines
4.9 KiB
TypeScript
148 lines
4.9 KiB
TypeScript
import Link from 'next/link'
|
|
import React, { useState } from 'react'
|
|
import { extensions } from 'shared-data'
|
|
import { GlassPanel, IconX, Input } from 'ui'
|
|
|
|
type Extension = {
|
|
name: string
|
|
comment: string
|
|
tags: string[]
|
|
link: string
|
|
}
|
|
|
|
type LinkTarget = React.ComponentProps<'a'>['target']
|
|
|
|
function getLinkTarget(link: string): LinkTarget {
|
|
// Link is relative, open in the same tab
|
|
if (link.startsWith('/')) {
|
|
return '_self'
|
|
}
|
|
// Link is external, open in a new tab
|
|
return '_blank'
|
|
}
|
|
|
|
function getUniqueTags(json: Extension[]): string[] {
|
|
const tags = []
|
|
for (const item of json) {
|
|
if (item.tags) {
|
|
tags.push(...item.tags)
|
|
}
|
|
}
|
|
return [...new Set(tags)]
|
|
}
|
|
|
|
export default function Extensions() {
|
|
const [searchTerm, setSearchTerm] = useState<string>('')
|
|
const [filters, setFilters] = useState<string[]>([])
|
|
|
|
const tags = getUniqueTags(extensions)
|
|
|
|
function handleChecked(tag: string) {
|
|
if (filters.includes(tag)) {
|
|
setFilters(filters.filter((x) => x !== tag))
|
|
} else {
|
|
setFilters([...filters, tag])
|
|
}
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<div className="mb-8 grid">
|
|
<label className="mb-2 text-xs text-scale-1100">Search extensions</label>
|
|
<Input
|
|
type="text"
|
|
placeholder="Extension name"
|
|
onChange={(e) => setSearchTerm(e.target.value)}
|
|
/>
|
|
</div>
|
|
<div className="lg:grid lg:grid-cols-12">
|
|
<div className="col-span-3 not-prose">
|
|
<div className="lg:sticky top-24">
|
|
<h3 className="text-sm text-scale-1100">Filter</h3>
|
|
<ul className="mt-3 flex flex-wrap lg:grid gap-2 grow">
|
|
{tags.sort().map((tag) => (
|
|
<li key={tag}>
|
|
<label
|
|
htmlFor={tag}
|
|
className={`text-sm text-scale-1000 py-0.5 px-2 capitalize inline-block rounded-lg hover:bg-slate-400 hover:border-slate-400 cursor-pointer border ${
|
|
filters.includes(tag) ? 'bg-slate-400 ' : ''
|
|
}`}
|
|
>
|
|
<span className="flex items-center gap-1">
|
|
<input
|
|
type="checkbox"
|
|
className="sr-only"
|
|
id={tag}
|
|
name={tag}
|
|
value={tag}
|
|
onChange={() => handleChecked(tag)}
|
|
checked={filters.includes(tag)}
|
|
/>
|
|
{tag}
|
|
<span>{filters.includes(tag) && <IconX size={12} />}</span>
|
|
</span>
|
|
</label>
|
|
</li>
|
|
))}
|
|
</ul>
|
|
<p className="mt-2">
|
|
<button
|
|
type="reset"
|
|
className="text-xs hover:underline"
|
|
onClick={() => setFilters([])}
|
|
>
|
|
Reset
|
|
</button>
|
|
</p>
|
|
</div>
|
|
</div>
|
|
|
|
<div className="col-span-9 mt-4 lg:mt-0">
|
|
<div className="grid gap-4">
|
|
{extensions
|
|
.filter((x) => x.name.indexOf(searchTerm) >= 0)
|
|
.filter((x) =>
|
|
filters.length === 0 ? x : x.tags.some((item) => filters.includes(item))
|
|
)
|
|
.map((extension) => (
|
|
<Link
|
|
href={extension.link}
|
|
target={getLinkTarget(extension.link)}
|
|
className="no-underline"
|
|
>
|
|
<GlassPanel title={extension.name} background={false} key={extension.name}>
|
|
<p className="mt-4">
|
|
{extension.comment.charAt(0).toUpperCase() + extension.comment.slice(1)}
|
|
</p>
|
|
</GlassPanel>
|
|
</Link>
|
|
// <div className="my-2 px-2 relative" key={extension.name}>
|
|
// <div className="border rounded-sm p-4">
|
|
// <h3 className="m-0">
|
|
// <code className="text-sm">{extension.name}</code>
|
|
// </h3>
|
|
// <p className=" mt-4">
|
|
// {extension.comment.charAt(0).toUpperCase() + extension.comment.slice(1)}
|
|
// </p>
|
|
// {extension.link && (
|
|
// <Link href={extension.link}>
|
|
// <a
|
|
// target="_blank"
|
|
// className="text-xs no-underline absolute top-2 right-4 bg-slate-200 hover:bg-slate-400 transition-colors p-2 rounded-md"
|
|
// >
|
|
// <span>
|
|
// <IconLink size={14} className="" />
|
|
// </span>
|
|
// </a>
|
|
// </Link>
|
|
// )}
|
|
// </div>
|
|
// </div>
|
|
))}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</>
|
|
)
|
|
}
|