用javascript为css规则添加前缀

x0fgdtte  于 2023-04-10  发布在  Java
关注(0)|答案(6)|浏览(176)

我有一个字符串,下面的CSS,我需要处理与JavaScript

h1
{
    color: red;
}

.info
{
    border: 1px dotted blue;
    padding: 10px;
}

#rect
{
    background: pink;
}

h2,
h3,
h4
{
    font-weight: bold;
}

table td:last td
{
    background: gainsboro;
}

我怎样才能给每个规则添加前缀.page,这样css就不会被破坏?
我想要这个结果

.page h1
{
    color: red;
}

.page .info
{
    border: 1px dotted blue;
    padding: 10px;
}

...

现在,我通过查找缩进来解决它,但是代码在这种情况下失败了

h1
{
color: red;
}

最后就变成了

.page h1
{
.page color: red;
}

我还可以查找只有方括号的行,但这种情况会失败

h1 { color: red; }

我不想构建自己的css解析器,我发现的那些解析器主要处理应用于DOM中元素的css,而不是用css处理字符串。有没有一个好的解析器,或者可以通过其他方式实现?

ijnw1ujt

ijnw1ujt1#

下面是一个函数,它可以在所有选择器前面加上一个类名。这个函数正确地处理了@ rules:

var prefixCssSelectors = function(rules, className) {
  var classLen = className.length,
    char, nextChar, isAt, isIn;

  // makes sure the className will not concatenate the selector
  className += ' ';

  // removes comments
  rules = rules.replace( /\/\*(?:(?!\*\/)[\s\S])*\*\/|[\r\n\t]+/g, '' );

  // makes sure nextChar will not target a space
  rules = rules.replace( /}(\s*)@/g, '}@' );
  rules = rules.replace( /}(\s*)}/g, '}}' );

  for (var i = 0; i < rules.length-2; i++) {
    char = rules[i];
    nextChar = rules[i+1];

    if (char === '@' && nextChar !== 'f') isAt = true;
    if (!isAt && char === '{') isIn = true;
    if (isIn && char === '}') isIn = false;

    if (
      !isIn &&
      nextChar !== '@' &&
      nextChar !== '}' &&
      (
        char === '}' ||
        char === ',' ||
        ((char === '{' || char === ';') && isAt)
      )
    ) {
      rules = rules.slice(0, i+1) + className + rules.slice(i+1);
      i += classLen;
      isAt = false;
    }
  };

  // prefix the first select if it is not `@media` and if it is not yet prefixed
  if (rules.indexOf(className) !== 0 && rules.indexOf('@') !== 0) rules = className+rules;

  return rules;
}

// Some examples:
console.log(prefixCssSelectors('div { width: 100%; }', '.page'));

console.log(prefixCssSelectors('@charset "utf-8"; div { width: 100%; }', '.page'));

console.log(prefixCssSelectors('@media only screen { div { width: 100%; } p { size: 1.2rem; } } @media only print { p { size: 1.2rem; } } div { height: 100%; font-family: "Arial", Times; }', '.page'));

console.log(prefixCssSelectors('@font-face { font-family: "Open Sans"; src: url("/fonts/OpenSans-Regular-webfont.woff2") format("woff2"); } div { width: 100%; }', '.page'));

如果你只想有htmlbody选择器的类名,添加这个:

rules = rules.replace(/( html| body)/g, '');
ilmyapht

ilmyapht2#

你已经提出了一个问题,如何做你认为是你的实际问题的解决方案,这似乎是,我如何将一些CSS只应用到一个文档的一个部分?
最终,解决方案将是<style scoped>。在我们等待的同时,您可以尝试使用polyfill,例如https://github.com/PM5544/scoped-polyfillhttp://thomaspark.co/2015/07/polyfill-for-scoped-css/https://github.com/thingsinjars/jQuery-Scoped-CSS-pluginhttps://github.com/thomaspark/scoper
如果你有一个CSS字符串(你从哪里得到的?),那么要使用这些polyfill,你可以简单地在.page元素下插入一个<style scoped>元素到你的DOM中,包含你在字符串中得到的CSS。
所有这些polyfill都有一个主要的优点,那就是它们是基于浏览器正确解析的CSS,并且不会在出现问题的第一个迹象时就崩溃,因为你不明智地接受了regexp解决方案,而所有其他regexp解决方案都不可避免地会崩溃。

9rygscc1

9rygscc13#

如果你想使用sass,你可以简单地写如下

.page {
  h1 {
    color: red;
  }

  h2 {
    color: green;
  }
}

这将被编译成你想要的。但是如果这不是一个选择,你必须编写自己的解析器。

yftpprvb

yftpprvb4#

一个解决方案是,循环CSS规则并动态更新它们。

/* debug purpose */
var j = JSON.stringify;
var el = document.getElementById('el');
var log = function( val ){el.innerHTML+='<div>'+val+'</div>'}

var doIt = function(){

// you have to identify your stylesheet to get it
var ss = document.styleSheets[0];
// we get all the rules from this stylesheets
var rules = ss.cssRules;

// we loop over all rules
for( var i = 0 ; i < rules.length; i++){
  var rule = rules[i];
  
  var selector = rule.selectorText;
  var def = rule.cssText.replace(selector , '');
  
  // we update the selector
  var selector2 = selector.replace(/([^,]+,?)/g , '.page $1 ' ); 

  // we modify the rules on the fly !
  var def2 = def.replace('red' , 'green');
  def2 = def2.replace('blue' , 'red');
  def2 = def2.replace('pink' , 'lime');
  
  log(i);
  log( 'def : ' + def);
  log( 'modified : ' + def2);
  log( 'selector : ' + selector);
  log( 'updated : ' + selector2);
  log('----------');
  ss.deleteRule(i);// we remove the old 
  ss.insertRule(selector2 + def2 , i);// we add the new 
  
};

  
  // we check the stylecheet !
  log( ' checking the results');
  for( var i = 0 ; i < rules.length; i++){
  var rule = rules[i];
  
  var curRule = rule.cssText;
  log( i + ':  curRule => ' + curRule);
 
};
  
};
h1
{
    color: red;
}

.info
{
    border: 1px dotted blue;
    padding: 10px;
}

#rect
{
    background: pink;
}

h2,
h3,
h4
{
    font-weight: bold;
}

table td:last td
{
    background: gainsboro;
}
<div class='page'>
  <div id='rect'> RECT </div>
  <div class='info'> INFO </div>
<div>
<button onclick='doIt()'>update it</button>
<div id='el'><div>
eivgtgni

eivgtgni5#

如果你不想使用Regex,而想利用postcss,你可以安装postcss-prefix-selector并这样做:

import postcss from "postcss"
import prefixer from "postcss-prefix-selector"

const inputCssString = `ul { padding: 20px 0; flex: 1; } li { font-family:'Lato'; color: whitesmoke; line-height: 44px; }`
const out = postcss()
            .use(
                prefixer({
                    prefix: `.my-prefix`,
                })
            )
            .process(inputCssString).css

返回的输出:

.my-prefix ul { padding: 20px 0; flex: 1; } .my-prefix li { font-family:'Lato'; color: whitesmoke; line-height: 44px; }
wfsdck30

wfsdck306#

看看正则表达式
例如像这样:

/([^]*?)({[^]*?}|,)/g

应该在第一组中捕获css字符串中的所有选择器,在第二组中捕获和规则。然后你做类似这样的事情来替换匹配项

var str = "your css text here";
re = /([^]*?)({[^]*?}|,)/g;
var new_str = str.replace(re, ".prefix $1 $2"); //prefixed css
console.log(new_str);

如果您不熟悉正则表达式,我建议您阅读Eloquent Javascript中的相关章节

相关问题