postcss-best-practices

安装量: 75
排名: #10421

安装

npx skills add https://github.com/mindrally/skills --skill postcss-best-practices

PostCSS Best Practices

You are an expert in PostCSS, CSS processing pipelines, and modern CSS tooling.

Key Principles Use PostCSS as a modular CSS processor with purpose-specific plugins Write future-proof CSS using modern syntax with appropriate transpilation Optimize CSS output for production with minification and autoprefixing Keep plugin configurations minimal and purposeful What is PostCSS

PostCSS is a tool for transforming CSS with JavaScript plugins. Unlike preprocessors (Sass/Less), PostCSS:

Is modular - add only the features you need Can parse and transform standard CSS Enables future CSS syntax today Optimizes and minifies output Works with any build tool Project Setup Installation

Core PostCSS

npm install postcss postcss-cli --save-dev

Common plugins

npm install autoprefixer cssnano postcss-preset-env --save-dev

Optional plugins

npm install postcss-import postcss-nested postcss-custom-media --save-dev

Configuration File // postcss.config.js module.exports = { plugins: [ require('postcss-import'), require('postcss-preset-env')({ stage: 2, features: { 'nesting-rules': true, 'custom-media-queries': true, 'custom-properties': true, }, }), require('autoprefixer'), require('cssnano')({ preset: ['default', { discardComments: { removeAll: true }, }], }), ], };

Build Tool Integration Webpack // webpack.config.js module.exports = { module: { rules: [ { test: /.css$/, use: [ 'style-loader', 'css-loader', 'postcss-loader', ], }, ], }, };

Vite // vite.config.js import { defineConfig } from 'vite';

export default defineConfig({ css: { postcss: './postcss.config.js', }, });

Next.js // postcss.config.js (auto-detected) module.exports = { plugins: { 'postcss-import': {}, 'tailwindcss/nesting': {}, tailwindcss: {}, autoprefixer: {}, }, };

Essential Plugins postcss-preset-env

Enables modern CSS features with polyfills:

// postcss.config.js module.exports = { plugins: [ require('postcss-preset-env')({ stage: 2, // Stage 2 features are stable features: { 'nesting-rules': true, 'custom-media-queries': true, 'custom-selectors': true, 'gap-properties': true, 'logical-properties-and-values': true, }, autoprefixer: { grid: true }, browsers: 'last 2 versions', }), ], };

autoprefixer

Adds vendor prefixes automatically:

// postcss.config.js module.exports = { plugins: [ require('autoprefixer')({ grid: 'autoplace', // Enable grid prefixes flexbox: true, }), ], };

// Input CSS .element { display: flex; user-select: none; }

// Output CSS .element { display: -webkit-box; display: -ms-flexbox; display: flex; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; }

cssnano

Minifies and optimizes CSS for production:

// postcss.config.js const isProduction = process.env.NODE_ENV === 'production';

module.exports = { plugins: [ require('autoprefixer'), isProduction && require('cssnano')({ preset: ['default', { discardComments: { removeAll: true }, normalizeWhitespace: true, minifyFontValues: true, minifyGradients: true, reduceIdents: false, // Preserve animation names mergeRules: true, mergeLonghand: true, }], }), ].filter(Boolean), };

postcss-import

Inline @import statements:

// postcss.config.js module.exports = { plugins: [ require('postcss-import')({ path: ['src/styles'], }), ], };

/ main.css / @import 'variables.css'; @import 'base.css'; @import 'components/buttons.css'; @import 'components/cards.css';

postcss-nested

Enables Sass-like nesting:

// postcss.config.js module.exports = { plugins: [ require('postcss-nested'), ], };

/ Input / .card { background: white;

&__header { padding: 16px; }

&__body { padding: 16px; }

&:hover { box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); }

@media (min-width: 768px) { display: flex; } }

/ Output / .card { background: white; } .card__header { padding: 16px; } .card__body { padding: 16px; } .card:hover { box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); } @media (min-width: 768px) { .card { display: flex; } }

Modern CSS Features CSS Nesting (Native) / Modern browsers support native nesting / .card { background: white; border-radius: 8px;

& .card-header { padding: 1rem; border-bottom: 1px solid #eee; }

& .card-body { padding: 1rem; }

&:hover { box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1); }

@media (min-width: 768px) { display: flex; } }

Custom Properties (CSS Variables) :root { / Colors / --color-primary: #3498db; --color-primary-light: color-mix(in srgb, var(--color-primary), white 20%); --color-primary-dark: color-mix(in srgb, var(--color-primary), black 20%); --color-text: #333333; --color-background: #ffffff; --color-border: #e0e0e0;

/ Typography / --font-family-base: 'Helvetica Neue', Arial, sans-serif; --font-size-base: 1rem; --line-height-base: 1.5;

/ Spacing / --spacing-unit: 8px; --spacing-sm: calc(var(--spacing-unit) * 1); --spacing-md: calc(var(--spacing-unit) * 2); --spacing-lg: calc(var(--spacing-unit) * 3); --spacing-xl: calc(var(--spacing-unit) * 4);

/ Transitions / --transition-base: 0.3s ease;

/ Shadows / --shadow-sm: 0 1px 2px rgba(0, 0, 0, 0.05); --shadow-md: 0 4px 6px rgba(0, 0, 0, 0.1); }

.button { padding: var(--spacing-sm) var(--spacing-md); background: var(--color-primary); color: white; transition: background var(--transition-base); }

.button:hover { background: var(--color-primary-dark); }

Custom Media Queries / Define custom media queries / @custom-media --viewport-sm (min-width: 576px); @custom-media --viewport-md (min-width: 768px); @custom-media --viewport-lg (min-width: 992px); @custom-media --viewport-xl (min-width: 1200px);

@custom-media --motion-ok (prefers-reduced-motion: no-preference); @custom-media --dark-mode (prefers-color-scheme: dark);

/ Usage / .element { width: 100%; }

@media (--viewport-md) { .element { width: 50%; } }

@media (--viewport-lg) { .element { width: 33.333%; } }

@media (--motion-ok) { .element { transition: transform 0.3s ease; } }

Custom Selectors / Define reusable selectors / @custom-selector :--heading h1, h2, h3, h4, h5, h6; @custom-selector :--button button, [type="button"], [type="submit"]; @custom-selector :--enter :hover, :focus;

/ Usage / :--heading { font-family: var(--font-family-heading); line-height: 1.2; margin-bottom: 1rem; }

:--button { cursor: pointer; font-family: inherit; }

.link:--enter { color: var(--color-primary); text-decoration: underline; }

Logical Properties / Use logical properties for internationalization / .element { / Instead of margin-left/right / margin-inline: auto;

/ Instead of padding-top/bottom / padding-block: var(--spacing-md);

/ Instead of width/height / inline-size: 100%; block-size: auto;

/ Instead of border-left / border-inline-start: 2px solid var(--color-primary);

/ Instead of text-align: left / text-align: start; }

Container Queries / Modern container queries / .card-container { container-type: inline-size; container-name: card; }

.card { display: block; }

@container card (min-width: 400px) { .card { display: flex; } }

@container card (min-width: 600px) { .card { gap: 2rem; } }

File Organization Structure src/ ├── styles/ │ ├── base/ │ │ ├── reset.css │ │ ├── typography.css │ │ └── base.css │ ├── components/ │ │ ├── buttons.css │ │ ├── cards.css │ │ ├── forms.css │ │ └── navigation.css │ ├── layout/ │ │ ├── header.css │ │ ├── footer.css │ │ └── grid.css │ ├── utilities/ │ │ ├── spacing.css │ │ ├── colors.css │ │ └── display.css │ ├── variables.css │ └── main.css ├── postcss.config.js └── package.json

Main Entry Point / main.css /

/ Variables first / @import 'variables.css';

/ Reset and base styles / @import 'base/reset.css'; @import 'base/typography.css'; @import 'base/base.css';

/ Layout / @import 'layout/grid.css'; @import 'layout/header.css'; @import 'layout/footer.css';

/ Components / @import 'components/buttons.css'; @import 'components/cards.css'; @import 'components/forms.css'; @import 'components/navigation.css';

/ Utilities (last for specificity) / @import 'utilities/spacing.css'; @import 'utilities/colors.css'; @import 'utilities/display.css';

Production Configuration Environment-Based Config // postcss.config.js const isProduction = process.env.NODE_ENV === 'production';

module.exports = { plugins: [ // Always run these require('postcss-import'), require('postcss-preset-env')({ stage: 2, features: { 'nesting-rules': true, 'custom-media-queries': true, }, }), require('autoprefixer'),

// Production only
isProduction && require('cssnano')({
  preset: ['default', {
    discardComments: { removeAll: true },
    reduceIdents: false,
    zindex: false,
  }],
}),

isProduction && require('@fullhuman/postcss-purgecss')({
  content: [
    './src/**/*.html',
    './src/**/*.js',
    './src/**/*.jsx',
    './src/**/*.tsx',
  ],
  safelist: {
    standard: [/^is-/, /^has-/, /^js-/],
    deep: [/modal/, /tooltip/],
  },
}),

].filter(Boolean), };

Browser Support // package.json { "browserslist": [ "> 1%", "last 2 versions", "not dead", "not ie 11" ] }

// Or .browserslistrc

1% last 2 versions not dead not ie 11

Useful Plugin Configurations postcss-custom-media require('postcss-custom-media')({ importFrom: [ { customMedia: { '--viewport-sm': '(min-width: 576px)', '--viewport-md': '(min-width: 768px)', '--viewport-lg': '(min-width: 992px)', '--viewport-xl': '(min-width: 1200px)', }, }, ], });

postcss-mixins require('postcss-mixins')({ mixins: { 'flex-center': { display: 'flex', 'align-items': 'center', 'justify-content': 'center', }, 'visually-hidden': { position: 'absolute', width: '1px', height: '1px', overflow: 'hidden', clip: 'rect(0, 0, 0, 0)', }, }, });

postcss-simple-vars require('postcss-simple-vars')({ variables: { 'color-primary': '#3498db', 'spacing-unit': '8px', }, });

Performance Tips Order plugins correctly: import, preset-env/nesting, autoprefixer, cssnano Use PurgeCSS to remove unused styles in production Enable source maps only in development Cache PostCSS results in build tools when possible Avoid unnecessary plugins - each adds processing time Use browserslist to target only necessary browsers Common Plugin Combinations Minimal Setup module.exports = { plugins: [ require('autoprefixer'), require('cssnano'), ], };

Modern CSS Features module.exports = { plugins: [ require('postcss-import'), require('postcss-preset-env')({ stage: 2 }), require('autoprefixer'), require('cssnano'), ], };

Sass-like Features module.exports = { plugins: [ require('postcss-import'), require('postcss-mixins'), require('postcss-simple-vars'), require('postcss-nested'), require('autoprefixer'), require('cssnano'), ], };

With Tailwind CSS module.exports = { plugins: { 'postcss-import': {}, 'tailwindcss/nesting': {}, tailwindcss: {}, autoprefixer: {}, ...(process.env.NODE_ENV === 'production' ? { cssnano: {} } : {}), }, };

Debugging Enable Source Maps // postcss.config.js module.exports = { map: process.env.NODE_ENV !== 'production' ? { inline: false } : false, plugins: [ // ...plugins ], };

Verbose Output

CLI with verbose output

postcss src/main.css -o dist/main.css --verbose

Check Plugin Order

If styles aren't working as expected:

Check plugin order (import before others) Verify preset-env stage settings Check browserslist configuration Review cssnano options (may be over-optimizing)

返回排行榜