Изменить цвет заливки SVG при подаче в качестве фона-Image

размещение вывода SVG непосредственно в строке с кодом страницы я могу просто изменить цвета заливки с помощью CSS, например:

polygon.mystar {
    fill: blue;
}​

circle.mycircle {
    fill: green;
}

Это отлично работает, однако я ищу способ изменить атрибут "fill" SVG, когда он служит в качестве фонового изображения.

html {      
    background-image: url(../img/bg.svg);
}

как я могу изменить цвет? Это вообще возможно?

Для справки, вот содержимое моего внешнего SVG-файла:

<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
     width="320px" height="100px" viewBox="0 0 320 100" enable-background="new 0 0 320 100" xml:space="preserve">
<polygon class="mystar" fill="#3CB54A" points="134.973,14.204 143.295,31.066 161.903,33.77 148.438,46.896 151.617,65.43 134.973,56.679 
    118.329,65.43 121.507,46.896 108.042,33.77 126.65,31.066 "/>
<circle class="mycircle" fill="#ED1F24" cx="202.028" cy="58.342" r="12.26"/>
</svg>

14 ответов


Я думаю, что единственный способ сделать это-обслуживать svg с какого-то механизма на стороне сервера. Просто создайте сервер ресурсов, который выводит svg в соответствии с параметрами GET, и вы обслуживаете его по определенному url-адресу.

тогда вы просто используете этот url в своем css.

потому что в качестве фонового img он не является частью DOM, и вы не можете им манипулировать. Другой возможностью было бы использовать его регулярно, встроить его в страницу обычным способом, но позиционировать его абсолютно, сделайте его полной шириной и высотой страницы, а затем используйте свойство css z-index, чтобы поместить его за все другие элементы DOM на странице.


мне нужно было что-то подобное и я хотел придерживаться CSS. Вот меньше и SCSS mixins, а также простой CSS, которые могут помочь вам в этом. К сожалению, поддержка браузера немного слабая. Подробнее о поддержке браузера см. ниже.

меньше mixin:

.element-color(@color) {
  background-image: url('data:image/svg+xml;utf8,<svg ...><g stroke="@{color}" ... /></g></svg>');
}

меньшее потребление:

.element-color(#fff);

SCSS mixin:

@mixin element-color($color) {
  background-image: url('data:image/svg+xml;utf8,<svg ...><g stroke="#{$color}" ... /></g></svg>');
}

SCSS использование:

@include element-color(#fff);

CSS:

// color: red
background-image: url('data:image/svg+xml;utf8,<svg ...><g stroke="red" ... /></g></svg>');

здесь больше информации при встраивании полного SVG-кода в ваш CSS-файл. Он также упомянул совместимость браузера, которая немного слишком мала, чтобы это было жизнеспособным вариантом.


еще один подход - использовать маску. Затем вы изменяете цвет фона маскируемого элемента. Это имеет тот же эффект, что и изменение атрибута fill svg.

HTML-код:

<glyph class="star"/>
<glyph class="heart" />
<glyph class="heart" style="background-color: green"/>
<glyph class="heart" style="background-color: blue"/>

CSS:

glyph {
    display: inline-block;
    width:  24px;
    height: 24px;
}

glyph.star {
  -webkit-mask: url(star.svg) no-repeat 100% 100%;
  mask: url(star.svg) no-repeat 100% 100%;
  -webkit-mask-size: cover;
  mask-size: cover;
  background-color: yellow;
}

glyph.heart {
  -webkit-mask: url(heart.svg) no-repeat 100% 100%;
  mask: url(heart.svg) no-repeat 100% 100%;
  -webkit-mask-size: cover;
  mask-size: cover;
  background-color: red;
}

вы найдете полный учебник здесь:http://codepen.io/noahblon/blog/coloring-svgs-in-css-background-images (не мой собственный). Он предлагает различные подходы (не ограничиваясь маской).


вы можете использовать маски CSS, со свойством' mask ' вы создаете маску, которая применяется к элементу.

.icon {
    background-color: red;
    -webkit-mask-image: url(icon.svg);
    mask-image: url(icon.svg);
}

больше видеть эту большую статью: https://codepen.io/noahblon/post/coloring-svgs-in-css-background-images


это возможно с Sass! Единственное, что вам нужно сделать, это url-кодировать ваш svg-код. И это возможно с вспомогательной функцией в Sass. Я сделал кодовое имя для этого. Посмотрите на это:

http://codepen.io/philippkuehn/pen/zGEjxB

// choose a color

$icon-color: #F84830;


// functions to urlencode the svg string

@function str-replace($string, $search, $replace: '') {
  $index: str-index($string, $search);
  @if $index {
    @return str-slice($string, 1, $index - 1) + $replace + str-replace(str-slice($string, $index + str-length($search)), $search, $replace);
  }
  @return $string;
}

@function url-encode($string) {
  $map: (
    "%": "%25",
    "<": "%3C",
    ">": "%3E",
    " ": "%20",
    "!": "%21",
    "*": "%2A",
    "'": "%27",
    '"': "%22",
    "(": "%28",
    ")": "%29",
    ";": "%3B",
    ":": "%3A",
    "@": "%40",
    "&": "%26",
    "=": "%3D",
    "+": "%2B",
    "$": "%24",
    ",": "%2C",
    "/": "%2F",
    "?": "%3F",
    "#": "%23",
    "[": "%5B",
    "]": "%5D"
  );
  $new: $string;
  @each $search, $replace in $map {
    $new: str-replace($new, $search, $replace);
  }
  @return $new;
}

@function inline-svg($string) {
  @return url('data:image/svg+xml;utf8,#{url-encode($string)}');
}


// icon styles
// note the fill="' + $icon-color + '"

.icon {
  display: inline-block;
  width: 50px;
  height: 50px;
  background: inline-svg('<svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
   viewBox="0 0 30 30" enable-background="new 0 0 30 30" xml:space="preserve">
<path fill="' + $icon-color + '" d="M18.7,10.1c-0.6,0.7-1,1.6-0.9,2.6c0,0.7-0.6,0.8-0.9,0.3c-1.1-2.1-0.4-5.1,0.7-7.2c0.2-0.4,0-0.8-0.5-0.7
  c-5.8,0.8-9,6.4-6.4,12c0.1,0.3-0.2,0.6-0.5,0.5c-0.6-0.3-1.1-0.7-1.6-1.3c-0.2-0.3-0.4-0.5-0.6-0.8c-0.2-0.4-0.7-0.3-0.8,0.3
  c-0.5,2.5,0.3,5.3,2.1,7.1c4.4,4.5,13.9,1.7,13.4-5.1c-0.2-2.9-3.2-4.2-3.3-7.1C19.6,10,19.1,9.6,18.7,10.1z"/>
</svg>');
}

загрузите svg в виде текста.

измените текст svg с помощью javascript, чтобы изменить цвет краски/штриха / заливки[s].

затем вставьте измененную строку svg в css, как описано здесь.


теперь вы можете достичь этого на стороне клиента следующим образом:

var green = '3CB54A';
var red = 'ED1F24';
var svg = '<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"  width="320px" height="100px" viewBox="0 0 320 100" enable-background="new 0 0 320 100" xml:space="preserve"> <polygon class="mystar" fill="#'+green+'" points="134.973,14.204 143.295,31.066 161.903,33.77 148.438,46.896 151.617,65.43 134.973,56.679 118.329,65.43 121.507,46.896 108.042,33.77 126.65,31.066 "/><circle class="mycircle" fill="#'+red+'" cx="202.028" cy="58.342" r="12.26"/></svg>';      
var encoded = window.btoa(svg);
document.body.style.background = "url(data:image/svg+xml;base64,"+encoded+")";

скрипку здесь!


вы можете сохранить SVG в переменной. Затем манипулируйте строкой SVG в зависимости от ваших потребностей (т. е. установите ширину, высоту, цвет и т. д.). Затем используйте результат для установки фона, например

$circle-icon-svg: '<svg xmlns="http://www.w3.org/2000/svg"><circle cx="10" cy="10" r="10" /></svg>';

$icon-color: #f00;
$icon-color-hover: #00f;

@function str-replace($string, $search, $replace: '') {
    $index: str-index($string, $search);

    @if $index {
        @return str-slice($string, 1, $index - 1) + $replace + str-replace(str-slice($string, $index + str-length($search)), $search, $replace);
    }

    @return $string;
}

@function svg-fill ($svg, $color) {
  @return str-replace($svg, '<svg', '<svg fill="#{$color}"');
}

@function svg-size ($svg, $width, $height) {
  $svg: str-replace($svg, '<svg', '<svg width="#{$width}"');
  $svg: str-replace($svg, '<svg', '<svg height="#{$height}"');

  @return $svg;
}

.icon {
  $icon-svg: svg-size($circle-icon-svg, 20, 20);

  width: 20px; height: 20px; background: url('data:image/svg+xml;utf8,#{svg-fill($icon-svg, $icon-color)}');

  &:hover {
    background: url('data:image/svg+xml;utf8,#{svg-fill($icon-svg, $icon-color-hover)}');
  }
}

Я тоже сделал демо,http://sassmeister.com/gist/4cf0265c5d0143a9e734.

этот код делает несколько предположений о SVG, например, что <svg /> элемент не имеет существующего цвета заливки и не имеет свойств width или height. С входные данные жестко закодированы в документе SCSS, довольно легко обеспечить соблюдение этих ограничений.

Не беспокойтесь о дублировании кода. С помощью gzip сжатие делает разницу незначительной.


Вы можете создать свою собственную функцию SCSS для этого. Добавление следующего в конфигурацию.файл rb.

require 'sass'
require 'cgi'

module Sass::Script::Functions

  def inline_svg_image(path, fill)
    real_path = File.join(Compass.configuration.images_path, path.value)
    svg = data(real_path)
    svg.gsub! '{color}', fill.value
    encoded_svg = CGI::escape(svg).gsub('+', '%20')
    data_url = "url('data:image/svg+xml;charset=utf-8," + encoded_svg + "')"
    Sass::Script::String.new(data_url)
  end

private

  def data(real_path)
    if File.readable?(real_path)
      File.open(real_path, "rb") {|io| io.read}
    else
      raise Compass::Error, "File not found or cannot be read: #{real_path}"
    end
  end

end

тогда вы можете использовать его в своем CSS:

.icon {
  background-image: inline-svg-image('icons/icon.svg', '#555');
}

вам нужно будет отредактировать ваши SVG-файлы и заменить любые атрибуты заливки в разметке на fill= " {color}"

путь значка всегда относительно вашего параметра images_dir в той же конфигурации.файл rb.

подобно некоторым другим решениям, но это довольно чисто и держит ваш Файлы SCSS аккуратные!


 .icon { 
  width: 48px;
  height: 48px;
  display: inline-block;
  background: url(https://s3-us-west-2.amazonaws.com/s.cdpn.io/18515/heart.svg) no-repeat 50% 50%; 
  background-size: cover;
}

.icon-orange { 
  -webkit-filter: hue-rotate(40deg) saturate(0.5) brightness(390%) saturate(4); 
  filter: hue-rotate(40deg) saturate(0.5) brightness(390%) saturate(4); 
}

.icon-yellow {
  -webkit-filter: hue-rotate(70deg) saturate(100);
  filter: hue-rotate(70deg) saturate(100);
}

codeben статья и демо


поздно на шоу здесь, но я смог добавить цвет заливки в многоугольник SVG, если вы можете напрямую редактировать код SVG, например, следующий svg отображает красный, а не черный по умолчанию. Я не тестировал вне Chrome, хотя:

<svg version="1.1" id="Layer_1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" x="0px" y="0px"
 width="500px" height="500px" viewBox="0 0 500 500" enable-background="new 0 0 500 500" xml:space="preserve">
    <polygon 


        fill="red"


        fill-rule="evenodd" clip-rule="evenodd" points="452.5,233.85 452.5,264.55 110.15,264.2 250.05,390.3 229.3,413.35 
47.5,250.7 229.3,86.7 250.05,109.75 112.5,233.5 "/>
</svg>

в некоторых (очень конкретных) ситуациях это может быть достигнуто с помощью фильтр. Например, вы можете изменить синее изображение SVG на фиолетовое, повернув оттенок на 45 градусов с помощью filter: hue-rotate(45deg);. Поддержка браузера минимальна, но это все еще интересная техника.

демо


Это мой любимый метод, но поддержка вашего браузера должна быть очень прогрессивной. С помощью свойства mask создается маска, которая применяется к элементу. Везде, где маска непрозрачна или тверда, сквозь нее просвечивает основное изображение. Там, где он прозрачен, основное изображение замаскировано или скрыто. Синтаксис для CSS mask-image аналогичен background-image.посмотрите на codepenmask


много IFs, но если ваш предварительно base64 закодирован SVG начинается:

<svg fill="#000000

затем начнется строка в кодировке base64:

PHN2ZyBmaWxsPSIjMDAwMDAw

если предварительно закодированная строка начинается:

<svg fill="#bfa76e

тогда это кодирует:

PHN2ZyBmaWxsPSIjYmZhNzZl

как закодированные строки начинаются одинаково:

PHN2ZyBmaWxsPSIj

причуда кодирования base64 заключается в том, что каждые 3 входных символа становятся 4 выходными символами. При запуске SVG, как это, то 6-символьный шестнадцатеричный цвет заливки начинается точно на "границе" блока кодирования. Поэтому вы можете легко сделать кросс-браузер JS заменить:

output = input.replace(/MDAwMDAw/, "YmZhNzZl");

но ответ tnt-rox выше-это способ двигаться вперед.