@gallop.software/canon 2.1.0 → 2.2.1
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.
- package/dist/eslint/configs/recommended.d.ts +9 -10
- package/dist/eslint/configs/recommended.js +16 -12
- package/dist/eslint/configs/speedwell.d.ts +9 -10
- package/dist/eslint/configs/speedwell.js +16 -15
- package/dist/eslint/index.d.ts +13 -14
- package/dist/eslint/index.js +3 -1
- package/dist/eslint/rules/background-image-rounded.d.ts +3 -0
- package/dist/eslint/rules/background-image-rounded.js +122 -0
- package/package.json +1 -1
- package/patterns/019-background-image-rounded.md +52 -0
- package/schema.json +11 -1
|
@@ -1,14 +1,13 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Recommended configuration
|
|
2
|
+
* Recommended configuration for ESLint flat config
|
|
3
3
|
* A sensible default for any Gallop-based template
|
|
4
4
|
*/
|
|
5
|
-
declare const
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
};
|
|
5
|
+
declare const recommendedRules: {
|
|
6
|
+
readonly 'gallop/no-client-blocks': "warn";
|
|
7
|
+
readonly 'gallop/no-container-in-section': "warn";
|
|
8
|
+
readonly 'gallop/prefer-component-props': "warn";
|
|
9
|
+
readonly 'gallop/prefer-typography-components': "warn";
|
|
10
|
+
readonly 'gallop/prefer-layout-components': "warn";
|
|
11
|
+
readonly 'gallop/background-image-rounded': "warn";
|
|
13
12
|
};
|
|
14
|
-
export default
|
|
13
|
+
export default recommendedRules;
|
|
@@ -1,16 +1,20 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Recommended configuration
|
|
2
|
+
* Recommended configuration for ESLint flat config
|
|
3
3
|
* A sensible default for any Gallop-based template
|
|
4
4
|
*/
|
|
5
|
-
const
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
5
|
+
const recommendedRules = {
|
|
6
|
+
// Blocks should be server components
|
|
7
|
+
'gallop/no-client-blocks': 'warn',
|
|
8
|
+
// Section already provides containment
|
|
9
|
+
'gallop/no-container-in-section': 'warn',
|
|
10
|
+
// Use component props instead of className for style values
|
|
11
|
+
'gallop/prefer-component-props': 'warn',
|
|
12
|
+
// Use Typography components instead of raw p/span tags
|
|
13
|
+
'gallop/prefer-typography-components': '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
|
-
export default
|
|
16
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
19
|
+
export default recommendedRules;
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicmVjb21tZW5kZWQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvZXNsaW50L2NvbmZpZ3MvcmVjb21tZW5kZWQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7OztHQUdHO0FBQ0gsTUFBTSxnQkFBZ0IsR0FBRztJQUN2QixxQ0FBcUM7SUFDckMseUJBQXlCLEVBQUUsTUFBTTtJQUVqQyx1Q0FBdUM7SUFDdkMsZ0NBQWdDLEVBQUUsTUFBTTtJQUV4Qyw0REFBNEQ7SUFDNUQsK0JBQStCLEVBQUUsTUFBTTtJQUV2Qyx1REFBdUQ7SUFDdkQscUNBQXFDLEVBQUUsTUFBTTtJQUU3Qyx3REFBd0Q7SUFDeEQsaUNBQWlDLEVBQUUsTUFBTTtJQUV6QyxxREFBcUQ7SUFDckQsaUNBQWlDLEVBQUUsTUFBTTtDQUNqQyxDQUFBO0FBRVYsZUFBZSxnQkFBZ0IsQ0FBQSIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogUmVjb21tZW5kZWQgY29uZmlndXJhdGlvbiBmb3IgRVNMaW50IGZsYXQgY29uZmlnXG4gKiBBIHNlbnNpYmxlIGRlZmF1bHQgZm9yIGFueSBHYWxsb3AtYmFzZWQgdGVtcGxhdGVcbiAqL1xuY29uc3QgcmVjb21tZW5kZWRSdWxlcyA9IHtcbiAgLy8gQmxvY2tzIHNob3VsZCBiZSBzZXJ2ZXIgY29tcG9uZW50c1xuICAnZ2FsbG9wL25vLWNsaWVudC1ibG9ja3MnOiAnd2FybicsXG5cbiAgLy8gU2VjdGlvbiBhbHJlYWR5IHByb3ZpZGVzIGNvbnRhaW5tZW50XG4gICdnYWxsb3Avbm8tY29udGFpbmVyLWluLXNlY3Rpb24nOiAnd2FybicsXG5cbiAgLy8gVXNlIGNvbXBvbmVudCBwcm9wcyBpbnN0ZWFkIG9mIGNsYXNzTmFtZSBmb3Igc3R5bGUgdmFsdWVzXG4gICdnYWxsb3AvcHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6ICd3YXJuJyxcblxuICAvLyBVc2UgVHlwb2dyYXBoeSBjb21wb25lbnRzIGluc3RlYWQgb2YgcmF3IHAvc3BhbiB0YWdzXG4gICdnYWxsb3AvcHJlZmVyLXR5cG9ncmFwaHktY29tcG9uZW50cyc6ICd3YXJuJyxcblxuICAvLyBVc2UgR3JpZC9Db2x1bW5zIGluc3RlYWQgb2YgcmF3IGRpdiB3aXRoIGdyaWQgY2xhc3Nlc1xuICAnZ2FsbG9wL3ByZWZlci1sYXlvdXQtY29tcG9uZW50cyc6ICd3YXJuJyxcblxuICAvLyBCYWNrZ3JvdW5kIGltYWdlcyBtdXN0IGhhdmUgcm91bmRlZD1cInJvdW5kZWQtbm9uZVwiXG4gICdnYWxsb3AvYmFja2dyb3VuZC1pbWFnZS1yb3VuZGVkJzogJ3dhcm4nLFxufSBhcyBjb25zdFxuXG5leHBvcnQgZGVmYXVsdCByZWNvbW1lbmRlZFJ1bGVzXG4iXX0=
|
|
@@ -1,14 +1,13 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Speedwell template configuration
|
|
2
|
+
* Speedwell template configuration for ESLint flat config
|
|
3
3
|
* Enables all Gallop rules relevant to the Speedwell architecture
|
|
4
4
|
*/
|
|
5
|
-
declare const
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
};
|
|
5
|
+
declare const speedwellRules: {
|
|
6
|
+
readonly 'gallop/no-client-blocks': "warn";
|
|
7
|
+
readonly 'gallop/no-container-in-section': "warn";
|
|
8
|
+
readonly 'gallop/prefer-component-props': "warn";
|
|
9
|
+
readonly 'gallop/prefer-typography-components': "warn";
|
|
10
|
+
readonly 'gallop/prefer-layout-components': "warn";
|
|
11
|
+
readonly 'gallop/background-image-rounded': "warn";
|
|
13
12
|
};
|
|
14
|
-
export default
|
|
13
|
+
export default speedwellRules;
|
|
@@ -1,19 +1,20 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Speedwell template configuration
|
|
2
|
+
* Speedwell template configuration for ESLint flat config
|
|
3
3
|
* Enables all Gallop rules relevant to the Speedwell architecture
|
|
4
4
|
*/
|
|
5
|
-
const
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
5
|
+
const speedwellRules = {
|
|
6
|
+
// Blocks should be server components - extract client logic to components
|
|
7
|
+
'gallop/no-client-blocks': 'warn',
|
|
8
|
+
// Section already provides containment
|
|
9
|
+
'gallop/no-container-in-section': 'warn',
|
|
10
|
+
// Use component props instead of className for style values
|
|
11
|
+
'gallop/prefer-component-props': 'warn',
|
|
12
|
+
// Use Typography components instead of raw p/span tags
|
|
13
|
+
'gallop/prefer-typography-components': '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',
|
|
17
18
|
};
|
|
18
|
-
export default
|
|
19
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
19
|
+
export default speedwellRules;
|
|
20
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BlZWR3ZWxsLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vc3JjL2VzbGludC9jb25maWdzL3NwZWVkd2VsbC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLGNBQWMsR0FBRztJQUNyQiwwRUFBMEU7SUFDMUUseUJBQXlCLEVBQUUsTUFBTTtJQUVqQyx1Q0FBdUM7SUFDdkMsZ0NBQWdDLEVBQUUsTUFBTTtJQUV4Qyw0REFBNEQ7SUFDNUQsK0JBQStCLEVBQUUsTUFBTTtJQUV2Qyx1REFBdUQ7SUFDdkQscUNBQXFDLEVBQUUsTUFBTTtJQUU3Qyx3REFBd0Q7SUFDeEQsaUNBQWlDLEVBQUUsTUFBTTtJQUV6QyxxREFBcUQ7SUFDckQsaUNBQWlDLEVBQUUsTUFBTTtDQUNqQyxDQUFBO0FBRVYsZUFBZSxjQUFjLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIFNwZWVkd2VsbCB0ZW1wbGF0ZSBjb25maWd1cmF0aW9uIGZvciBFU0xpbnQgZmxhdCBjb25maWdcbiAqIEVuYWJsZXMgYWxsIEdhbGxvcCBydWxlcyByZWxldmFudCB0byB0aGUgU3BlZWR3ZWxsIGFyY2hpdGVjdHVyZVxuICovXG5jb25zdCBzcGVlZHdlbGxSdWxlcyA9IHtcbiAgLy8gQmxvY2tzIHNob3VsZCBiZSBzZXJ2ZXIgY29tcG9uZW50cyAtIGV4dHJhY3QgY2xpZW50IGxvZ2ljIHRvIGNvbXBvbmVudHNcbiAgJ2dhbGxvcC9uby1jbGllbnQtYmxvY2tzJzogJ3dhcm4nLFxuXG4gIC8vIFNlY3Rpb24gYWxyZWFkeSBwcm92aWRlcyBjb250YWlubWVudFxuICAnZ2FsbG9wL25vLWNvbnRhaW5lci1pbi1zZWN0aW9uJzogJ3dhcm4nLFxuXG4gIC8vIFVzZSBjb21wb25lbnQgcHJvcHMgaW5zdGVhZCBvZiBjbGFzc05hbWUgZm9yIHN0eWxlIHZhbHVlc1xuICAnZ2FsbG9wL3ByZWZlci1jb21wb25lbnQtcHJvcHMnOiAnd2FybicsXG5cbiAgLy8gVXNlIFR5cG9ncmFwaHkgY29tcG9uZW50cyBpbnN0ZWFkIG9mIHJhdyBwL3NwYW4gdGFnc1xuICAnZ2FsbG9wL3ByZWZlci10eXBvZ3JhcGh5LWNvbXBvbmVudHMnOiAnd2FybicsXG5cbiAgLy8gVXNlIEdyaWQvQ29sdW1ucyBpbnN0ZWFkIG9mIHJhdyBkaXYgd2l0aCBncmlkIGNsYXNzZXNcbiAgJ2dhbGxvcC9wcmVmZXItbGF5b3V0LWNvbXBvbmVudHMnOiAnd2FybicsXG5cbiAgLy8gQmFja2dyb3VuZCBpbWFnZXMgbXVzdCBoYXZlIHJvdW5kZWQ9XCJyb3VuZGVkLW5vbmVcIlxuICAnZ2FsbG9wL2JhY2tncm91bmQtaW1hZ2Utcm91bmRlZCc6ICd3YXJuJyxcbn0gYXMgY29uc3RcblxuZXhwb3J0IGRlZmF1bHQgc3BlZWR3ZWxsUnVsZXNcbiJdfQ==
|
package/dist/eslint/index.d.ts
CHANGED
|
@@ -15,25 +15,24 @@ declare const plugin: {
|
|
|
15
15
|
};
|
|
16
16
|
'prefer-typography-components': import("eslint").Rule.RuleModule;
|
|
17
17
|
'prefer-layout-components': import("eslint").Rule.RuleModule;
|
|
18
|
+
'background-image-rounded': import("eslint").Rule.RuleModule;
|
|
18
19
|
};
|
|
19
20
|
configs: {
|
|
20
21
|
speedwell: {
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
};
|
|
22
|
+
readonly 'gallop/no-client-blocks': "warn";
|
|
23
|
+
readonly 'gallop/no-container-in-section': "warn";
|
|
24
|
+
readonly 'gallop/prefer-component-props': "warn";
|
|
25
|
+
readonly 'gallop/prefer-typography-components': "warn";
|
|
26
|
+
readonly 'gallop/prefer-layout-components': "warn";
|
|
27
|
+
readonly 'gallop/background-image-rounded': "warn";
|
|
28
28
|
};
|
|
29
29
|
recommended: {
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
};
|
|
30
|
+
readonly 'gallop/no-client-blocks': "warn";
|
|
31
|
+
readonly 'gallop/no-container-in-section': "warn";
|
|
32
|
+
readonly 'gallop/prefer-component-props': "warn";
|
|
33
|
+
readonly 'gallop/prefer-typography-components': "warn";
|
|
34
|
+
readonly 'gallop/prefer-layout-components': "warn";
|
|
35
|
+
readonly 'gallop/background-image-rounded': "warn";
|
|
37
36
|
};
|
|
38
37
|
};
|
|
39
38
|
};
|
package/dist/eslint/index.js
CHANGED
|
@@ -3,6 +3,7 @@ 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
5
|
import preferLayoutComponents from './rules/prefer-layout-components.js';
|
|
6
|
+
import backgroundImageRounded from './rules/background-image-rounded.js';
|
|
6
7
|
import speedwellConfig from './configs/speedwell.js';
|
|
7
8
|
import recommendedConfig from './configs/recommended.js';
|
|
8
9
|
const plugin = {
|
|
@@ -16,6 +17,7 @@ const plugin = {
|
|
|
16
17
|
'prefer-component-props': preferComponentProps,
|
|
17
18
|
'prefer-typography-components': preferTypographyComponents,
|
|
18
19
|
'prefer-layout-components': preferLayoutComponents,
|
|
20
|
+
'background-image-rounded': backgroundImageRounded,
|
|
19
21
|
},
|
|
20
22
|
configs: {
|
|
21
23
|
speedwell: speedwellConfig,
|
|
@@ -23,4 +25,4 @@ const plugin = {
|
|
|
23
25
|
},
|
|
24
26
|
};
|
|
25
27
|
export default plugin;
|
|
26
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvZXNsaW50L2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sY0FBYyxNQUFNLDZCQUE2QixDQUFBO0FBQ3hELE9BQU8sb0JBQW9CLE1BQU0sb0NBQW9DLENBQUE7QUFDckUsT0FBTyxvQkFBb0IsTUFBTSxtQ0FBbUMsQ0FBQTtBQUNwRSxPQUFPLDBCQUEwQixNQUFNLHlDQUF5QyxDQUFBO0FBQ2hGLE9BQU8sc0JBQXNCLE1BQU0scUNBQXFDLENBQUE7QUFDeEUsT0FBTyxzQkFBc0IsTUFBTSxxQ0FBcUMsQ0FBQTtBQUN4RSxPQUFPLGVBQWUsTUFBTSx3QkFBd0IsQ0FBQTtBQUNwRCxPQUFPLGlCQUFpQixNQUFNLDBCQUEwQixDQUFBO0FBRXhELE1BQU0sTUFBTSxHQUFHO0lBQ2IsSUFBSSxFQUFFO1FBQ0osSUFBSSxFQUFFLHNCQUFzQjtRQUM1QixPQUFPLEVBQUUsT0FBTztLQUNqQjtJQUNELEtBQUssRUFBRTtRQUNMLGtCQUFrQixFQUFFLGNBQWM7UUFDbEMseUJBQXlCLEVBQUUsb0JBQW9CO1FBQy9DLHdCQUF3QixFQUFFLG9CQUFvQjtRQUM5Qyw4QkFBOEIsRUFBRSwwQkFBMEI7UUFDMUQsMEJBQTBCLEVBQUUsc0JBQXNCO1FBQ2xELDBCQUEwQixFQUFFLHNCQUFzQjtLQUNuRDtJQUNELE9BQU8sRUFBRTtRQUNQLFNBQVMsRUFBRSxlQUFlO1FBQzFCLFdBQVcsRUFBRSxpQkFBaUI7S0FDL0I7Q0FDRixDQUFBO0FBRUQsZUFBZSxNQUFNLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgbm9DbGllbnRCbG9ja3MgZnJvbSAnLi9ydWxlcy9uby1jbGllbnQtYmxvY2tzLmpzJ1xuaW1wb3J0IG5vQ29udGFpbmVySW5TZWN0aW9uIGZyb20gJy4vcnVsZXMvbm8tY29udGFpbmVyLWluLXNlY3Rpb24uanMnXG5pbXBvcnQgcHJlZmVyQ29tcG9uZW50UHJvcHMgZnJvbSAnLi9ydWxlcy9wcmVmZXItY29tcG9uZW50LXByb3BzLmpzJ1xuaW1wb3J0IHByZWZlclR5cG9ncmFwaHlDb21wb25lbnRzIGZyb20gJy4vcnVsZXMvcHJlZmVyLXR5cG9ncmFwaHktY29tcG9uZW50cy5qcydcbmltcG9ydCBwcmVmZXJMYXlvdXRDb21wb25lbnRzIGZyb20gJy4vcnVsZXMvcHJlZmVyLWxheW91dC1jb21wb25lbnRzLmpzJ1xuaW1wb3J0IGJhY2tncm91bmRJbWFnZVJvdW5kZWQgZnJvbSAnLi9ydWxlcy9iYWNrZ3JvdW5kLWltYWdlLXJvdW5kZWQuanMnXG5pbXBvcnQgc3BlZWR3ZWxsQ29uZmlnIGZyb20gJy4vY29uZmlncy9zcGVlZHdlbGwuanMnXG5pbXBvcnQgcmVjb21tZW5kZWRDb25maWcgZnJvbSAnLi9jb25maWdzL3JlY29tbWVuZGVkLmpzJ1xuXG5jb25zdCBwbHVnaW4gPSB7XG4gIG1ldGE6IHtcbiAgICBuYW1lOiAnZXNsaW50LXBsdWdpbi1nYWxsb3AnLFxuICAgIHZlcnNpb246ICcxLjAuMScsXG4gIH0sXG4gIHJ1bGVzOiB7XG4gICAgJ25vLWNsaWVudC1ibG9ja3MnOiBub0NsaWVudEJsb2NrcyxcbiAgICAnbm8tY29udGFpbmVyLWluLXNlY3Rpb24nOiBub0NvbnRhaW5lckluU2VjdGlvbixcbiAgICAncHJlZmVyLWNvbXBvbmVudC1wcm9wcyc6IHByZWZlckNvbXBvbmVudFByb3BzLFxuICAgICdwcmVmZXItdHlwb2dyYXBoeS1jb21wb25lbnRzJzogcHJlZmVyVHlwb2dyYXBoeUNvbXBvbmVudHMsXG4gICAgJ3ByZWZlci1sYXlvdXQtY29tcG9uZW50cyc6IHByZWZlckxheW91dENvbXBvbmVudHMsXG4gICAgJ2JhY2tncm91bmQtaW1hZ2Utcm91bmRlZCc6IGJhY2tncm91bmRJbWFnZVJvdW5kZWQsXG4gIH0sXG4gIGNvbmZpZ3M6IHtcbiAgICBzcGVlZHdlbGw6IHNwZWVkd2VsbENvbmZpZyxcbiAgICByZWNvbW1lbmRlZDogcmVjb21tZW5kZWRDb25maWcsXG4gIH0sXG59XG5cbmV4cG9ydCBkZWZhdWx0IHBsdWdpblxuIl19
|
|
@@ -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"]}
|
package/package.json
CHANGED
|
@@ -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
|
@@ -220,6 +220,16 @@
|
|
|
220
220
|
"enforcement": "eslint",
|
|
221
221
|
"rule": "gallop/prefer-layout-components",
|
|
222
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\""
|
|
223
233
|
}
|
|
224
234
|
],
|
|
225
235
|
"guarantees": [
|
|
@@ -249,7 +259,7 @@
|
|
|
249
259
|
"name": "Design System Compliance",
|
|
250
260
|
"since": "1.0.0",
|
|
251
261
|
"status": "stable",
|
|
252
|
-
"patterns": ["003", "004", "009", "010", "011", "018"]
|
|
262
|
+
"patterns": ["003", "004", "009", "010", "011", "018", "019"]
|
|
253
263
|
}
|
|
254
264
|
]
|
|
255
265
|
}
|