mirror of
https://github.com/supabase/supabase.git
synced 2026-06-08 10:33:55 +08:00
* Add a API endpoint for generating queries for the SQL editor. * Merge all multiline props. * Add a new component for diff actions. * Copy components for the AI panel. * Add useChat hook. * Add a feature preview for the this feature. The preview is dependent on the feature flag. * Reorder the nesting in the SQL editor to accomodate the AI assistant. * Try to fit both AI assistants in the SQL editor, available via a feature preview. * Refactor the SQL editor to make the diff work correctly in all cases. * Minor fixes for the old AI feature. * Fix the debug functionality to work with both assistants. * Fix some copy-paste leftovers. * Remove unneeded code. * Make the icons softer. * Fix the name of the panel component. * Fix console.logs. * Add overflow to the AI assistant. * surface opt in config in ai settings button * Skip diffing if editor is empty * Add selected state when selecting a message to insert/replace code * Add sample prompts * Add SQL ai dislaimer when replacing code * Light mode action bar nudges * Add text for the feature preview. * lang nudges * Hide the command suggestions for now. * Set the discussion url to undefined. * Don't add the disclaimer twice. --------- Co-authored-by: Joshen Lim <joshenlimek@gmail.com> Co-authored-by: Terry Sutton <saltcod@gmail.com>
177 lines
6.7 KiB
TypeScript
177 lines
6.7 KiB
TypeScript
import { useTelemetryProps } from 'common'
|
|
import { FlaskConical } from 'lucide-react'
|
|
import Link from 'next/link'
|
|
import { useRouter } from 'next/router'
|
|
import { useEffect, useState } from 'react'
|
|
import { Button, IconExternalLink, IconEye, IconEyeOff, Modal, ScrollArea, cn } from 'ui'
|
|
|
|
import { useFlag } from 'hooks'
|
|
import { LOCAL_STORAGE_KEYS } from 'lib/constants'
|
|
import Telemetry from 'lib/telemetry'
|
|
import { useAppStateSnapshot } from 'state/app-state'
|
|
import APISidePanelPreview from './APISidePanelPreview'
|
|
import CLSPreview from './CLSPreview'
|
|
import { useFeaturePreviewContext } from './FeaturePreviewContext'
|
|
import RLSAIAssistantPreview from './RLSAIAssistantPreview'
|
|
import { SQLEditorAIAssistantPreview } from './SQLEditorAIAssistantPreview'
|
|
|
|
const FeaturePreviewModal = () => {
|
|
const isAIConversational = useFlag('sqlEditorConversationalAi')
|
|
|
|
// [Ivan] We should probably move this to a separate file, together with LOCAL_STORAGE_KEYS. We should make adding new feature previews as simple as possible.
|
|
const FEATURE_PREVIEWS: { key: string; name: string; content: any; discussionsUrl?: string }[] = [
|
|
{
|
|
key: LOCAL_STORAGE_KEYS.UI_PREVIEW_API_SIDE_PANEL,
|
|
name: 'Project API documentation',
|
|
content: <APISidePanelPreview />,
|
|
discussionsUrl: 'https://github.com/orgs/supabase/discussions/18038',
|
|
},
|
|
{
|
|
key: LOCAL_STORAGE_KEYS.UI_PREVIEW_RLS_AI_ASSISTANT,
|
|
name: 'Supabase Assistant for RLS policies',
|
|
content: <RLSAIAssistantPreview />,
|
|
discussionsUrl: 'https://github.com/orgs/supabase/discussions/19594',
|
|
},
|
|
{
|
|
key: LOCAL_STORAGE_KEYS.UI_PREVIEW_CLS,
|
|
name: 'Column-level privileges',
|
|
content: <CLSPreview />,
|
|
discussionsUrl: 'https://github.com/orgs/supabase/discussions/20295',
|
|
},
|
|
// the user should only be able to see the panel for the AI assistant if the feature flag is true
|
|
...(isAIConversational
|
|
? [
|
|
{
|
|
key: LOCAL_STORAGE_KEYS.UI_PREVIEW_SQL_EDITOR_AI_ASSISTANT,
|
|
name: 'Supabase Assistant for SQL editor',
|
|
content: <SQLEditorAIAssistantPreview />,
|
|
discussionsUrl: undefined,
|
|
},
|
|
]
|
|
: []),
|
|
]
|
|
|
|
const router = useRouter()
|
|
const snap = useAppStateSnapshot()
|
|
const telemetryProps = useTelemetryProps()
|
|
const featurePreviewContext = useFeaturePreviewContext()
|
|
|
|
const selectedFeaturePreview =
|
|
snap.selectedFeaturePreview === '' ? FEATURE_PREVIEWS[0].key : snap.selectedFeaturePreview
|
|
|
|
const [selectedFeatureKey, setSelectedFeatureKey] = useState<string>(selectedFeaturePreview)
|
|
|
|
// this modal can be triggered on other pages
|
|
// Update local state when valtio state changes
|
|
useEffect(() => {
|
|
if (snap.selectedFeaturePreview !== '') {
|
|
setSelectedFeatureKey(snap.selectedFeaturePreview)
|
|
}
|
|
}, [snap.selectedFeaturePreview])
|
|
|
|
const { flags, onUpdateFlag } = featurePreviewContext
|
|
const selectedFeature = FEATURE_PREVIEWS.find((preview) => preview.key === selectedFeatureKey)
|
|
const isSelectedFeatureEnabled = flags[selectedFeatureKey]
|
|
|
|
const toggleFeature = () => {
|
|
onUpdateFlag(selectedFeatureKey, !isSelectedFeatureEnabled)
|
|
Telemetry.sendEvent(
|
|
{
|
|
category: 'ui_feature_previews',
|
|
action: isSelectedFeatureEnabled ? 'disabled' : 'enabled',
|
|
label: selectedFeatureKey,
|
|
},
|
|
telemetryProps,
|
|
router
|
|
)
|
|
}
|
|
|
|
function handleCloseFeaturePreviewModal() {
|
|
snap.setShowFeaturePreviewModal(false)
|
|
snap.setSelectedFeaturePreview(FEATURE_PREVIEWS[0].key)
|
|
}
|
|
|
|
return (
|
|
<Modal
|
|
hideFooter
|
|
showCloseButton
|
|
size="xlarge"
|
|
className="max-w-4xl"
|
|
header="Dashboard feature previews"
|
|
visible={snap.showFeaturePreviewModal}
|
|
onCancel={handleCloseFeaturePreviewModal}
|
|
>
|
|
{FEATURE_PREVIEWS.length > 0 ? (
|
|
<div className="flex border-t">
|
|
<div>
|
|
<ScrollArea className="h-[550px] w-[280px] border-r">
|
|
{FEATURE_PREVIEWS.map((feature) => {
|
|
const isEnabled = flags[feature.key] ?? false
|
|
|
|
return (
|
|
<div
|
|
key={feature.key}
|
|
onClick={() => setSelectedFeatureKey(feature.key)}
|
|
className={cn(
|
|
'flex items-center space-x-3 p-4 border-b cursor-pointer bg transition',
|
|
selectedFeatureKey === feature.key ? 'bg-surface-300' : 'bg-surface-100'
|
|
)}
|
|
>
|
|
{isEnabled ? (
|
|
<IconEye size={14} strokeWidth={2} className="text-brand" />
|
|
) : (
|
|
<IconEyeOff size={14} strokeWidth={1.5} className="text-foreground-light" />
|
|
)}
|
|
<p className="text-sm truncate" title={feature.name}>
|
|
{feature.name}
|
|
</p>
|
|
</div>
|
|
)
|
|
})}
|
|
</ScrollArea>
|
|
</div>
|
|
<div className="flex-grow max-h-[550px] p-4 space-y-3 overflow-y-auto">
|
|
<div className="flex items-center justify-between">
|
|
<p>{selectedFeature?.name}</p>
|
|
<div className="flex items-center space-x-2">
|
|
{selectedFeature?.discussionsUrl !== undefined && (
|
|
<Button asChild type="default" icon={<IconExternalLink strokeWidth={1.5} />}>
|
|
<Link href={selectedFeature.discussionsUrl} target="_blank" rel="noreferrer">
|
|
Give feedback
|
|
</Link>
|
|
</Button>
|
|
)}
|
|
<Button type="default" onClick={() => toggleFeature()}>
|
|
{isSelectedFeatureEnabled ? 'Disable' : 'Enable'} feature
|
|
</Button>
|
|
</div>
|
|
</div>
|
|
{selectedFeature?.content}
|
|
</div>
|
|
</div>
|
|
) : (
|
|
<div className="h-[550px] flex flex-col items-center justify-center">
|
|
<FlaskConical size={30} strokeWidth={1.5} className="text-foreground-light" />
|
|
<div className="mt-1 mb-3 flex flex-col items-center gap-y-0.5">
|
|
<p className="text-sm">No feature previews available</p>
|
|
<p className="text-sm text-foreground-light">
|
|
Have an idea for the dashboard? Let us know via Github Discussions!
|
|
</p>
|
|
</div>
|
|
<Button asChild type="default" icon={<IconExternalLink strokeWidth={1.5} />}>
|
|
<Link
|
|
href="https://github.com/orgs/supabase/discussions/categories/feature-requests"
|
|
target="_blank"
|
|
rel="noreferrer"
|
|
>
|
|
Github Discussions
|
|
</Link>
|
|
</Button>
|
|
</div>
|
|
)}
|
|
</Modal>
|
|
)
|
|
}
|
|
|
|
export default FeaturePreviewModal
|