@gallop.software/canon 2.0.2 → 2.2.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -8,6 +8,8 @@ declare const recommendedConfig: {
8
8
  'gallop/no-client-blocks': string;
9
9
  'gallop/no-container-in-section': string;
10
10
  'gallop/prefer-component-props': string;
11
+ 'gallop/prefer-layout-components': string;
12
+ 'gallop/background-image-rounded': string;
11
13
  };
12
14
  };
13
15
  export default recommendedConfig;
@@ -9,7 +9,9 @@ const recommendedConfig = {
9
9
  'gallop/no-client-blocks': 'warn',
10
10
  'gallop/no-container-in-section': 'warn',
11
11
  'gallop/prefer-component-props': 'warn',
12
+ 'gallop/prefer-layout-components': 'warn',
13
+ 'gallop/background-image-rounded': 'warn',
12
14
  },
13
15
  };
14
16
  export default recommendedConfig;
15
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb21tZW5kZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L2NvbmZpZ3MvcmVjb21tZW5kZWQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBQ0gsTUFBTSxpQkFBaUIsR0FBRztJQUN4QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMENBQTBDO1FBQzFDLHlCQUF5QixFQUFFLE1BQU07UUFDakMsZ0NBQWdDLEVBQUUsTUFBTTtRQUN4QywrQkFBK0IsRUFBRSxNQUFNO0tBQ3hDO0NBQ0YsQ0FBQTtBQUVELGVBQWUsaUJBQWlCLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFJlY29tbWVuZGVkIGNvbmZpZ3VyYXRpb25cbiAqIEEgc2Vuc2libGUgZGVmYXVsdCBmb3IgYW55IEdhbGxvcC1iYXNlZCB0ZW1wbGF0ZVxuICovXG5jb25zdCByZWNvbW1lbmRlZENvbmZpZyA9IHtcbiAgcGx1Z2luczogWydnYWxsb3AnXSxcbiAgcnVsZXM6IHtcbiAgICAvLyBDb3JlIHJ1bGVzIHRoYXQgYXBwbHkgdG8gbW9zdCB0ZW1wbGF0ZXNcbiAgICAnZ2FsbG9wL25vLWNsaWVudC1ibG9ja3MnOiAnd2FybicsXG4gICAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcbiAgICAnZ2FsbG9wL3ByZWZlci1jb21wb25lbnQtcHJvcHMnOiAnd2FybicsXG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHJlY29tbWVuZGVkQ29uZmlnXG4iXX0=
17
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb21tZW5kZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L2NvbmZpZ3MvcmVjb21tZW5kZWQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBQ0gsTUFBTSxpQkFBaUIsR0FBRztJQUN4QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMENBQTBDO1FBQzFDLHlCQUF5QixFQUFFLE1BQU07UUFDakMsZ0NBQWdDLEVBQUUsTUFBTTtRQUN4QywrQkFBK0IsRUFBRSxNQUFNO1FBQ3ZDLGlDQUFpQyxFQUFFLE1BQU07UUFDekMsaUNBQWlDLEVBQUUsTUFBTTtLQUMxQztDQUNGLENBQUE7QUFFRCxlQUFlLGlCQUFpQixDQUFBIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBSZWNvbW1lbmRlZCBjb25maWd1cmF0aW9uXG4gKiBBIHNlbnNpYmxlIGRlZmF1bHQgZm9yIGFueSBHYWxsb3AtYmFzZWQgdGVtcGxhdGVcbiAqL1xuY29uc3QgcmVjb21tZW5kZWRDb25maWcgPSB7XG4gIHBsdWdpbnM6IFsnZ2FsbG9wJ10sXG4gIHJ1bGVzOiB7XG4gICAgLy8gQ29yZSBydWxlcyB0aGF0IGFwcGx5IHRvIG1vc3QgdGVtcGxhdGVzXG4gICAgJ2dhbGxvcC9uby1jbGllbnQtYmxvY2tzJzogJ3dhcm4nLFxuICAgICdnYWxsb3Avbm8tY29udGFpbmVyLWluLXNlY3Rpb24nOiAnd2FybicsXG4gICAgJ2dhbGxvcC9wcmVmZXItY29tcG9uZW50LXByb3BzJzogJ3dhcm4nLFxuICAgICdnYWxsb3AvcHJlZmVyLWxheW91dC1jb21wb25lbnRzJzogJ3dhcm4nLFxuICAgICdnYWxsb3AvYmFja2dyb3VuZC1pbWFnZS1yb3VuZGVkJzogJ3dhcm4nLFxuICB9LFxufVxuXG5leHBvcnQgZGVmYXVsdCByZWNvbW1lbmRlZENvbmZpZ1xuIl19
@@ -8,6 +8,8 @@ declare const speedwellConfig: {
8
8
  'gallop/no-client-blocks': string;
9
9
  'gallop/no-container-in-section': string;
10
10
  'gallop/prefer-component-props': string;
11
+ 'gallop/prefer-layout-components': string;
12
+ 'gallop/background-image-rounded': string;
11
13
  };
12
14
  };
13
15
  export default speedwellConfig;
@@ -11,7 +11,11 @@ const speedwellConfig = {
11
11
  'gallop/no-container-in-section': 'warn',
12
12
  // Use component props instead of className for style values
13
13
  'gallop/prefer-component-props': 'warn',
14
+ // Use Grid/Columns instead of raw div with grid classes
15
+ 'gallop/prefer-layout-components': 'warn',
16
+ // Background images must have rounded="rounded-none"
17
+ 'gallop/background-image-rounded': 'warn',
14
18
  },
15
19
  };
16
20
  export default speedwellConfig;
17
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BlZWR3ZWxsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2VzbGludC9jb25maWdzL3NwZWVkd2VsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLGVBQWUsR0FBRztJQUN0QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMEVBQTBFO1FBQzFFLHlCQUF5QixFQUFFLE1BQU07UUFFakMsdUNBQXVDO1FBQ3ZDLGdDQUFnQyxFQUFFLE1BQU07UUFFeEMsNERBQTREO1FBQzVELCtCQUErQixFQUFFLE1BQU07S0FDeEM7Q0FDRixDQUFBO0FBRUQsZUFBZSxlQUFlLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNwZWVkd2VsbCB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uXG4gKiBFbmFibGVzIGFsbCBHYWxsb3AgcnVsZXMgcmVsZXZhbnQgdG8gdGhlIFNwZWVkd2VsbCBhcmNoaXRlY3R1cmVcbiAqL1xuY29uc3Qgc3BlZWR3ZWxsQ29uZmlnID0ge1xuICBwbHVnaW5zOiBbJ2dhbGxvcCddLFxuICBydWxlczoge1xuICAgIC8vIEJsb2NrcyBzaG91bGQgYmUgc2VydmVyIGNvbXBvbmVudHMgLSBleHRyYWN0IGNsaWVudCBsb2dpYyB0byBjb21wb25lbnRzXG4gICAgJ2dhbGxvcC9uby1jbGllbnQtYmxvY2tzJzogJ3dhcm4nLFxuXG4gICAgLy8gU2VjdGlvbiBhbHJlYWR5IHByb3ZpZGVzIGNvbnRhaW5tZW50XG4gICAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcblxuICAgIC8vIFVzZSBjb21wb25lbnQgcHJvcHMgaW5zdGVhZCBvZiBjbGFzc05hbWUgZm9yIHN0eWxlIHZhbHVlc1xuICAgICdnYWxsb3AvcHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6ICd3YXJuJyxcbiAgfSxcbn1cblxuZXhwb3J0IGRlZmF1bHQgc3BlZWR3ZWxsQ29uZmlnXG4iXX0=
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BlZWR3ZWxsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2VzbGludC9jb25maWdzL3NwZWVkd2VsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLGVBQWUsR0FBRztJQUN0QixPQUFPLEVBQUUsQ0FBQyxRQUFRLENBQUM7SUFDbkIsS0FBSyxFQUFFO1FBQ0wsMEVBQTBFO1FBQzFFLHlCQUF5QixFQUFFLE1BQU07UUFFakMsdUNBQXVDO1FBQ3ZDLGdDQUFnQyxFQUFFLE1BQU07UUFFeEMsNERBQTREO1FBQzVELCtCQUErQixFQUFFLE1BQU07UUFFdkMsd0RBQXdEO1FBQ3hELGlDQUFpQyxFQUFFLE1BQU07UUFFekMscURBQXFEO1FBQ3JELGlDQUFpQyxFQUFFLE1BQU07S0FDMUM7Q0FDRixDQUFBO0FBRUQsZUFBZSxlQUFlLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNwZWVkd2VsbCB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uXG4gKiBFbmFibGVzIGFsbCBHYWxsb3AgcnVsZXMgcmVsZXZhbnQgdG8gdGhlIFNwZWVkd2VsbCBhcmNoaXRlY3R1cmVcbiAqL1xuY29uc3Qgc3BlZWR3ZWxsQ29uZmlnID0ge1xuICBwbHVnaW5zOiBbJ2dhbGxvcCddLFxuICBydWxlczoge1xuICAgIC8vIEJsb2NrcyBzaG91bGQgYmUgc2VydmVyIGNvbXBvbmVudHMgLSBleHRyYWN0IGNsaWVudCBsb2dpYyB0byBjb21wb25lbnRzXG4gICAgJ2dhbGxvcC9uby1jbGllbnQtYmxvY2tzJzogJ3dhcm4nLFxuXG4gICAgLy8gU2VjdGlvbiBhbHJlYWR5IHByb3ZpZGVzIGNvbnRhaW5tZW50XG4gICAgJ2dhbGxvcC9uby1jb250YWluZXItaW4tc2VjdGlvbic6ICd3YXJuJyxcblxuICAgIC8vIFVzZSBjb21wb25lbnQgcHJvcHMgaW5zdGVhZCBvZiBjbGFzc05hbWUgZm9yIHN0eWxlIHZhbHVlc1xuICAgICdnYWxsb3AvcHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6ICd3YXJuJyxcblxuICAgIC8vIFVzZSBHcmlkL0NvbHVtbnMgaW5zdGVhZCBvZiByYXcgZGl2IHdpdGggZ3JpZCBjbGFzc2VzXG4gICAgJ2dhbGxvcC9wcmVmZXItbGF5b3V0LWNvbXBvbmVudHMnOiAnd2FybicsXG5cbiAgICAvLyBCYWNrZ3JvdW5kIGltYWdlcyBtdXN0IGhhdmUgcm91bmRlZD1cInJvdW5kZWQtbm9uZVwiXG4gICAgJ2dhbGxvcC9iYWNrZ3JvdW5kLWltYWdlLXJvdW5kZWQnOiAnd2FybicsXG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHNwZWVkd2VsbENvbmZpZ1xuIl19
@@ -14,6 +14,8 @@ declare const plugin: {
14
14
  name: string;
15
15
  };
16
16
  'prefer-typography-components': import("eslint").Rule.RuleModule;
17
+ 'prefer-layout-components': import("eslint").Rule.RuleModule;
18
+ 'background-image-rounded': import("eslint").Rule.RuleModule;
17
19
  };
18
20
  configs: {
19
21
  speedwell: {
@@ -22,6 +24,8 @@ declare const plugin: {
22
24
  'gallop/no-client-blocks': string;
23
25
  'gallop/no-container-in-section': string;
24
26
  'gallop/prefer-component-props': string;
27
+ 'gallop/prefer-layout-components': string;
28
+ 'gallop/background-image-rounded': string;
25
29
  };
26
30
  };
27
31
  recommended: {
@@ -30,6 +34,8 @@ declare const plugin: {
30
34
  'gallop/no-client-blocks': string;
31
35
  'gallop/no-container-in-section': string;
32
36
  'gallop/prefer-component-props': string;
37
+ 'gallop/prefer-layout-components': string;
38
+ 'gallop/background-image-rounded': string;
33
39
  };
34
40
  };
35
41
  };
@@ -2,6 +2,8 @@ import noClientBlocks from './rules/no-client-blocks.js';
2
2
  import noContainerInSection from './rules/no-container-in-section.js';
3
3
  import preferComponentProps from './rules/prefer-component-props.js';
4
4
  import preferTypographyComponents from './rules/prefer-typography-components.js';
5
+ import preferLayoutComponents from './rules/prefer-layout-components.js';
6
+ import backgroundImageRounded from './rules/background-image-rounded.js';
5
7
  import speedwellConfig from './configs/speedwell.js';
6
8
  import recommendedConfig from './configs/recommended.js';
7
9
  const plugin = {
@@ -14,6 +16,8 @@ const plugin = {
14
16
  'no-container-in-section': noContainerInSection,
15
17
  'prefer-component-props': preferComponentProps,
16
18
  'prefer-typography-components': preferTypographyComponents,
19
+ 'prefer-layout-components': preferLayoutComponents,
20
+ 'background-image-rounded': backgroundImageRounded,
17
21
  },
18
22
  configs: {
19
23
  speedwell: speedwellConfig,
@@ -21,4 +25,4 @@ const plugin = {
21
25
  },
22
26
  };
23
27
  export default plugin;
24
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXNsaW50L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sY0FBYyxNQUFNLDZCQUE2QixDQUFBO0FBQ3hELE9BQU8sb0JBQW9CLE1BQU0sb0NBQW9DLENBQUE7QUFDckUsT0FBTyxvQkFBb0IsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNwRSxPQUFPLDBCQUEwQixNQUFNLHlDQUF5QyxDQUFBO0FBQ2hGLE9BQU8sZUFBZSxNQUFNLHdCQUF3QixDQUFBO0FBQ3BELE9BQU8saUJBQWlCLE1BQU0sMEJBQTBCLENBQUE7QUFFeEQsTUFBTSxNQUFNLEdBQUc7SUFDYixJQUFJLEVBQUU7UUFDSixJQUFJLEVBQUUsc0JBQXNCO1FBQzVCLE9BQU8sRUFBRSxPQUFPO0tBQ2pCO0lBQ0QsS0FBSyxFQUFFO1FBQ0wsa0JBQWtCLEVBQUUsY0FBYztRQUNsQyx5QkFBeUIsRUFBRSxvQkFBb0I7UUFDL0Msd0JBQXdCLEVBQUUsb0JBQW9CO1FBQzlDLDhCQUE4QixFQUFFLDBCQUEwQjtLQUMzRDtJQUNELE9BQU8sRUFBRTtRQUNQLFNBQVMsRUFBRSxlQUFlO1FBQzFCLFdBQVcsRUFBRSxpQkFBaUI7S0FDL0I7Q0FDRixDQUFBO0FBRUQsZUFBZSxNQUFNLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgbm9DbGllbnRCbG9ja3MgZnJvbSAnLi9ydWxlcy9uby1jbGllbnQtYmxvY2tzLmpzJ1xuaW1wb3J0IG5vQ29udGFpbmVySW5TZWN0aW9uIGZyb20gJy4vcnVsZXMvbm8tY29udGFpbmVyLWluLXNlY3Rpb24uanMnXG5pbXBvcnQgcHJlZmVyQ29tcG9uZW50UHJvcHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItY29tcG9uZW50LXByb3BzLmpzJ1xuaW1wb3J0IHByZWZlclR5cG9ncmFwaHlDb21wb25lbnRzIGZyb20gJy4vcnVsZXMvcHJlZmVyLXR5cG9ncmFwaHktY29tcG9uZW50cy5qcydcbmltcG9ydCBzcGVlZHdlbGxDb25maWcgZnJvbSAnLi9jb25maWdzL3NwZWVkd2VsbC5qcydcbmltcG9ydCByZWNvbW1lbmRlZENvbmZpZyBmcm9tICcuL2NvbmZpZ3MvcmVjb21tZW5kZWQuanMnXG5cbmNvbnN0IHBsdWdpbiA9IHtcbiAgbWV0YToge1xuICAgIG5hbWU6ICdlc2xpbnQtcGx1Z2luLWdhbGxvcCcsXG4gICAgdmVyc2lvbjogJzEuMC4xJyxcbiAgfSxcbiAgcnVsZXM6IHtcbiAgICAnbm8tY2xpZW50LWJsb2Nrcyc6IG5vQ2xpZW50QmxvY2tzLFxuICAgICduby1jb250YWluZXItaW4tc2VjdGlvbic6IG5vQ29udGFpbmVySW5TZWN0aW9uLFxuICAgICdwcmVmZXItY29tcG9uZW50LXByb3BzJzogcHJlZmVyQ29tcG9uZW50UHJvcHMsXG4gICAgJ3ByZWZlci10eXBvZ3JhcGh5LWNvbXBvbmVudHMnOiBwcmVmZXJUeXBvZ3JhcGh5Q29tcG9uZW50cyxcbiAgfSxcbiAgY29uZmlnczoge1xuICAgIHNwZWVkd2VsbDogc3BlZWR3ZWxsQ29uZmlnLFxuICAgIHJlY29tbWVuZGVkOiByZWNvbW1lbmRlZENvbmZpZyxcbiAgfSxcbn1cblxuZXhwb3J0IGRlZmF1bHQgcGx1Z2luXG4iXX0=
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXNsaW50L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sY0FBYyxNQUFNLDZCQUE2QixDQUFBO0FBQ3hELE9BQU8sb0JBQW9CLE1BQU0sb0NBQW9DLENBQUE7QUFDckUsT0FBTyxvQkFBb0IsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNwRSxPQUFPLDBCQUEwQixNQUFNLHlDQUF5QyxDQUFBO0FBQ2hGLE9BQU8sc0JBQXNCLE1BQU0scUNBQXFDLENBQUE7QUFDeEUsT0FBTyxzQkFBc0IsTUFBTSxxQ0FBcUMsQ0FBQTtBQUN4RSxPQUFPLGVBQWUsTUFBTSx3QkFBd0IsQ0FBQTtBQUNwRCxPQUFPLGlCQUFpQixNQUFNLDBCQUEwQixDQUFBO0FBRXhELE1BQU0sTUFBTSxHQUFHO0lBQ2IsSUFBSSxFQUFFO1FBQ0osSUFBSSxFQUFFLHNCQUFzQjtRQUM1QixPQUFPLEVBQUUsT0FBTztLQUNqQjtJQUNELEtBQUssRUFBRTtRQUNMLGtCQUFrQixFQUFFLGNBQWM7UUFDbEMseUJBQXlCLEVBQUUsb0JBQW9CO1FBQy9DLHdCQUF3QixFQUFFLG9CQUFvQjtRQUM5Qyw4QkFBOEIsRUFBRSwwQkFBMEI7UUFDMUQsMEJBQTBCLEVBQUUsc0JBQXNCO1FBQ2xELDBCQUEwQixFQUFFLHNCQUFzQjtLQUNuRDtJQUNELE9BQU8sRUFBRTtRQUNQLFNBQVMsRUFBRSxlQUFlO1FBQzFCLFdBQVcsRUFBRSxpQkFBaUI7S0FDL0I7Q0FDRixDQUFBO0FBRUQsZUFBZSxNQUFNLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgbm9DbGllbnRCbG9ja3MgZnJvbSAnLi9ydWxlcy9uby1jbGllbnQtYmxvY2tzLmpzJ1xuaW1wb3J0IG5vQ29udGFpbmVySW5TZWN0aW9uIGZyb20gJy4vcnVsZXMvbm8tY29udGFpbmVyLWluLXNlY3Rpb24uanMnXG5pbXBvcnQgcHJlZmVyQ29tcG9uZW50UHJvcHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItY29tcG9uZW50LXByb3BzLmpzJ1xuaW1wb3J0IHByZWZlclR5cG9ncmFwaHlDb21wb25lbnRzIGZyb20gJy4vcnVsZXMvcHJlZmVyLXR5cG9ncmFwaHktY29tcG9uZW50cy5qcydcbmltcG9ydCBwcmVmZXJMYXlvdXRDb21wb25lbnRzIGZyb20gJy4vcnVsZXMvcHJlZmVyLWxheW91dC1jb21wb25lbnRzLmpzJ1xuaW1wb3J0IGJhY2tncm91bmRJbWFnZVJvdW5kZWQgZnJvbSAnLi9ydWxlcy9iYWNrZ3JvdW5kLWltYWdlLXJvdW5kZWQuanMnXG5pbXBvcnQgc3BlZWR3ZWxsQ29uZmlnIGZyb20gJy4vY29uZmlncy9zcGVlZHdlbGwuanMnXG5pbXBvcnQgcmVjb21tZW5kZWRDb25maWcgZnJvbSAnLi9jb25maWdzL3JlY29tbWVuZGVkLmpzJ1xuXG5jb25zdCBwbHVnaW4gPSB7XG4gIG1ldGE6IHtcbiAgICBuYW1lOiAnZXNsaW50LXBsdWdpbi1nYWxsb3AnLFxuICAgIHZlcnNpb246ICcxLjAuMScsXG4gIH0sXG4gIHJ1bGVzOiB7XG4gICAgJ25vLWNsaWVudC1ibG9ja3MnOiBub0NsaWVudEJsb2NrcyxcbiAgICAnbm8tY29udGFpbmVyLWluLXNlY3Rpb24nOiBub0NvbnRhaW5lckluU2VjdGlvbixcbiAgICAncHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6IHByZWZlckNvbXBvbmVudFByb3BzLFxuICAgICdwcmVmZXItdHlwb2dyYXBoeS1jb21wb25lbnRzJzogcHJlZmVyVHlwb2dyYXBoeUNvbXBvbmVudHMsXG4gICAgJ3ByZWZlci1sYXlvdXQtY29tcG9uZW50cyc6IHByZWZlckxheW91dENvbXBvbmVudHMsXG4gICAgJ2JhY2tncm91bmQtaW1hZ2Utcm91bmRlZCc6IGJhY2tncm91bmRJbWFnZVJvdW5kZWQsXG4gIH0sXG4gIGNvbmZpZ3M6IHtcbiAgICBzcGVlZHdlbGw6IHNwZWVkd2VsbENvbmZpZyxcbiAgICByZWNvbW1lbmRlZDogcmVjb21tZW5kZWRDb25maWcsXG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHBsdWdpblxuIl19
@@ -0,0 +1,3 @@
1
+ import type { Rule } from 'eslint';
2
+ declare const rule: Rule.RuleModule;
3
+ export default rule;
@@ -0,0 +1,122 @@
1
+ import { getCanonUrl, getCanonPattern } from '../utils/canon.js';
2
+ const RULE_NAME = 'background-image-rounded';
3
+ const pattern = getCanonPattern(RULE_NAME);
4
+ const rule = {
5
+ meta: {
6
+ type: 'suggestion',
7
+ docs: {
8
+ description: pattern?.summary || 'Background images must have rounded="rounded-none"',
9
+ recommended: true,
10
+ url: getCanonUrl(RULE_NAME),
11
+ },
12
+ messages: {
13
+ requireRoundedNone: `[Canon ${pattern?.id || '019'}] Background Image components (with absolute inset-0) must have rounded="rounded-none" to prevent corner clipping.`,
14
+ },
15
+ schema: [],
16
+ },
17
+ create(context) {
18
+ const filename = context.filename || context.getFilename();
19
+ // Only apply to block files
20
+ if (!filename.includes('/blocks/')) {
21
+ return {};
22
+ }
23
+ return {
24
+ JSXOpeningElement(node) {
25
+ const elementName = node.name?.name;
26
+ // Only check Image components
27
+ if (elementName !== 'Image') {
28
+ return;
29
+ }
30
+ // Check if className contains 'absolute' and 'inset-0'
31
+ const classNameAttr = node.attributes?.find((attr) => attr.type === 'JSXAttribute' &&
32
+ attr.name?.name === 'className');
33
+ if (!classNameAttr) {
34
+ return;
35
+ }
36
+ const classValue = getClassNameValue(classNameAttr);
37
+ if (!classValue) {
38
+ return;
39
+ }
40
+ // Check if this is a background image pattern
41
+ if (!isBackgroundImage(classValue)) {
42
+ return;
43
+ }
44
+ // Check if rounded="rounded-none" is set
45
+ const roundedAttr = node.attributes?.find((attr) => attr.type === 'JSXAttribute' &&
46
+ attr.name?.name === 'rounded');
47
+ if (!roundedAttr) {
48
+ context.report({
49
+ node,
50
+ messageId: 'requireRoundedNone',
51
+ });
52
+ return;
53
+ }
54
+ // Check the value of rounded prop
55
+ const roundedValue = getRoundedValue(roundedAttr);
56
+ if (roundedValue !== 'rounded-none') {
57
+ context.report({
58
+ node,
59
+ messageId: 'requireRoundedNone',
60
+ });
61
+ }
62
+ },
63
+ };
64
+ },
65
+ };
66
+ /**
67
+ * Extract className value from attribute
68
+ */
69
+ function getClassNameValue(attr) {
70
+ // Handle string literal
71
+ if (attr.value?.type === 'Literal' && typeof attr.value.value === 'string') {
72
+ return attr.value.value;
73
+ }
74
+ // Handle JSX expression container with template literal
75
+ if (attr.value?.type === 'JSXExpressionContainer') {
76
+ const expr = attr.value.expression;
77
+ if (expr.type === 'TemplateLiteral') {
78
+ // Combine all quasis
79
+ return expr.quasis?.map((q) => q.value?.raw || '').join(' ') || null;
80
+ }
81
+ // Handle clsx or other function calls - extract string arguments
82
+ if (expr.type === 'CallExpression') {
83
+ const strings = [];
84
+ for (const arg of expr.arguments || []) {
85
+ if (arg.type === 'Literal' && typeof arg.value === 'string') {
86
+ strings.push(arg.value);
87
+ }
88
+ if (arg.type === 'TemplateLiteral') {
89
+ strings.push(arg.quasis?.map((q) => q.value?.raw || '').join(' ') || '');
90
+ }
91
+ }
92
+ return strings.join(' ');
93
+ }
94
+ }
95
+ return null;
96
+ }
97
+ /**
98
+ * Check if className indicates a background image pattern
99
+ */
100
+ function isBackgroundImage(classValue) {
101
+ const classes = classValue.split(/\s+/);
102
+ const hasAbsolute = classes.includes('absolute');
103
+ const hasInset0 = classes.includes('inset-0');
104
+ return hasAbsolute && hasInset0;
105
+ }
106
+ /**
107
+ * Extract rounded prop value
108
+ */
109
+ function getRoundedValue(attr) {
110
+ if (attr.value?.type === 'Literal' && typeof attr.value.value === 'string') {
111
+ return attr.value.value;
112
+ }
113
+ if (attr.value?.type === 'JSXExpressionContainer') {
114
+ const expr = attr.value.expression;
115
+ if (expr.type === 'Literal' && typeof expr.value === 'string') {
116
+ return expr.value;
117
+ }
118
+ }
119
+ return null;
120
+ }
121
+ export default rule;
122
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"background-image-rounded.js","sourceRoot":"","sources":["../../../src/eslint/rules/background-image-rounded.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Rule } from 'eslint'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'background-image-rounded'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst rule: Rule.RuleModule = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Background images must have rounded=\"rounded-none\"',\n      recommended: true,\n      url: getCanonUrl(RULE_NAME),\n    },\n    messages: {\n      requireRoundedNone: `[Canon ${pattern?.id || '019'}] Background Image components (with absolute inset-0) must have rounded=\"rounded-none\" to prevent corner clipping.`,\n    },\n    schema: [],\n  },\n\n  create(context) {\n    const filename = context.filename || context.getFilename()\n\n    // Only apply to block files\n    if (!filename.includes('/blocks/')) {\n      return {}\n    }\n\n    return {\n      JSXOpeningElement(node: any) {\n        const elementName = node.name?.name\n\n        // Only check Image components\n        if (elementName !== 'Image') {\n          return\n        }\n\n        // Check if className contains 'absolute' and 'inset-0'\n        const classNameAttr = node.attributes?.find(\n          (attr: any) =>\n            attr.type === 'JSXAttribute' &&\n            attr.name?.name === 'className'\n        )\n\n        if (!classNameAttr) {\n          return\n        }\n\n        const classValue = getClassNameValue(classNameAttr)\n        if (!classValue) {\n          return\n        }\n\n        // Check if this is a background image pattern\n        if (!isBackgroundImage(classValue)) {\n          return\n        }\n\n        // Check if rounded=\"rounded-none\" is set\n        const roundedAttr = node.attributes?.find(\n          (attr: any) =>\n            attr.type === 'JSXAttribute' &&\n            attr.name?.name === 'rounded'\n        )\n\n        if (!roundedAttr) {\n          context.report({\n            node,\n            messageId: 'requireRoundedNone',\n          })\n          return\n        }\n\n        // Check the value of rounded prop\n        const roundedValue = getRoundedValue(roundedAttr)\n        if (roundedValue !== 'rounded-none') {\n          context.report({\n            node,\n            messageId: 'requireRoundedNone',\n          })\n        }\n      },\n    }\n  },\n}\n\n/**\n * Extract className value from attribute\n */\nfunction getClassNameValue(attr: any): string | null {\n  // Handle string literal\n  if (attr.value?.type === 'Literal' && typeof attr.value.value === 'string') {\n    return attr.value.value\n  }\n\n  // Handle JSX expression container with template literal\n  if (attr.value?.type === 'JSXExpressionContainer') {\n    const expr = attr.value.expression\n\n    if (expr.type === 'TemplateLiteral') {\n      // Combine all quasis\n      return expr.quasis?.map((q: any) => q.value?.raw || '').join(' ') || null\n    }\n\n    // Handle clsx or other function calls - extract string arguments\n    if (expr.type === 'CallExpression') {\n      const strings: string[] = []\n      for (const arg of expr.arguments || []) {\n        if (arg.type === 'Literal' && typeof arg.value === 'string') {\n          strings.push(arg.value)\n        }\n        if (arg.type === 'TemplateLiteral') {\n          strings.push(arg.quasis?.map((q: any) => q.value?.raw || '').join(' ') || '')\n        }\n      }\n      return strings.join(' ')\n    }\n  }\n\n  return null\n}\n\n/**\n * Check if className indicates a background image pattern\n */\nfunction isBackgroundImage(classValue: string): boolean {\n  const classes = classValue.split(/\\s+/)\n  const hasAbsolute = classes.includes('absolute')\n  const hasInset0 = classes.includes('inset-0')\n  \n  return hasAbsolute && hasInset0\n}\n\n/**\n * Extract rounded prop value\n */\nfunction getRoundedValue(attr: any): string | null {\n  if (attr.value?.type === 'Literal' && typeof attr.value.value === 'string') {\n    return attr.value.value\n  }\n\n  if (attr.value?.type === 'JSXExpressionContainer') {\n    const expr = attr.value.expression\n    if (expr.type === 'Literal' && typeof expr.value === 'string') {\n      return expr.value\n    }\n  }\n\n  return null\n}\n\nexport default rule\n"]}
@@ -0,0 +1,3 @@
1
+ import type { Rule } from 'eslint';
2
+ declare const rule: Rule.RuleModule;
3
+ export default rule;
@@ -0,0 +1,113 @@
1
+ import { getCanonUrl, getCanonPattern } from '../utils/canon.js';
2
+ const RULE_NAME = 'prefer-layout-components';
3
+ const pattern = getCanonPattern(RULE_NAME);
4
+ const rule = {
5
+ meta: {
6
+ type: 'suggestion',
7
+ docs: {
8
+ description: pattern?.summary || 'Use Grid/Columns, not raw div with grid',
9
+ recommended: true,
10
+ url: getCanonUrl(RULE_NAME),
11
+ },
12
+ messages: {
13
+ useLayoutComponent: `[Canon ${pattern?.id || '018'}] Use the Grid or Columns component instead of <div className="grid ...">. Import: import { Grid, Columns, Column } from "@/components"`,
14
+ },
15
+ schema: [],
16
+ },
17
+ create(context) {
18
+ const filename = context.filename || context.getFilename();
19
+ // Only apply to block files
20
+ if (!filename.includes('/blocks/')) {
21
+ return {};
22
+ }
23
+ return {
24
+ JSXOpeningElement(node) {
25
+ const elementName = node.name?.name;
26
+ // Only check div elements
27
+ if (elementName !== 'div') {
28
+ return;
29
+ }
30
+ // Check if className contains 'grid'
31
+ const classNameAttr = node.attributes?.find((attr) => attr.type === 'JSXAttribute' &&
32
+ attr.name?.name === 'className');
33
+ if (!classNameAttr) {
34
+ return;
35
+ }
36
+ // Handle string literal className
37
+ if (classNameAttr.value?.type === 'Literal') {
38
+ const classValue = classNameAttr.value.value;
39
+ if (typeof classValue === 'string' && hasGridClass(classValue)) {
40
+ context.report({
41
+ node,
42
+ messageId: 'useLayoutComponent',
43
+ });
44
+ }
45
+ return;
46
+ }
47
+ // Handle template literal className
48
+ if (classNameAttr.value?.type === 'JSXExpressionContainer') {
49
+ const expr = classNameAttr.value.expression;
50
+ // Direct template literal: className={`grid ...`}
51
+ if (expr.type === 'TemplateLiteral') {
52
+ const quasis = expr.quasis || [];
53
+ for (const quasi of quasis) {
54
+ if (quasi.value?.raw && hasGridClass(quasi.value.raw)) {
55
+ context.report({
56
+ node,
57
+ messageId: 'useLayoutComponent',
58
+ });
59
+ return;
60
+ }
61
+ }
62
+ }
63
+ // clsx call: className={clsx('grid', ...)}
64
+ if (expr.type === 'CallExpression') {
65
+ const args = expr.arguments || [];
66
+ for (const arg of args) {
67
+ if (arg.type === 'Literal' && typeof arg.value === 'string') {
68
+ if (hasGridClass(arg.value)) {
69
+ context.report({
70
+ node,
71
+ messageId: 'useLayoutComponent',
72
+ });
73
+ return;
74
+ }
75
+ }
76
+ // Check template literals in clsx args
77
+ if (arg.type === 'TemplateLiteral') {
78
+ const quasis = arg.quasis || [];
79
+ for (const quasi of quasis) {
80
+ if (quasi.value?.raw && hasGridClass(quasi.value.raw)) {
81
+ context.report({
82
+ node,
83
+ messageId: 'useLayoutComponent',
84
+ });
85
+ return;
86
+ }
87
+ }
88
+ }
89
+ }
90
+ }
91
+ }
92
+ },
93
+ };
94
+ },
95
+ };
96
+ /**
97
+ * Check if a className string contains grid classes
98
+ * Matches: 'grid', 'grid-cols-', etc.
99
+ * Does NOT match: 'grid-area', component names with 'grid' in them
100
+ */
101
+ function hasGridClass(classString) {
102
+ // Split by whitespace and check each class
103
+ const classes = classString.split(/\s+/);
104
+ for (const cls of classes) {
105
+ // Match standalone 'grid' or 'grid-cols-*' patterns
106
+ if (cls === 'grid' || cls.startsWith('grid-cols-')) {
107
+ return true;
108
+ }
109
+ }
110
+ return false;
111
+ }
112
+ export default rule;
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"prefer-layout-components.js","sourceRoot":"","sources":["../../../src/eslint/rules/prefer-layout-components.ts"],"names":[],"mappings":"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","sourcesContent":["import type { Rule } from 'eslint'\nimport { getCanonUrl, getCanonPattern } from '../utils/canon.js'\n\nconst RULE_NAME = 'prefer-layout-components'\nconst pattern = getCanonPattern(RULE_NAME)\n\nconst rule: Rule.RuleModule = {\n  meta: {\n    type: 'suggestion',\n    docs: {\n      description: pattern?.summary || 'Use Grid/Columns, not raw div with grid',\n      recommended: true,\n      url: getCanonUrl(RULE_NAME),\n    },\n    messages: {\n      useLayoutComponent: `[Canon ${pattern?.id || '018'}] Use the Grid or Columns component instead of <div className=\"grid ...\">. Import: import { Grid, Columns, Column } from \"@/components\"`,\n    },\n    schema: [],\n  },\n\n  create(context) {\n    const filename = context.filename || context.getFilename()\n\n    // Only apply to block files\n    if (!filename.includes('/blocks/')) {\n      return {}\n    }\n\n    return {\n      JSXOpeningElement(node: any) {\n        const elementName = node.name?.name\n\n        // Only check div elements\n        if (elementName !== 'div') {\n          return\n        }\n\n        // Check if className contains 'grid'\n        const classNameAttr = node.attributes?.find(\n          (attr: any) =>\n            attr.type === 'JSXAttribute' &&\n            attr.name?.name === 'className'\n        )\n\n        if (!classNameAttr) {\n          return\n        }\n\n        // Handle string literal className\n        if (classNameAttr.value?.type === 'Literal') {\n          const classValue = classNameAttr.value.value\n          if (typeof classValue === 'string' && hasGridClass(classValue)) {\n            context.report({\n              node,\n              messageId: 'useLayoutComponent',\n            })\n          }\n          return\n        }\n\n        // Handle template literal className\n        if (classNameAttr.value?.type === 'JSXExpressionContainer') {\n          const expr = classNameAttr.value.expression\n\n          // Direct template literal: className={`grid ...`}\n          if (expr.type === 'TemplateLiteral') {\n            const quasis = expr.quasis || []\n            for (const quasi of quasis) {\n              if (quasi.value?.raw && hasGridClass(quasi.value.raw)) {\n                context.report({\n                  node,\n                  messageId: 'useLayoutComponent',\n                })\n                return\n              }\n            }\n          }\n\n          // clsx call: className={clsx('grid', ...)}\n          if (expr.type === 'CallExpression') {\n            const args = expr.arguments || []\n            for (const arg of args) {\n              if (arg.type === 'Literal' && typeof arg.value === 'string') {\n                if (hasGridClass(arg.value)) {\n                  context.report({\n                    node,\n                    messageId: 'useLayoutComponent',\n                  })\n                  return\n                }\n              }\n              // Check template literals in clsx args\n              if (arg.type === 'TemplateLiteral') {\n                const quasis = arg.quasis || []\n                for (const quasi of quasis) {\n                  if (quasi.value?.raw && hasGridClass(quasi.value.raw)) {\n                    context.report({\n                      node,\n                      messageId: 'useLayoutComponent',\n                    })\n                    return\n                  }\n                }\n              }\n            }\n          }\n        }\n      },\n    }\n  },\n}\n\n/**\n * Check if a className string contains grid classes\n * Matches: 'grid', 'grid-cols-', etc.\n * Does NOT match: 'grid-area', component names with 'grid' in them\n */\nfunction hasGridClass(classString: string): boolean {\n  // Split by whitespace and check each class\n  const classes = classString.split(/\\s+/)\n  \n  for (const cls of classes) {\n    // Match standalone 'grid' or 'grid-cols-*' patterns\n    if (cls === 'grid' || cls.startsWith('grid-cols-')) {\n      return true\n    }\n  }\n  \n  return false\n}\n\nexport default rule\n"]}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@gallop.software/canon",
3
- "version": "2.0.2",
3
+ "version": "2.2.0",
4
4
  "type": "module",
5
5
  "description": "Gallop Canon - Architecture patterns, ESLint plugin, and CLI for template governance",
6
6
  "main": "dist/index.js",
@@ -0,0 +1,85 @@
1
+ # 018: Layout Components
2
+
3
+ **Category:** Layout
4
+ **Status:** Stable
5
+ **Enforcement:** ESLint (`gallop/prefer-layout-components`)
6
+
7
+ ## Summary
8
+
9
+ Use the `Grid` or `Columns` component instead of raw `<div>` elements with grid classes.
10
+
11
+ ## Rationale
12
+
13
+ Raw `<div>` elements with grid classes bypass the design system's layout abstractions. The `Grid` and `Columns` components provide:
14
+
15
+ - **Consistent defaults** for gaps, columns, and alignment
16
+ - **Semantic intent** - code is more readable when intent is clear
17
+ - **Centralized updates** - layout defaults can be updated in one place
18
+ - **Reduced errors** - no need to remember all required grid classes
19
+
20
+ ## Bad
21
+
22
+ ```tsx
23
+ <div className="grid grid-cols-3 gap-6">
24
+ <Card>...</Card>
25
+ <Card>...</Card>
26
+ <Card>...</Card>
27
+ </div>
28
+
29
+ <div className="grid grid-cols-1 lg:grid-cols-2 gap-8 items-center">
30
+ <div>Left content</div>
31
+ <div>Right content</div>
32
+ </div>
33
+ ```
34
+
35
+ ## Good
36
+
37
+ ```tsx
38
+ import { Grid, Columns, Column } from '@/components'
39
+
40
+ <Grid cols="grid-cols-3" gap="gap-6">
41
+ <Card>...</Card>
42
+ <Card>...</Card>
43
+ <Card>...</Card>
44
+ </Grid>
45
+
46
+ <Columns>
47
+ <Column>Left content</Column>
48
+ <Column>Right content</Column>
49
+ </Columns>
50
+ ```
51
+
52
+ ## Component Reference
53
+
54
+ ### Grid
55
+
56
+ For multi-item grid layouts (3+ columns, card grids, galleries):
57
+
58
+ ```tsx
59
+ <Grid
60
+ cols="grid-cols-1 lg:grid-cols-3" // optional, defaults to 1 → 3
61
+ gap="gap-6" // optional, has sensible default
62
+ >
63
+ {children}
64
+ </Grid>
65
+ ```
66
+
67
+ ### Columns
68
+
69
+ For two-column layouts with optional reversal:
70
+
71
+ ```tsx
72
+ <Columns
73
+ cols="grid-cols-1 lg:grid-cols-2" // optional
74
+ gap="gap-8" // optional
75
+ align="items-center" // optional
76
+ reverseColumns={true} // swap order on lg+
77
+ >
78
+ <Column>Left</Column>
79
+ <Column>Right</Column>
80
+ </Columns>
81
+ ```
82
+
83
+ ## Exceptions
84
+
85
+ The rule only applies to files in `/blocks/`. Component files may use raw divs with grid when building the layout primitives themselves.
@@ -0,0 +1,52 @@
1
+ # 019: Background Image Rounded
2
+
3
+ **Category:** Components
4
+ **Status:** Stable
5
+ **Enforcement:** ESLint (`gallop/background-image-rounded`)
6
+
7
+ ## Summary
8
+
9
+ When using the `Image` component as a background image (with `absolute inset-0` positioning), always set `rounded="rounded-none"`.
10
+
11
+ ## Rationale
12
+
13
+ The Image component defaults to `rounded-lg` for rounded corners. When used as a full-bleed background image, these rounded corners:
14
+
15
+ - **Cause visual artifacts** - corners get clipped unexpectedly
16
+ - **Conflict with container rounding** - the parent container should control edge styling
17
+ - **Create inconsistent edges** - background images should fill their container completely
18
+
19
+ ## Bad
20
+
21
+ ```tsx
22
+ {/* Background image without rounded prop - uses default rounded-lg */}
23
+ <Image
24
+ src="/images/hero-bg.jpg"
25
+ alt="Background"
26
+ className="absolute inset-0 w-full h-full object-cover"
27
+ />
28
+ ```
29
+
30
+ ## Good
31
+
32
+ ```tsx
33
+ {/* Background image with explicit rounded-none */}
34
+ <Image
35
+ src="/images/hero-bg.jpg"
36
+ alt="Background"
37
+ className="absolute inset-0 w-full h-full object-cover"
38
+ rounded="rounded-none"
39
+ />
40
+ ```
41
+
42
+ ## Detection
43
+
44
+ The rule identifies background images by checking for:
45
+ - `Image` component usage
46
+ - `className` containing both `absolute` and `inset-0`
47
+
48
+ When detected, the rule requires `rounded="rounded-none"` to be explicitly set.
49
+
50
+ ## Exceptions
51
+
52
+ This rule only applies to files in `/blocks/`. Component files that implement background image patterns internally are not flagged.
package/schema.json CHANGED
@@ -210,6 +210,26 @@
210
210
  "enforcement": "documentation",
211
211
  "rule": null,
212
212
  "summary": "PageMetadata structure, structured data"
213
+ },
214
+ {
215
+ "id": "018",
216
+ "title": "Layout Components",
217
+ "file": "patterns/018-layout-components.md",
218
+ "category": "layout",
219
+ "status": "stable",
220
+ "enforcement": "eslint",
221
+ "rule": "gallop/prefer-layout-components",
222
+ "summary": "Use Grid/Columns, not raw div with grid"
223
+ },
224
+ {
225
+ "id": "019",
226
+ "title": "Background Image Rounded",
227
+ "file": "patterns/019-background-image-rounded.md",
228
+ "category": "components",
229
+ "status": "stable",
230
+ "enforcement": "eslint",
231
+ "rule": "gallop/background-image-rounded",
232
+ "summary": "Background images must have rounded=\"rounded-none\""
213
233
  }
214
234
  ],
215
235
  "guarantees": [
@@ -239,7 +259,7 @@
239
259
  "name": "Design System Compliance",
240
260
  "since": "1.0.0",
241
261
  "status": "stable",
242
- "patterns": ["003", "004", "009", "010", "011"]
262
+ "patterns": ["003", "004", "009", "010", "011", "018", "019"]
243
263
  }
244
264
  ]
245
265
  }