Projects
Services
Migration
Blog
Alt textAlt text

How we use .mdc rules

Reading Time

3 min read

Published on

April 1, 2025

Tags

Let's give you a quick guide on how and why we use Cursor .mdc rules. If you don't know what .mdc stands for, well...neither do we. But if I had to hazard a guess, I'd say it's probably Misguided Decisions & Chaos.

Moving quickly on from the Dad joke, I'm going to talk about why the rules are important, the specific things that this rule does, and why we would advise heavily tweaking it to your own preferences.

Schema layout

We have a very rigid way of building schema because, in most situations, you don't want to be debugging a large website project with one giant index file with 300+ schema types imported. That's why we have pre-determined patterns for adding indexes and spread operators on a folder-by-folder level.

This way, it becomes far easier to debug at scale and quickly include new schemas from within the same folder. The only trip hazard with this is that if you do not update the import in the index, you will not see your page builder block, so please be aware of this.

Typesafety

You'll notice a recurring theme in the way we build our Sanity websites... We hate having to manually type interfaces. We want to make as much typescript automated as physically possible, and for that reason, you'll notice it's heavily embedded within our .mdc file as well. Ultimately, you're making it easier for yourself in the long run, and the more you can automate, the easier you can scale.

Predefined schema

You may also notice there are a couple of predefined schema titles, names and types. There's a good reason for this. Usually, we're not the biggest fan of the DRY (Don't Repeat Yourself) way of coding. Purely because we find it's very easy to create spaghetti code when one requirement's goalposts inevitably shift and mutate content.

However, you will notice we regularly recycle the same naming conventions on different blocks. This helps to reuse Tailwindcss and ultimately increases your development momentum.

Image recognition

Finally, there's a real advantage to screenshot wireframes within Figma and get a rough draft for your schema. That's why we included some standardisation that correlates the point above of matching certain UI layouts with certain schema types to ensure you create consistent, user-friendly fields with descriptions and icons.

The rules themselves

markdown
1# Sanity Development Guidelines
2
3## Sanity Schema Rules
4
5When creating sanity schema make sure to include an appropriate icon for the schema using lucide-react or sanity icons as a fallback. Make sure it's always a named export, make sure you're always using the Sanity typescript definitions if it's a ts file.
6
7### Basic Schema Structure
8
9For TypeScript files, always import the necessary Sanity types:
10
11```typescript
12import {defineField, defineType, defineArrayMember} from 'sanity'
13```
14
15Always use `defineField` on every field and `defineType` throughout the whole type. Only import `defineArrayMember` if needed:
16
17```typescript
18defineType({
19 type: 'object',
20 name: 'custom-object',
21 fields: [
22 defineField({
23 type: 'array',
24 name: 'arrayField',
25 title: 'Things',
26 of: [
27 defineArrayMember({
28 type: 'object',
29 name: 'type-name-in-array',
30 fields: [defineField({type: 'string', name: 'title', title: 'Title'})],
31 }),
32 ],
33 }),
34 ],
35})
36```
37
38### Adding icons
39
40When adding icons to a schema, make sure you use the default sanity/icons first, and then if no icon is relevant, refer to any other iconset the user has installed - e.g lucide-react.
41
42### Structuring files and folders
43
44This is a rough idea of how to structure folders and files, ensuring you always have an index within the folder to create an array of documents/blocks. Do not use these as exact names, it's used purely for layout purposes.
45
46 │ ├── studio/
47 │ │ ├── README.md
48 │ │ ├── eslint.config.mjs
49 │ │ ├── location.ts
50 │ │ ├── package.json
51 │ │ ├── prettier.config.mjs
52 │ │ ├── sanity-typegen.json
53 │ │ ├── sanity.cli.ts
54 │ │ ├── sanity.config.ts
55 │ │ ├── schema.json
56 │ │ ├── structure.ts
57 │ │ ├── tsconfig.json
58 │ │ ├── .env.example
59 │ │ ├── .gitignore
60 │ │ ├── components/
61 │ │ │ ├── logo.tsx
62 │ │ │ └── slug-field-component.tsx
63 │ │ ├── plugins/
64 │ │ │ └── presentation-url.ts
65 │ │ ├── schemaTypes/
66 │ │ │ ├── common.ts
67 │ │ │ ├── index.ts
68 │ │ │ ├── blocks/
69 │ │ │ │ ├── cta.ts
70 │ │ │ │ ├── faq-accordion.ts
71 │ │ │ │ ├── feature-cards-icon.ts
72 │ │ │ │ ├── hero.ts
73 │ │ │ │ ├── image-link-cards.ts
74 │ │ │ │ ├── index.ts
75 │ │ │ │ └── subscribe-newsletter.ts
76 │ │ │ ├── definitions/
77 │ │ │ │ ├── button.ts
78 │ │ │ │ ├── custom-url.ts
79 │ │ │ │ ├── index.ts
80 │ │ │ │ ├── pagebuilder.ts
81 │ │ │ │ └── rich-text.ts
82 │ │ │ └── documents/
83 │ │ │ ├── author.ts
84 │ │ │ ├── blog.ts
85 │ │ │ ├── faq.ts
86 │ │ │ └── page.ts
87 │ │ └── utils/
88 │ │ ├── const-mock-data.ts
89 │ │ ├── constant.ts
90 │ │ ├── helper.ts
91 │ │ ├── mock-data.ts
92 │ │ ├── og-fields.ts
93 │ │ ├── parse-body.ts
94 │ │ ├── seo-fields.ts
95 │ │ ├── slug.ts
96 │ │ └── types.ts
97
98### Layout of page builder index example
99
100This is an example of how the blocks index file would be structured, you would create multiple of these on multiple nested routes to make it easier to create an array of files at each level, rather than bundling a large number of imports in a singular index.ts on the root
101
102```typescript
103import { callToAction } from './call-to-action';
104import { exploreHero } from './explore-hero';
105import { faqList } from './faq-list';
106import { htmlEmbed } from './html-embed';
107import { iconGrid } from './icon-grid';
108import { latestDocs } from './latest-docs';
109import { calculator } from './calculator';
110import { navigationCards } from './navigation-cards';
111import { quinstreetEmbed } from './quinstreet-embed';
112import { quote } from './quote';
113import { richTextBlock } from './rich-text-block';
114import { socialProof } from './social-proof';
115import { splitForm } from './split-form';
116import { statsCard } from './stats-card';
117import { trustCard } from './trust-card';
118import { rvEmbed } from './rv-embed';
119
120export const pagebuilderBlocks = [
121 navigationCards,
122 socialProof,
123 quote,
124 latestDocs,
125 faqList,
126 callToAction,
127 trustCard,
128 quinstreetEmbed,
129 statsCard,
130 iconGrid,
131 exploreHero,
132 splitForm,
133 richTextBlock,
134 calculator,
135 htmlEmbed,
136 rvEmbed,
137];
138
139export const blocks = [...pagebuilderBlocks];
140```
141
142### Common Field Templates
143
144When writing any Sanity schema, always include a description, name, title, and type. The description should explain functionality in simple terms for non-technical users. Place description above type.
145
146Use these templates when implementing common fields:
147
148#### Eyebrow
149```typescript
150defineField({
151 name: 'eyebrow',
152 title: 'Eyebrow',
153 description: 'The smaller text that sits above the title to provide context',
154 type: 'string',
155})
156```
157
158#### Title
159```typescript
160defineField({
161 name: 'title',
162 title: 'Title',
163 description: 'The large text that is the primary focus of the block',
164 type: 'string',
165})
166```
167
168#### Heading Level Toggle
169```typescript
170defineField({
171 name: 'isHeadingOne',
172 title: 'Is it a <h1>?',
173 type: 'boolean',
174 description:
175 'By default the title is a <h2> tag. If you use this as the top block on the page, you can toggle this on to make it a <h1> instead',
176 initialValue: false,
177})
178```
179
180#### Rich Text
181```typescript
182defineField({
183 name: 'richText',
184 title: 'Rich Text',
185 description: 'Large body of text that has links, ordered/unordered lists and headings.',
186 type: 'richText',
187})
188```
189
190#### Buttons
191```typescript
192defineField({
193 name: 'buttons',
194 title: 'Buttons',
195 description: 'Add buttons here, the website will handle the styling',
196 type: 'array',
197 of: [{type: 'button'}],
198})
199```
200
201#### Image
202```typescript
203defineField({
204 name: 'image',
205 title: 'Image',
206 type: 'image',
207 fields: [
208 defineField({
209 name: 'alt',
210 type: 'string',
211 description:
212 "Remember to use alt text for people to be able to read what is happening in the image if they are using a screen reader, it's also important for SEO",
213 title: 'Alt Text',
214 }),
215 ],
216})
217```
218
219### Type Generation
220
221After adding new Sanity schema, run the type command to generate TypeScript definitions:
222
223```bash
224sanity schema extract && sanity typegen generate --enforce-required-fields
225```
226
227## GROQ Rules
228
229Whenever there is an image within a GROQ query, do not expand it unless explicitly instructed to do so.
230
231
232## GROQ Query Structure and Organization
233
234- Import `defineQuery` and `groq` from `next-sanity` at the top of query files
235- Export queries as constants using the `defineQuery` function
236- Organize queries by content type (blogs, pages, products, etc.)
237- Group related queries together
238
239### Naming Conventions
240
241- Use camelCase for all query names
242- Prefix query names with action verb (get, getAll, etc.) followed by content type
243- Suffix all queries with "Query" (e.g., `getAllBlogIndexTranslationsQuery`)
244- Prefix reusable fragments with underscore (e.g., `_richText`, `_buttons`)
245
246### Fragment Reuse
247
248- Define common projection fragments at the top of the file
249- Create reusable fragments for repeated patterns (e.g., `_richText`, `_buttons`, `_icon`)
250- Use string interpolation to include fragments in queries
251- Ensure fragments are composable and focused on specific use cases
252
253### Query Parameters
254
255- Use `$` for parameters (e.g., `$slug`, `$locale`, `$id`)
256- Handle localization with consistent patterns (e.g., `${localeMatch}`)
257- Use `select()` for conditional logic within queries
258- Define default parameters using `coalesce()`
259
260### Response Types
261
262- Export TypeScript interfaces for query responses when needed
263- Use descriptive types that match the query structure
264- Follow the pattern: `export type GetAllMainPageTranslationsQueryResponse = string[];`
265
266### Best Practices
267
268- Use explicit filtering (`_type == "x"`) rather than implicit type checking
269- Prefer projection over returning entire documents
270- Use `order()` for explicit sorting rather than relying on document order
271- Check for defined fields (`defined(field)`) before accessing them
272- Use conditional projections for optional fields
273- Add pagination parameters (`[$start...$end]`) for list queries
274
275### Code Style
276
277- Use template literals for query strings
278- Indent nested query structures for readability
279- Keep related query parts together
280- Maintain consistent whitespace and indentation
281- Use comments to explain complex query logic
282
283
284## File Naming Conventions
285
286- Use kebab-case for ALL file names
287 - ✅ CORRECT: `user-profile.tsx`, `auth-layout.tsx`, `api-utils.ts`
288 - ❌ INCORRECT: `userProfile.tsx`, `AuthLayout.tsx`, `apiUtils.ts`
289- MUST use `.tsx` extension for React components
290- MUST use `.ts` extension for utility files
291- MUST use lowercase for all file names
292- MUST separate words with hyphens
293- MUST NOT use spaces or underscores
294
295## Screenshot Rules
296
297When asked to produce schema from screenshots, follow these guidelines:
298
299- Help describe types and interfaces using the provided image
300- Use the Sanity schema format shown above
301- Always include descriptions based on the visual elements in the image
302
303### Visual Cues
304
305- Tiny text above a title is likely an **eyebrow**
306- Large text without formatting that looks like a header should be a **title** or **subtitle**
307- Text with formatting (bold, italic, lists) likely needs **richText**
308- Images should include **alt text** fields
309- Background images should be handled appropriately
310- Use reusable button arrays for button patterns
311- If `richTextField` or `buttonsField` exists in the project, use them
312

Do your homework

Naturally, this .mdc file is very specific to our way of building Next.js and Sanity websites. We want you to take this ruleset and chop and change it to exactly the way you develop websites.

I would highly recommend seeing how we do it with our own open-source Turbo Start Sanity. The crux of our opinionation happens in the Sanity queries file. It's primarily around reusability in queries, especially when it comes to page builders. Inside of these, we want to nest things like rich text but don't necessarily want to figure out how to resolve internal links three blocks deep. That's where our query structure comes into play.

If you're new to Sanity and want something a little more chilled-out, we would highly recommend starting with Sanity Learn. When you're finished with that, we recommend reading Simeon's - Opinionated Guide to Sanity Studio.

Get in touch

Better than .mdc

Even better than .mdc files is having a skilled team to build everything for you. Forget the hard sell; we just want to talk Sanity and tech. Get in touch below, and we can help you scale your website and editorial experiences at a rapid pace.

Logo

Services

Legal

Like what you see?

Sign up for our newsletter to stay up to date with our latest projects and insights.

© 2025 Roboto Studio Ltd - 11126043

Roboto Studio Ltd,

71-75 Shelton Street,

Covent Garden,

London, WC2H 9JQ

Registered in England and Wales | VAT Number 426637679