Files
supabase/apps/studio/state/table-editor.tsx
Ali Waseem c39bb96d74 feat: Context view actions for views and material views (#46383)
## I have read the
[CONTRIBUTING.md](https://github.com/supabase/supabase/blob/master/CONTRIBUTING.md)
file.

YES

## What kind of change does this PR introduce?

Right now material views and views don't have any options on the context
menu, they only have a copy name. This adds copy schema, export CSV,
export SQL and delete table to that list

Added E2E tests to cover the use cases

<!-- This is an auto-generated comment: release notes by coderabbit.ai
-->
## Summary by CodeRabbit

* **New Features**
* Delete views and materialized views via confirmation dialogs with
optional cascade
  * Copy SQL definitions for views and materialized views
* Export views and materialized views as CSV and SQL from the entity
menu
* Confirmation modals now show dependency warnings and cascade toggle
consistently

* **Tests**
* End-to-end tests covering copy, export, and delete flows for views and
materialized views in the table editor

<!-- review_stack_entry_start -->

[![Review Change
Stack](https://storage.googleapis.com/coderabbit_public_assets/review-stack-in-coderabbit-ui.svg)](https://app.coderabbit.ai/change-stack/supabase/supabase/pull/46383?utm_source=github_walkthrough&utm_medium=github&utm_campaign=change_stack)

<!-- review_stack_entry_end -->
<!-- end of auto-generated comment: release notes by coderabbit.ai -->
2026-05-28 16:19:27 +00:00

398 lines
12 KiB
TypeScript

import * as Sentry from '@sentry/nextjs'
import type { PGColumn } from '@supabase/pg-meta'
import { useConstant } from 'common'
import { createContext, PropsWithChildren, useContext } from 'react'
import { proxy, useSnapshot } from 'valtio'
import {
NewQueuedOperation,
QueuedOperationType,
type OperationQueueState,
type QueueStatus,
} from './table-editor-operation-queue.types'
import type { SupaRow } from '@/components/grid/types'
import {
resolveDeleteRowConflicts,
resolveEditCellConflicts,
upsertOperation,
} from '@/components/grid/utils/queueConflictResolution'
import { generateTableChangeKey } from '@/components/grid/utils/queueOperationUtils'
import { ForeignKey } from '@/components/interfaces/TableGridEditor/SidePanelEditor/ForeignKeySelector/ForeignKeySelector.types'
import type { EditValue } from '@/components/interfaces/TableGridEditor/SidePanelEditor/RowEditor/RowEditor.types'
import type { TableField } from '@/components/interfaces/TableGridEditor/SidePanelEditor/TableEditor/TableEditor.types'
import type { SafePostgresColumn } from '@/lib/postgres-types'
import type { Dictionary } from '@/types'
export const TABLE_EDITOR_DEFAULT_ROWS_PER_PAGE = 100
type ForeignKeyState = {
foreignKey: ForeignKey
row: Dictionary<any>
column: PGColumn
}
export type SidePanel =
| { type: 'cell'; value?: { column: string; row: Dictionary<any> } }
| { type: 'row'; row?: Dictionary<any> }
| { type: 'column'; column?: SafePostgresColumn }
| { type: 'table'; mode: 'new' | 'edit' | 'duplicate'; templateData?: Partial<TableField> }
| { type: 'schema'; mode: 'new' | 'edit' }
| { type: 'json'; jsonValue: EditValue }
| {
type: 'foreign-row-selector'
foreignKey: ForeignKeyState
}
| { type: 'csv-import'; file?: File }
| { type: 'operation-queue' }
export type ConfirmationDialog =
| { type: 'table'; isDeleteWithCascade: boolean }
| { type: 'view'; isDeleteWithCascade: boolean }
| { type: 'materialized-view'; isDeleteWithCascade: boolean }
| { type: 'column'; column: SafePostgresColumn; isDeleteWithCascade: boolean }
// [Joshen] Just FYI callback, numRows, allRowsSelected is a temp workaround so that
// DeleteConfirmationDialog can trigger dispatch methods after the successful deletion of rows.
// Once we deprecate react tracked and move things to valtio, we can remove this.
| {
type: 'row'
rows: SupaRow[]
numRows?: number
allRowsSelected?: boolean
callback?: () => void
}
export type UIState =
| {
open: 'none'
}
| {
open: 'side-panel'
sidePanel: SidePanel
}
| {
open: 'confirmation-dialog'
confirmationDialog: ConfirmationDialog
}
/**
* Global table editor state for the table editor across multiple tables.
* See ./table-editor-table.tsx for table specific state.
*/
export const createTableEditorState = () => {
const state = proxy({
rowsPerPage: TABLE_EDITOR_DEFAULT_ROWS_PER_PAGE,
setRowsPerPage: (rowsPerPage: number) => {
state.rowsPerPage = rowsPerPage
},
ui: { open: 'none' } as UIState,
get sidePanel() {
return state.ui.open === 'side-panel' ? state.ui.sidePanel : undefined
},
get confirmationDialog() {
return state.ui.open === 'confirmation-dialog' ? state.ui.confirmationDialog : undefined
},
closeSidePanel: () => {
state.ui = { open: 'none' }
},
closeConfirmationDialog: () => {
state.ui = { open: 'none' }
},
onAddSchema: () => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'schema', mode: 'new' },
}
},
/* Tables */
onAddTable: (templateData?: Partial<TableField>) => {
// Record that the table creator was opened
Sentry.startSpan({ name: 'table_creator.opened', op: 'ui.action' }, (span) => {
span.setAttribute('table_creator.opened', 1)
})
state.ui = {
open: 'side-panel',
sidePanel: { type: 'table', mode: 'new', templateData },
}
},
onEditTable: () => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'table', mode: 'edit' },
}
},
onDuplicateTable: () => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'table', mode: 'duplicate' },
}
},
onDeleteTable: () => {
state.ui = {
open: 'confirmation-dialog',
confirmationDialog: { type: 'table', isDeleteWithCascade: false },
}
},
onDeleteView: () => {
state.ui = {
open: 'confirmation-dialog',
confirmationDialog: { type: 'view', isDeleteWithCascade: false },
}
},
onDeleteMaterializedView: () => {
state.ui = {
open: 'confirmation-dialog',
confirmationDialog: { type: 'materialized-view', isDeleteWithCascade: false },
}
},
/* Columns */
onAddColumn: () => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'column' },
}
},
onEditColumn: (column: SafePostgresColumn) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'column', column },
}
},
onDeleteColumn: (column: SafePostgresColumn) => {
state.ui = {
open: 'confirmation-dialog',
confirmationDialog: { type: 'column', column, isDeleteWithCascade: false },
}
},
/* Rows */
onAddRow: () => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'row' },
}
},
onEditRow: (row: Dictionary<any>) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'row', row },
}
},
onDeleteRows: (
rows: SupaRow[],
meta: { numRows?: number; allRowsSelected: boolean; callback?: () => void } = {
numRows: 0,
allRowsSelected: false,
callback: () => {},
}
) => {
const { numRows, allRowsSelected, callback } = meta
state.ui = {
open: 'confirmation-dialog',
confirmationDialog: { type: 'row', rows, numRows, allRowsSelected, callback },
}
},
/* Misc */
onExpandJSONEditor: (jsonValue: EditValue) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'json', jsonValue },
}
},
onExpandTextEditor: (column: string, row: Dictionary<any>) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'cell', value: { column, row } },
}
},
onEditForeignKeyColumnValue: (foreignKey: ForeignKeyState) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'foreign-row-selector', foreignKey },
}
},
onImportData: (file?: File) => {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'csv-import', file },
}
},
toggleViewOperationQueue: () => {
if (state.ui.open === 'side-panel' && state.ui.sidePanel.type === 'operation-queue') {
state.closeSidePanel()
} else {
state.ui = {
open: 'side-panel',
sidePanel: { type: 'operation-queue' },
}
}
},
/* Utils */
toggleConfirmationIsWithCascade: (overrideIsDeleteWithCascade?: boolean) => {
if (
state.ui.open === 'confirmation-dialog' &&
(state.ui.confirmationDialog.type === 'column' ||
state.ui.confirmationDialog.type === 'table' ||
state.ui.confirmationDialog.type === 'view' ||
state.ui.confirmationDialog.type === 'materialized-view')
) {
state.ui.confirmationDialog.isDeleteWithCascade =
overrideIsDeleteWithCascade ?? !state.ui.confirmationDialog.isDeleteWithCascade
}
},
// ========================================================================
// Operation Queue
// ========================================================================
operationQueue: {
operations: [],
status: 'idle',
} as OperationQueueState,
/**
* Queue a new operation for later processing.
* If an operation with the same key already exists, it will be overwritten.
* Handles conflict resolution:
* - DELETE_ROW on a row: remove any pending EDIT_CELL ops for that row
* - EDIT_CELL on a row pending deletion: reject (console.warn)
* - EDIT_CELL on a newly added row: merge edit into ADD_ROW's rowData
* - DELETE_ROW on a newly added row: cancel both operations
*/
queueOperation: (operation: NewQueuedOperation) => {
const updateQueueStatus = () => {
if (state.operationQueue.operations.length === 0) {
state.operationQueue.status = 'idle'
} else if (state.operationQueue.status === 'idle') {
state.operationQueue.status = 'pending'
}
}
// Handle DELETE_ROW conflicts
if (operation.type === QueuedOperationType.DELETE_ROW) {
const result = resolveDeleteRowConflicts(state.operationQueue.operations, operation)
state.operationQueue.operations = result.filteredOperations
if (result.action === 'skip') {
updateQueueStatus()
return
}
}
// Handle EDIT_CELL_CONTENT conflicts
if (operation.type === QueuedOperationType.EDIT_CELL_CONTENT) {
const result = resolveEditCellConflicts(state.operationQueue.operations, operation)
if (result.action === 'reject') {
console.warn(result.reason)
return
}
if (result.action === 'merge') {
state.operationQueue.operations = result.updatedOperations
updateQueueStatus()
return
}
}
// Normal upsert
const { operations } = upsertOperation(state.operationQueue.operations, operation)
state.operationQueue.operations = operations
updateQueueStatus()
},
/**
* Clear all operations from the queue
*/
clearQueue: () => {
state.operationQueue.operations = []
state.operationQueue.status = 'idle'
},
/**
* Remove a specific operation from the queue
*/
removeOperation: (operationId: string) => {
state.operationQueue.operations = state.operationQueue.operations.filter(
(op) => op.id !== operationId
)
if (state.operationQueue.operations.length === 0) {
state.operationQueue.status = 'idle'
}
},
/**
* Undo the latest operation from the queue
*/
undoLatestOperation: () => {
state.operationQueue.operations = state.operationQueue.operations.slice(0, -1)
if (state.operationQueue.operations.length === 0) {
state.operationQueue.status = 'idle'
}
},
/**
* Update the queue status
*/
setQueueStatus: (status: QueueStatus) => {
state.operationQueue.status = status
},
/**
* Check if there are any pending operations in the queue
*/
get hasPendingOperations(): boolean {
return state.operationQueue.operations.length > 0
},
hasPendingCellChange: (
tableId: number,
rowIdentifiers: Dictionary<unknown>,
columnName: string
): boolean => {
const key = generateTableChangeKey({
type: QueuedOperationType.EDIT_CELL_CONTENT,
tableId,
payload: {
columnName,
rowIdentifiers,
},
})
return state.operationQueue.operations.some((op) => op.id === key)
},
/**
* Toggle the preflight check behaviour for each table
*/
tablesToIgnorePreflightCheck: [] as number[],
setTableToIgnorePreflightCheck: (id: number) => {
const set = new Set<number>(state.tablesToIgnorePreflightCheck)
set.add(id)
state.tablesToIgnorePreflightCheck = [...set]
},
})
return state
}
export type TableEditorState = ReturnType<typeof createTableEditorState>
export const TableEditorStateContext = createContext<TableEditorState>(createTableEditorState())
export const TableEditorStateContextProvider = ({ children }: PropsWithChildren<{}>) => {
const state = useConstant(createTableEditorState)
return (
<TableEditorStateContext.Provider value={state}>{children}</TableEditorStateContext.Provider>
)
}
export const useTableEditorStateSnapshot = (options?: Parameters<typeof useSnapshot>[1]) => {
const state = useContext(TableEditorStateContext)
return useSnapshot(state, options)
}