GraphiQLInterface() — supabase Function Reference
Architecture documentation for the GraphiQLInterface() function in GraphiQL.tsx from the supabase codebase.
Entity Profile
Relationship Graph
Source Code
apps/studio/components/interfaces/GraphQL/GraphiQL.tsx lines 69–486
const GraphiQLInterface = ({ theme }: GraphiQLInterfaceProps) => {
const editorContext = useEditorContext({ nonNull: true })
const executionContext = useExecutionContext({ nonNull: true })
const schemaContext = useSchemaContext({ nonNull: true })
const pluginContext = usePluginContext()
const copy = useCopyQuery()
const merge = useMergeQuery()
const prettify = usePrettifyEditors()
const { can: canReadJWTSecret } = useAsyncCheckPermissions(
PermissionAction.READ,
'field.jwt_secret'
)
const [rlsBypassedWarningDismissed, setRlsBypassedWarningDismissed] = useLocalStorage(
LOCAL_STORAGE_KEYS.GRAPHIQL_RLS_BYPASS_WARNING,
false
)
const { setTheme } = useTheme()
useEffect(() => {
setTheme(theme)
}, [theme])
const PluginContent = pluginContext?.visiblePlugin?.content
const pluginResize = useDragResize({
defaultSizeRelation: 1 / 3,
direction: 'horizontal',
initiallyHidden: pluginContext?.visiblePlugin ? undefined : 'second',
onHiddenElementChange: (resizableElement) => {
if (resizableElement === 'second') {
pluginContext?.setVisiblePlugin(null)
}
},
sizeThresholdSecond: 200,
storageKey: 'docExplorerFlex',
})
const editorResize = useDragResize({
direction: 'horizontal',
storageKey: 'editorFlex',
})
const editorToolsResize = useDragResize({
defaultSizeRelation: 3,
direction: 'vertical',
initiallyHidden: (() => {
return editorContext.initialVariables || editorContext.initialHeaders ? undefined : 'second'
})(),
sizeThresholdSecond: 60,
storageKey: 'secondaryEditorFlex',
})
const [activeSecondaryEditor, setActiveSecondaryEditor] = useState<
'variables' | 'headers' | 'role-impersonation'
>(() => {
return !editorContext.initialVariables && editorContext.initialHeaders ? 'headers' : 'variables'
})
const toolbar = (
<>
<ToolbarButton onClick={prettify} label="Prettify query (Shift-Ctrl-P)">
<PrettifyIcon className="graphiql-toolbar-icon" aria-hidden="true" />
</ToolbarButton>
<ToolbarButton onClick={merge} label="Merge fragments into query (Shift-Ctrl-M)">
<MergeIcon className="graphiql-toolbar-icon" aria-hidden="true" />
</ToolbarButton>
<ToolbarButton onClick={copy} label="Copy query (Shift-Ctrl-C)">
<CopyIcon className="graphiql-toolbar-icon" aria-hidden="true" />
</ToolbarButton>
</>
)
const onClickReference = useCallback(() => {
if (pluginResize.hiddenElement === 'second') {
pluginResize.setHiddenElement(null)
}
}, [pluginResize])
const handleAddTab = editorContext.addTab
const handleRefetchSchema = schemaContext.introspect
const handleReorder = editorContext.moveTab
const handlePluginClick: MouseEventHandler<HTMLButtonElement> = useCallback(
(e) => {
const context = pluginContext!
const pluginIndex = Number(e.currentTarget.dataset.index!)
const plugin = context.plugins.find((_, index) => pluginIndex === index)!
const isVisible = plugin === context.visiblePlugin
if (isVisible) {
context.setVisiblePlugin(null)
pluginResize.setHiddenElement('second')
} else {
context.setVisiblePlugin(plugin)
pluginResize.setHiddenElement(null)
}
},
[pluginContext, pluginResize]
)
const handleToolsTabClick: MouseEventHandler<HTMLButtonElement> = useCallback(
(event) => {
if (editorToolsResize.hiddenElement === 'second') {
editorToolsResize.setHiddenElement(null)
}
setActiveSecondaryEditor(
event.currentTarget.dataset.name as 'variables' | 'headers' | 'role-impersonation'
)
},
[editorToolsResize]
)
const toggleEditorTools: MouseEventHandler<HTMLButtonElement> = useCallback(() => {
editorToolsResize.setHiddenElement(
editorToolsResize.hiddenElement === 'second' ? null : 'second'
)
}, [editorToolsResize])
const addTab = (
<Tooltip label="Add tab">
<UnStyledButton
type="button"
className="graphiql-tab-add text-sm"
onClick={handleAddTab}
aria-label="Add tab"
>
<PlusIcon aria-hidden="true" />
</UnStyledButton>
</Tooltip>
)
const hasSingleTab = editorContext.tabs.length === 1
return (
<Tooltip.Provider>
<div className={cn('graphiql-container', styles.graphiqlContainer)}>
<div className="graphiql-main">
<div
ref={pluginResize.firstRef}
style={{ minWidth: '750px' }}
className={cn('graphiql-sessions', styles.graphiqlSessions)}
>
<div
className={cn(
'graphiql-session-header',
!hasSingleTab && styles.graphiqlSessionHeader
)}
>
<Tabs
values={editorContext.tabs}
onReorder={handleReorder}
aria-label="Select active operation"
>
{editorContext.tabs.length > 1 && (
<>
{editorContext.tabs.map((tab, index) => (
<Tab
key={tab.id}
value={tab}
isActive={index === editorContext.activeTabIndex}
>
<Tab.Button
aria-controls="graphiql-session"
id={`graphiql-session-tab-${index}`}
onClick={() => {
executionContext.stop()
editorContext.changeTab(index)
}}
>
{tab.title}
</Tab.Button>
<Tab.Close
onClick={() => {
if (editorContext.activeTabIndex === index) {
executionContext.stop()
}
editorContext.closeTab(index)
}}
/>
</Tab>
))}
{addTab}
</>
)}
</Tabs>
<div className="graphiql-session-header-right">
{hasSingleTab && <div className={styles.graphiqlAddTabWrapper}>{addTab}</div>}
</div>
</div>
<div
role="tabpanel"
id="graphiql-session"
className={cn('graphiql-session', styles.graphiqlSession)}
aria-labelledby={`graphiql-session-tab-${editorContext.activeTabIndex}`}
>
<div ref={editorResize.firstRef}>
<div
className={cn(
'graphiql-editors',
styles.graphiqlEditors,
hasSingleTab && 'full-height'
)}
>
<div ref={editorToolsResize.firstRef}>
<section
className={cn('graphiql-query-editor text-sm', styles.graphiqlQueryEditor)}
aria-label="Query Editor"
>
<QueryEditor onClickReference={onClickReference} />
<div className="graphiql-toolbar" role="toolbar" aria-label="Editor Commands">
<ExecuteButton />
{toolbar}
</div>
</section>
</div>
<div ref={editorToolsResize.dragBarRef}>
<div className="graphiql-editor-tools">
<UnStyledButton
type="button"
className={
activeSecondaryEditor === 'variables' &&
editorToolsResize.hiddenElement !== 'second'
? 'active text-sm'
: 'text-sm'
}
onClick={handleToolsTabClick}
data-name="variables"
>
Variables
</UnStyledButton>
<UnStyledButton
type="button"
className={
activeSecondaryEditor === 'headers' &&
editorToolsResize.hiddenElement !== 'second'
? 'active text-sm'
: 'text-sm'
}
onClick={handleToolsTabClick}
data-name="headers"
>
Headers
</UnStyledButton>
{canReadJWTSecret && (
<UnStyledButton
type="button"
className={
activeSecondaryEditor === 'role-impersonation' &&
editorToolsResize.hiddenElement !== 'second'
? 'active text-sm'
: 'text-sm'
}
onClick={handleToolsTabClick}
data-name="role-impersonation"
>
Role Impersonation
</UnStyledButton>
)}
<Tooltip
label={
editorToolsResize.hiddenElement === 'second'
? 'Show editor tools'
: 'Hide editor tools'
}
>
<UnStyledButton
type="button"
onClick={toggleEditorTools}
aria-label={
editorToolsResize.hiddenElement === 'second'
? 'Show editor tools'
: 'Hide editor tools'
}
className="graphiql-toggle-editor-tools text-sm"
>
{editorToolsResize.hiddenElement === 'second' ? (
<ChevronUpIcon className="graphiql-chevron-icon" aria-hidden="true" />
) : (
<ChevronDownIcon className="graphiql-chevron-icon" aria-hidden="true" />
)}
</UnStyledButton>
</Tooltip>
</div>
</div>
<div ref={editorToolsResize.secondRef}>
<section
className="graphiql-editor-tool text-sm"
aria-label={activeSecondaryEditor === 'variables' ? 'Variables' : 'Headers'}
>
<VariableEditor
isHidden={activeSecondaryEditor !== 'variables'}
onClickReference={onClickReference}
/>
<HeaderEditor isHidden={activeSecondaryEditor !== 'headers'} />
{canReadJWTSecret && (
<div
className={cn(
'graphiql-editor px-1',
activeSecondaryEditor !== 'role-impersonation' && 'hidden'
)}
>
<RoleImpersonationSelector padded={false} />
</div>
)}
</section>
</div>
</div>
</div>
<div
className={cn('graphiql-horizontal-drag-bar', styles.graphiqlHorizontalDragBar)}
ref={editorResize.dragBarRef}
/>
<div ref={editorResize.secondRef}>
<div
className={cn(
'graphiql-response text-sm relative',
hasSingleTab
? styles.graphiqlResponseSingleTab
: styles.graphiqlResponseMultiTab
)}
>
{executionContext.isFetching ? <Spinner /> : null}
<ResponseEditor />
{!rlsBypassedWarningDismissed && (
<Alert_Shadcn_ variant="warning" className="absolute bottom-[5px] right-[5px]">
<AlertTriangle strokeWidth={2} />
<AlertTitle_Shadcn_ className="leading-5 text-foreground">
Please note that queries and mutations run in GraphiQL now use the service
role key by default.
<br />
<span className="text-amber-900">RLS will be bypassed.</span>
</AlertTitle_Shadcn_>
<AlertDescription_Shadcn_>
You can send queries as a specific role/user by using the role impersonation
tab.
</AlertDescription_Shadcn_>
<Button
type="outline"
aria-label="Dismiss"
className="absolute top-2 right-2 p-1 !pl-1"
onClick={() => {
setRlsBypassedWarningDismissed(true)
}}
>
<XIcon width={14} height={14} />
</Button>
</Alert_Shadcn_>
)}
</div>
</div>
</div>
</div>
{pluginContext?.visiblePlugin && (
<div className="graphiql-horizontal-drag-bar" ref={pluginResize.dragBarRef} />
)}
<div
ref={pluginResize.secondRef}
style={{
// Make sure the container shrinks when containing long
// non-breaking texts
minWidth: '200px',
}}
>
<div className="graphiql-plugin">{PluginContent ? <PluginContent /> : null}</div>
</div>
</div>
<div className={cn('graphiql-sidebar', styles.graphiqlSidebar)}>
<div className="graphiql-sidebar-section">
{pluginContext?.plugins.map((plugin, index) => {
const isVisible = plugin === pluginContext.visiblePlugin
const label = `${isVisible ? 'Hide' : 'Show'} ${plugin.title}`
const Icon = plugin.icon
return (
<Tooltip key={plugin.title} label={label}>
<UnStyledButton
type="button"
className={isVisible ? 'active text-sm' : 'text-sm'}
onClick={handlePluginClick}
data-index={index}
aria-label={label}
>
<Icon aria-hidden="true" />
</UnStyledButton>
</Tooltip>
)
})}
</div>
<div className="graphiql-sidebar-section">
<Tooltip label="Re-fetch GraphQL schema">
<UnStyledButton
type="button"
className="text-sm"
disabled={schemaContext.isFetching}
onClick={handleRefetchSchema}
aria-label="Re-fetch GraphQL schema"
>
<ReloadIcon
className={schemaContext.isFetching ? 'graphiql-spin' : ''}
aria-hidden="true"
/>
</UnStyledButton>
</Tooltip>
</div>
</div>
</div>
</Tooltip.Provider>
)
}
Domain
Subdomains
Source
Analyze Your Own Codebase
Get architecture documentation, dependency graphs, and domain analysis for your codebase in minutes.
Try Supermodel Free