Retool is a great option for developers to build frontends on top of their data much faster than they could with traditional code. Whether you have a Postgres database, MongoDB, Firebase, or another, Retool is a scalable platform for building secure solutions for your teams to interact with their data.
Retool is truly designed for developers who can leverage their JavaScript and SQL expertise to build out complex internal applications within a drag-and-drop interface. For many developers, Retool is one of the best platforms to build out their CRUD apps. Nevertheless, for those new to Retool, there are still some platform-specific nuances that builders must familiarize themselves with first.
If you are interested in using Retool and curious as to whether it would work for you, this tutorial will walk you through the basics of the platform and how you would use it to build a CRUD on top of your data.
For this step-by-step walkthrough, we’ll use Retool’s built-in SQL database for our data, but Retool natively integrates with almost all database options for simple connection and authentication.
Here’s a preview of what our final app will look like:
Set up your Retool database or connect to a resource
Visit Retool and sign up if you don’t already have an account. Registration is straightforward—you can use your email or log in with your Google account for quick access and get started on the Free plan.
After signing in, you’ll land on the homepage. Here you can access the application root folders, workflows, resources, environment settings and get started with building.
data:image/s3,"s3://crabby-images/330f6/330f684e5c7eb2d3dca23208067cd4c5230b8e00" alt="Retool homepage."
Let’s start with our data layer and head to resources. Here, you’ll have access to Retool Database and Storage which are built-in storage options available on all plans. You can also configure Retool Email, as well as AI and Vectors.
data:image/s3,"s3://crabby-images/752a7/752a7ba616aae0f761b486f187efe607b33c822d" alt="Retool Resources list, including Database, Storage, Email, AI and Vectors."
From this page, you can click on the Create new dropdown button and select Resource.
On the next page, you’ll see some of the most common resources Retool connects with. From here, it’s super simple to connect to integrated resources or configure a REST API for anything they don’t support.
If you’ll be starting with a Retool-hosted DB, click on Retool Database to proceed.
data:image/s3,"s3://crabby-images/a7318/a731809b3aaf98d7798570d2552f8338a851221e" alt="List of Retool resource types, with Retool Database selected."
Set the advanced options to ‘Read connection’ for now, then click Test Connection and Save Changes once everything is working.
data:image/s3,"s3://crabby-images/0bdca/0bdcaa96010954dbaa8f2629fe56047e6443a001" alt="Retool Database Advanced options."
Congratulations! You now should have successfully connected to the Retool Database.
Once in Retool's DB, you will typically find a variety of pre-generated tables with demo data. These tables are designed to help users quickly prototype and explore their applications and are a great place to get started with the platform.
data:image/s3,"s3://crabby-images/0a0af/0a0af5576c83fff7f480b4c356bf7e9c63fdde7a" alt="Example of Retool demo data."
If you want to use your own data, you can create a new table tailored to your needs. Click the + button option within the Retool Database interface. In the modal, you can choose to upload via CSV, generate a schema with AI or start from scratch.
- First, name the table. We called ours customer_details.
Define Fields (Columns):
Use the Fields section to add the columns needed for your table. Here is the schema we used:
- id (UUID - Primary Key): A unique identifier for each row, automatically set as the primary key.
- customer_name (Text): Stores customer names.
- customer_address (Text): Holds customer addresses.
- Date (Date): Records when the row is created; default value can be set to NOW().
- active_subscription/subscription_status (Boolean/text): Indicates if the customer has an active subscription.
- customer_email (Text): Stores customer email addresses.
data:image/s3,"s3://crabby-images/3b678/3b67838143bfcce0862de5fabdc1ebfa96d7850d" alt="Create Table options in Retool Database, highlighting the Fields section."
After defining all fields, click the Create button to finalize the table. You’re ready to go! From here you can choose to manually add a few lines of data or use the import CSV to populate it.
If you prefer to import our demo data csv instead, you can download it here.
Build the CRUD UI in Retool
Now let’s think about setting up our user interface.
Head back to the Retool homepage and click the blue Create button. From here, you could start with a blank web or mobile app, use a template, or import an app from JSON/ZIP. Select App to start creating your CRUD interface!
data:image/s3,"s3://crabby-images/bc2e5/bc2e57459f534010b12dff3f368eeb365332727e" alt="How to start creating an App from the Retool homepage."
When you open your Retool app you’ll land on the app canvas. Here, you can use the first tab to drag and drop components onto the canvas and configure them in the right-hand inspector. On the left you can also access the component tree, code, pages, release versions, and additional settings.
Now that we’ve got to know the Retool IDE a little better, let’s start setting up our app.
Read: Querying the Retool Database
To connect to Retool Database, open the code page in the left-hand panel (marked with a </> symbol) and click the + icon in the Code section to create a new query. For this example, we’ll add the query to the global scope, but it’s worth noting here that you can also add code to the page code to be used on a single-page basis.
Choose Resource Query as the query type and select Retool Database, or your own database.
data:image/s3,"s3://crabby-images/4f4e9/4f4e9f10b65bd60a7005f608921d4838ceb4fac7" alt="How to add a Resource query in Retool."
In the SQL editor, type the following command to retrieve all rows from the customer_details table:
SELECT * FROM customer_details;
data:image/s3,"s3://crabby-images/89364/89364c57e84c579ce0312466254d421b3d4703dd" alt="The Retool SQL editor and output panel."
This query fetches all columns and rows from the table. Click the Run button to execute the query and the results will be displayed in the output panel below.
Now, let’s display this data in a table. In the Retool editor, click the Add Component icon (marked with a +) in the left-hand toolbar. From the Components panel, locate the Table option under the "Commonly Used" section. Drag and drop the Table component to your editor.
data:image/s3,"s3://crabby-images/dbb6f/dbb6fa8362fb69afc9d4d6b7d86ef5a57d699954" alt="The Components panel within Retool, with the Table component chosen."
Most likely, the data source will default to your new query, but if not you can map the table to the results of your query by selecting getCustomerDetails as Data source from the dropdown.
data:image/s3,"s3://crabby-images/140d6/140d6df7b15f070e85e1b25fb0423cb488f98f8c" alt="Connecting the Table component to query results using the Data Source dropdown in Retool."
Here’s what your data will look like:
data:image/s3,"s3://crabby-images/a7129/a7129e8351b5f1be008d5945b7af88f11da71761" alt="Example data in a Retool table."
Let’s make our data as clear as possible…
Delete: Configuring your delete action
Head back to the Code tab and add a new query by clicking the </> in the left-hand menu. Click + to create a new query and this time name it (e.g., deleteCustomer).
Configure the Delete query:
- In the Table field, select the target table from the Retool Database (customer_details).
- In the Action Type field, choose Delete a record.
- Filter by: Select id (the primary key of the customer_details table).
- Operator: Set to = (equals).
- Value:{{ table1.selectedRow.data.id }} - this will insert the id of the selected row in the table
- Click the Save button.
data:image/s3,"s3://crabby-images/68d75/68d75e1029721ba0d2ae0c2b61be9dfc2c4a3188" alt="How to create a query to delete a record from a table in a Retool Database."
Add an Event Handler to the button to refresh the table data after a record is deleted:
- On success: Re-run the getCustomerDetails query to update the table view.
Now let’s modify this deleteCustomer query to show a confirmation message before every run. Click on the Advanced tab and tick the confirmation modal checkbox. Here you can configure a message to run.
data:image/s3,"s3://crabby-images/c3c1d/c3c1d4bb62f36982971653fcc676e2e40bd81e17" alt="How to show add a confirmation modal before running a query in Retool."
Note that you can use JavaScript escape hatches anywhere to configure more custom functionality. Here, for example, you could insert the customer name into the confirmation modal by using {{table1.selectedRow.customer_name}} to insert a dynamic value.
Now let’s set up a button to complete the delete action.
In the Table component, add a column with a Button. Name this button “delete”. You can leave the other fields blank and use a prefix icon ‘trashcan’ to keep the UI simple - don’t forget that users often respond best to familiar UIs, so take inspiration from the apps you use every day!
data:image/s3,"s3://crabby-images/f9c67/f9c67f57b06d69dc91838ab6e63d5d87d53afa34" alt="How to add a delete button within a table in Retool."
Next, in the column settings, add an Event handler to this button to trigger the deleteCustomer query.
data:image/s3,"s3://crabby-images/935a5/935a5199586774827b935b95f36c0977b8a00b96" alt="Using an event handler to connect a query with a button in Retool."
Now you’re all set to delete rows in your table. Make sure to test this step to ensure the table refreshes with the updated data.
Update: setting up an edit form
Now let’s set up the Update query and edit form for our records.
Head back to the query tab and add a new query. Click + to create a new query and name it clearly (e.g., ‘updateCustomers’).
Configure the edit query:
- In the Table field, select the target table from the Retool database (customer_details).
- In the Action Type field, choose Update an existing record.
- Filter by: Select id (the primary key of the customer_details table).
- Operator: Set to = (equals).
- Value:{{ table1.selectedRow.data.id }}
- Use the Object option and input {{ customerForm.data }} to edit existing customer details in your DB. You can also use the key-value toggle to enter your key-value pairs manually instead.
- Click the Save button.
Please note that your component names might differ to ours, so you should ensure that you are using the correct component names.
data:image/s3,"s3://crabby-images/1276b/1276b98cc977b0a571321daff3c8cab4102a08b3" alt="Creating a query with Retool which will update an existing record within a database."
Just like we did with the delete button, let’s head back to the table component and add a new column for the Edit button. For our button, we will keep it simple with a green pen icon.
data:image/s3,"s3://crabby-images/bf386/bf386307755cd107b8501cad9bba53eea5829228" alt="Adding an Edit button within a table row in Retool."
Now, let’s build our ‘update’ form. We’ll set this up in a modal so that the form pops out when we click the edit button. This ensures we’re less likely to make mistakes than when editing a table in-line.
From the left-hand component menu, add a Modal component to your App. Next, add a form into your Modal and click on “Generate from table” to auto-generate your form.
data:image/s3,"s3://crabby-images/5525f/5525f58e90c475e52000fc318e635f97e940defb" alt="A Retool Modal component containing a form."
Here, you can select the columns you want included in your form, the input type as well as the required status.
data:image/s3,"s3://crabby-images/be7ef/be7ef338fcad96756c3df11763078e4c957ab8e3" alt="The Form Generator options in Retool."
The form generator automatically maps the table fields to input components using the selectedRow attribute of the table component. That way, the fields will show the current values for the record by default.
data:image/s3,"s3://crabby-images/e30a1/e30a1f0c34a5e56f9e99473a44d01163f3eed3eb" alt="Using the selectedRow attribute of the table component in Retool to map table fields to input components in a form generator."
Now, let’s configure the submit button to trigger the updateCustomer query.
data:image/s3,"s3://crabby-images/4536f/4536ff0461c4e85006c57268c6fceb88fbfa4139" alt="How to run a query when the Submit button on a form is clicked using the submit handler."
To keep our UI nice and tidy, let’s also add an Event Handler to the cancel button to close the modal. You can also set the modal to close onSuccess of the update query.
data:image/s3,"s3://crabby-images/2313f/2313f5ab32f885bf655a9aa56bba94a2e70c293f" alt="Using an event handler in Retool on a Cancel button to close a Modal."
For our final step, let’s connect the modal to the Edit button in our table with another click handler. This would trigger the modal to open when clicked.
data:image/s3,"s3://crabby-images/aad39/aad39bd1fe73d801bcddfc8acd7acbf3bb52580a" alt="Using a click handler in Retool to trigger a Modal to open when a button is clicked."
Now you can easily update the records in your table with a simple set of clicks.
Create: Add a new customer
For the final ‘Create’ step, we’re going to follow a lot of the same steps that we did for our Update action. Drag and drop another Modal Component onto your Retool canvas.
Next, drag a form component onto the modal frame. Click ‘generate from a table’ and set up your Create form just like your update one.
data:image/s3,"s3://crabby-images/07245/072451d8e61538b04f25c5f15b9b6c36193e570a" alt="Form Generator options in Retool."
Once again, for our form we had the following input types:
- Text Input:
- Customer Name: For customer_name.
- Email: For customer_email.
- Address: For customer_address.
- Date Picker:
- Date of Birth: For Date.
- Checkbox/Select:
Active Subscription: For subscription_status.
data:image/s3,"s3://crabby-images/f4bf0/f4bf04612ffa6d049da7828fd90613c371e59b40" alt="A Retool Modal containing a Form entitled New Customer."
Once again, open the code tab and create a new query. Configure the Create Action:
- In the Table field, select the target table (customer_details).
- In the Action Type field, choose Insert a record.
Use the Object option and input {{ customerForm.data }} to add a new customer to your DBClick the Save button.
data:image/s3,"s3://crabby-images/77906/77906d75525c937862a808017851dbd2bc83003f" alt="A Retool query, entitled createCustomer, which will add a new record to a database."
Finally, configure the submit button to trigger the createCustomer query to add a new record.
data:image/s3,"s3://crabby-images/c4451/c445111a5bde0e240ee6bb5a49fb6c8b18b498ea" alt="An event handler being used to add a new record to a database upon clicking the Submit button of a form."
Optionally, add the cancel button to close the Modal.
data:image/s3,"s3://crabby-images/274e9/274e94d9f5015b2fa103b205072651f72f3d248a" alt="Adding a click handler to a Cancel button on a form to close the containing Modal."
Finally, we need a button to open our Create form. Drag and drop a Button Component onto your Retool app interface and set the label to Add Customer.
Under ‘Event handlers’, add a click handler that triggers the modal to open.
data:image/s3,"s3://crabby-images/80397/803970b76a83a61bf42d309bb894cc1e57c1e081" alt="Adding a click handler to a button entitled Add Customer, which will open a Modal."
Extra UI tips
Sidebar for navigation
Now, before we move on, let’s spend a quick moment making this app look and feel like a polished product by setting up some navigation in a side panel.
Click the + icon in the left-hand toolbar to open the Components panel. In the global section of your app, click the + icon and select Sidebar.
data:image/s3,"s3://crabby-images/64aa6/64aa646db9aad7150a500746e03539675545fc2b" alt="Adding a Sidebar to an app in Retool."
The sidebar will default to have a title and a navigation component. You can change the appearance of the sidebar and configure the navigation bar in the inspector panel.
data:image/s3,"s3://crabby-images/55e74/55e74cded450c3479c4a05816dd8a497352d98ae" alt="The inspector panel for a sidebar in Retool."
data:image/s3,"s3://crabby-images/8ff16/8ff1671835d8329aa7beeb78c0ff17b4eafd85d2" alt="The inspector panel for the navigation bar in a sidebar in Retool."
data:image/s3,"s3://crabby-images/54ef6/54ef6d4e47a4eefb267a655b90f29287e54ce18b" alt="A Retool app entitled CRUD APP with a sidebar."
Set up a table search
Finally, let’s make sure our users can search our table. With just a few clicks we can add a Search Customer feature, allowing users to dynamically filter table data using a text input. Luckily, you can easily do this from the table component settings.
Drag and drop a Text Input component:
- Place a Text Input component above the table in your Retool app.
- In the Search section of the table settings:
- Set the Search mode to Fuzzy Match or Exact Match, depending on your requirements.
- Connect the Search Term: In the Search Term field, connect it to the Text Input’s value:{{ textInput5.value }}
- Test the Search is working correctly
data:image/s3,"s3://crabby-images/4adec/4adec5fd78d30b8db80f1df0f8fd3671625fa611" alt="Adding a search bar to a table in Retool."
In less than an hour, you now have your first simple app built on Retool. Stay tuned for more tutorials to learn more advanced functionality in Retool!