Jak zmienić kolor elementu SVG?


349

Chcę użyć tej techniki http://css-tricks.com/svg-fallbacks/ i zmienić kolor svg, ale jak dotąd nie byłem w stanie tego zrobić. Umieszczam to w css, ale mój obraz jest zawsze czarny, bez względu na wszystko. Mój kod:

.change-my-color {
  fill: green;
}
<svg>
    <image class="change-my-color" xlink:href="https://svgur.com/i/AFM.svg" width="96" height="96" src="ppngfallback.png" />
</svg>


Nie jestem ekspertem od svg, ale czy próbowałeś zmienić wypełnienie na kolor tła?
Meg

@Megan w kolorze tła svg jest określane za pomocą właściwości „fill”, a obramowanie za pomocą „stroke” (jak w programie Illustrator). w3.org/TR/SVG/propidx.html
Barbara

4
CSS z twojego dokumentu HTML nie będzie miał zastosowania do elementów SVG w <img />
pawel

1
To jest teraz możliwe. Prosta i funkcjonalna odpowiedź tutaj: stackoverflow.com/a/53336754/467240
mtyson

Odpowiedzi:


188

W ten sposób nie można zmienić koloru obrazu. Jeśli załadujesz SVG jako obraz, nie możesz zmienić sposobu jego wyświetlania za pomocą CSS lub Javascript w przeglądarce.

Jeśli chcesz zmienić swój wizerunek SVG, trzeba załadować go za pomocą <object>, <iframe>lub za pomocą <svg>wbudowanej.

Jeśli chcesz użyć technik na stronie, potrzebujesz biblioteki Modernizr, w której możesz sprawdzić obsługę SVG i warunkowo wyświetlić lub nie wyświetlać obrazu zastępczego. Następnie możesz wstawić plik SVG i zastosować potrzebne style.

Widzieć :

Możesz wstawić plik SVG, oznaczyć obraz zastępczy nazwą klasy ( my-svg-alternate):

<svg width="96px" height="96px" viewBox="0 0 512 512" enable-background="new 0 0 512 512" xml:space="preserve">
<path id="time-3-icon" .../>
</svg>

<image class="my-svg-alternate" width="96" height="96" src="ppngfallback.png" />

A w CSS skorzystaj z no-svgklasy Modernizr (CDN: http://ajax.aspnetcdn.com/ajax/modernizr/modernizr-2.7.2.js ), aby sprawdzić obsługę SVG. Jeśli nie ma obsługi SVG, blok SVG zostanie zignorowany i obraz zostanie wyświetlony, w przeciwnym razie obraz zostanie usunięty z drzewa DOM ( display: none):

.my-svg-alternate {
  display: none;
}
.no-svg .my-svg-alternate {
  display: block;
  width: 100px;
  height: 100px;
  background-image: url(image.png);
}

Następnie możesz zmienić kolor wstawionego elementu:

#time-3-icon {
   fill: green;
}

5
Nie można stylizować osadzonych objectplików SVG z dokumentu hostingowego.
Javier Rey,

1
@JavierRey możesz wstawić stylizację do zawartości znacznika obiektu za pomocą javascript. Ale masz rację, że nie ma zastosowania, jeśli po prostu dodasz go do arkusza stylów dokumentu hostingowego.
Robert Longson

2
Korzystam z rozwiązania z @ manish-menaria i działa idealnie.
Ryan Ellis

1
Zaakceptowana odpowiedź powinna zostać zmieniona na: stackoverflow.com/a/53336754/467240
mtyson

375

Odpowiedź na 2020 r

Filtr CSS działa we wszystkich obecnych przeglądarkach

Aby zmienić dowolny kolor SVG

  1. Dodaj obraz SVG za pomocą <img>znacznika.
<img src="dotted-arrow.svg" class="filter-green"/>
  1. Aby przefiltrować według określonego koloru, użyj następującego Codepen (kliknij tutaj, aby otworzyć codepen), aby przekonwertować szesnastkowy kod koloru na filtr CSS:

Na przykład dane wyjściowe dla #00EE00to

filter: invert(42%) sepia(93%) saturate(1352%) hue-rotate(87deg) brightness(119%) contrast(119%);
  1. Dodaj CSS filterdo tej klasy.
    .filter-green{
        filter: invert(48%) sepia(79%) saturate(2476%) hue-rotate(86deg) brightness(118%) contrast(119%);
    }

5
Jest to związane ze zwykłym zastrzeżeniem dotyczącym braku obsługi starszych wersji przeglądarki: developer.mozilla.org/en-US/docs/Web/CSS/…
Kevin Wang

17
Jak zauważono w CodePen, jeśli twój plik SVG nie jest czarny (mój był szary), dodanie brightness(0) saturate(100%)na początku listy filtrów najpierw zmieni kolor na 100% czarny, co pozwoli innym filtrom zmienić go na właściwy kolor.
jdunning

2
Również wiele fascynujących doświadczeń dotyczących rozwiązania w tym pytaniu StackOverflow, które poinformowało CodePen.
jdunning

3
Mój ziom. Wsparcie wydaje się akceptowalne caniuse.com/#feat=css-filters .
Sam Doidge,

działa tak dobrze, użyłem tego niepowiązanego codepen, aby wprowadzić mój hex do filtra: codepen.io/sosuke/pen/Pjoqqp
WEBjuju

220

Aby zmienić kolor dowolnego pliku SVG, możesz bezpośrednio zmienić kod svg, otwierając plik svg w dowolnym edytorze tekstu . Kod może wyglądać jak poniższy kod

<?xml version="1.0" encoding="utf-8"?>
    <!-- Generator: Adobe Illustrator 16.0.0, SVG Export Plug-In . SVG Version: 6.00 Build 0)  -->
    <!DOCTYPE svg PUBLIC "-//W3C//DTD SVG 1.1//EN" "http://www.w3.org/Graphics/SVG/1.1/DTD/svg11.dtd">
    <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">
    <g>
        <path d="M114.26,436.584L99.023,483h301.953l-15.237-46.416H114.26z M161.629,474.404h-49.592l9.594-29.225h69.223
            C181.113,454.921,171.371,464.663,161.629,474.404z"/>
    /*Some more code goes on*/
    </g>
    </svg>

Możesz zaobserwować, że istnieją pewne tagi XML, takie jak ścieżka, okrąg, wielokąt itp . Tam możesz dodać własny kolor za pomocą atrybutu stylu . Spójrz na poniższy przykład

<path style="fill:#AB7C94;" d="M114.26,436.584L99.023,483h301.953l-15.237-46.416H114.26z M161.629,474.404h-49.592l9.594-29.225h69.223
                C181.113,454.921,171.371,464.663,161.629,474.404z"/>

Dodaj atrybut stylu do wszystkich tagów, aby uzyskać SVG wymaganego koloru


48
Dlaczego nie po prostu przy użyciu atrybutu filltak: fill = "#AB7C94"? Nie jestem pewien, dlaczego styleatrybut jest potrzebny
bg17aw

4
Cześć Daniel, tak, to działa. Nie wiedziałem, że wypełnienia można użyć jako atrybutu. Przepraszam, że nie zauważyłem tak długo komentarza @ bg17aw
sushant047,

30

Dodano stronę testową - do pokolorowania SVG poprzez ustawienia Filtru:

NA PRZYKŁAD filter: invert(0.5) sepia(1) saturate(5) hue-rotate(175deg)

Prześlij i pokoloruj swój plik SVG - Jsfiddle

Wziął pomysł od: https://blog.union.io/code/2017/08/10/img-svg-fill/


1
Dzięki, po prostu mnie ocaliłeś. .custom-disabled > svg {filter:invert(0.2) sepia(1) saturte(0) hue-rotate(0);}wykonałem właśnie tę pracę, której potrzebuję, aby wyłączyć ikonę.
roger,

20

Tylko SVG z informacjami o ścieżce . nie możesz tego zrobić na obrazie. jako ścieżkę możesz zmienić obrys i wypełnić informacje i gotowe. jak Illustrator

więc: za pomocą CSS możesz zastąpić fillwartość ścieżki

path { fill: orange; }

ale jeśli chcesz bardziej elastycznego sposobu, ponieważ chcesz go zmienić za pomocą tekstu, gdy efekt unosi się, użyj ...

path { fill: currentcolor; }

body {
  background: #ddd;
  text-align: center;
  padding-top: 2em;
}

.parent {
  width: 320px;
  height: 50px;
  display: block;
  transition: all 0.3s;
  cursor: pointer;
  padding: 12px;
  box-sizing: border-box;
}

/***  desired colors for children  ***/
.parent{
  color: #000;
  background: #def;
}
.parent:hover{
  color: #fff;
  background: #85c1fc;
}

.parent span{
  font-size: 18px;
  margin-right: 8px;
  font-weight: bold;
  font-family: 'Helvetica';
  line-height: 26px;
  vertical-align: top;
}
.parent svg{
  max-height: 26px;
  width: auto;
  display: inline;
}

/****  magic trick  *****/
.parent svg path{
  fill: currentcolor;
}
<div class='parent'>
  <span>TEXT WITH SVG</span>
  <svg version="1.1" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="128" height="128" viewBox="0 0 32 32">
<path d="M30.148 5.588c-2.934-3.42-7.288-5.588-12.148-5.588-8.837 0-16 7.163-16 16s7.163 16 16 16c4.86 0 9.213-2.167 12.148-5.588l-10.148-10.412 10.148-10.412zM22 3.769c1.232 0 2.231 0.999 2.231 2.231s-0.999 2.231-2.231 2.231-2.231-0.999-2.231-2.231c0-1.232 0.999-2.231 2.231-2.231z"></path>
</svg>
</div>


12

najprostszym sposobem byłoby utworzenie czcionki z SVG za pomocą usługi takiej jak https://icomoon.io/app/#/select lub podobna . prześlij plik SVG, kliknij „generuj czcionkę”, dołącz pliki czcionek i css do swojej strony i po prostu używaj i stylizuj jak każdy inny tekst. Zawsze używam tego w ten sposób, ponieważ znacznie ułatwia stylizację.

EDYCJA: Jak wspomniano w artykule skomentowanym przez @ CodeMouse92 ikona czcionki zepsują czytniki ekranu (i prawdopodobnie są złe dla SEO). Raczej trzymaj się SVG.


4
To także popsuwa czytniki ekranu. Zobacz „Death to Icon Fonts” Seren Davies
CodeMouse92,

7

Maska SVG na elemencie ramki z kolorem tła spowoduje:

.icon{
  --size     : 70px;
  display    : inline-block;
  width      : var(--size);
  height     : var(--size);
  transition : .12s;
  -webkit-mask-size: cover;
  mask-size  : cover;
}

.icon-bike{
  background: black;
  -webkit-mask-image: url(https://image.flaticon.com/icons/svg/89/89139.svg);
  mask-image: url(hhttps://image.flaticon.com/icons/svg/89/89139.svg);
  animation: 1s frames infinite ease;
}

@keyframes frames {
  50% { background:red;  }
}
<i class="icon icon-bike" style="--size:150px"></i>


Uwaga - maski SVG nie są obsługiwane w przeglądarkach Internet Explorer


1
Wielkie dzięki, @vsync to po prostu najlepszy hack dla tego, czego potrzebuję.
Vixson

6

Możesz zmienić kolorystykę SVG za pomocą css, jeśli używasz niektórych sztuczek. Napisałem do tego mały skrypt.

  • przejrzyj listę elementów, które mają obraz svg
  • załaduj plik svg jako xml
  • pobierz tylko część svg
  • zmień kolor ścieżki
  • zamień src na zmodyfikowany plik svg jako obraz wbudowany
$('img.svg-changeable').each(function () {
  var $e = $(this);
  var imgURL = $e.prop('src');

  $.get(imgURL, function (data) {
    // Get the SVG tag, ignore the rest
    var $svg = $(data).find('svg');

    // change the color
    $svg.find('path').attr('fill', '#000');

    $e.prop('src', "data:image/svg+xml;base64," + window.btoa($svg.prop('outerHTML')));
  });

});

powyższy kod może nie działać poprawnie, zaimplementowałem to dla elementów z obrazem tła svg, który działa prawie podobnie do tego. Ale i tak musisz zmodyfikować ten skrypt, aby pasował do twojej sprawy. mam nadzieję, że to pomogło.


Nawiasem mówiąc: Jeśli jesteś programistą RoR, możesz dodać nową metodę dla prekompilatora sass, który również może wykonać to zadanie. Jest to o wiele lepsze, ponieważ w skompilowanym pliku css będziesz mieć zakodowany obraz base64, poprawny kolorowy obraz. JS nie jest już potrzebny! Może mógłbym podać kod, który napisałem, porozmawiać z CTO.
cydoc

2
+1 za dostarczenie rozwiązania, a nie stwierdzenie, że nie można tego zrobić. Ta odpowiedź jest również istotna: stackoverflow.com/questions/11978995/…
claytronicon

5

Skieruj ścieżkę w pliku svg:

<svg>
   <path>....
</svg>

Możesz robić inline, na przykład:

<path fill="#ccc">

Lub

svg{
   path{
        fill: #ccc


2

Jeśli chcesz to zrobić dla wbudowanego pliku svg, który jest na przykład obrazem tła w twoim css:

background: url("data:image/svg+xml;charset=utf8,%3Csvg xmlns='http://www.w3.org/2000/svg' fill='rgba(31,159,215,1)' viewBox='...'/%3E%3C/svg%3E");

oczywiście zamień ... na wbudowany kod obrazu


2

Na przykład w kodzie HTML:

<body>
  <svg viewBox="" width="" height="">
    <path id="struct1" fill="#xxxxxx" d="M203.3,71.6c-.........."></path>
  </svg>
</body>

Użyj jQuery:

$("#struct1").css("fill","<desired colour>");

Działa to tylko wtedy, gdy do pliku HTML dołączony jest plik SVG. Zredagowałem twoją odpowiedź, aby to wyjaśnić.
Flimm

0

W rzeczywistości istnieje bardziej elastyczne rozwiązanie tego problemu: napisanie komponentu WWW, który załatuje SVG jako tekst w czasie wykonywania. Publikowane również w gist z linkiem do JSFiddle

👍 filtr: odwrócenie (42%) sepia (93%) nasycenie (1352%) jasność-obrót (87 stopni) jasność (119%) kontrast (119%);

<html>

<head>
  <title>SVG with color</title>
</head>

<body>
  <script>
    (function () {
      const createSvg = (color = '#ff9933') => `
          <svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" version="1.1" width="76px" height="22px" viewBox="-0.5 -0.5 76 22">
            <defs/>
              <g>
                <ellipse cx="5" cy="10" rx="5" ry="5" fill="#ff9933" stroke="none" pointer-events="all"/>
                <ellipse cx="70" cy="10" rx="5" ry="5" fill="#ff9933" stroke="none" pointer-events="all"/>
                <path d="M 9.47 12.24 L 17.24 16.12 Q 25 20 30 13 L 32.5 9.5 Q 35 6 40 9 L 42.5 10.5 Q 45 12 50 6 L 52.5 3 Q 55 0 60.73 3.23 L 66.46 6.46" fill="none" stroke="#ff9933" stroke-miterlimit="10" pointer-events="stroke"/>
              </g>
          </svg>`.split('#ff9933').join(color);

      function SvgWithColor() {
        const div = Reflect.construct(HTMLElement, [], SvgWithColor);
        const color = div.hasAttribute('color') ? div.getAttribute('color') : 'cyan';
        div.innerHTML = createSvg(color);
        return div;
      }

      SvgWithColor.prototype = Object.create(HTMLElement.prototype);
      customElements.define('svg-with-color', SvgWithColor);

      document.body.innerHTML += `<svg-with-color
        color='magenta' 
      ></svg-with-color>`;

    })();

  </script>
</body>

</html>

-1

najkrótszy sposób zgodny z Bootstrap, bez JavaScript:

.cameraicon {
height: 1.6em;/* set your own icon size */
mask: url(/camera.svg); /* path to your image */
-webkit-mask: url(/camera.svg) no-repeat center;
}

i używaj go w następujący sposób:

<td class="text-center">
    <div class="bg-secondary cameraicon"/><!-- "bg-secondary" sets actual color of your icon -->
</td>

-1

Otwórz obraz za pomocą przeglądarki, kliknij obraz prawym przyciskiem myszy, kliknij źródło widoku strony, a zobaczysz tag svg obrazu. Poradzić sobie i wkleić do html, a następnie zmień wypełnienie na wybrany kolor


Odpowiedź zaakceptowany już sugeruje, inline, a następnie ustawienie koloru jako rozwiązanie.
Robert Longson

-1

Po prostu dodaj fill: „pożądanyKolor” w znaczniku svg obrazu: przykład:

<svg xmlns="http://www.w3.org/2000/svg" width="24" height="24" viewBox="0 0 24 24" fill="#bbb9c6">
<path d="M20 2H4c-1.1 0-1.99.9-1.99 2L2 22l4-4h14c1.1 0 2-.9 2-2V4c0-1.1-.9-2-2-2zm-2 12H6v-2h12v2zm0-3H6V9h12v2zm0-3H6V6h12v2z"/><path d="M0 0h24v24H0z" fill="none"/></svg>

To w zasadzie ta sama sugestia jak ta odpowiedź lub ta
Robert Longson

w tych odpowiedziach wspomniano o wypełnieniu tagu ścieżki, zadziałało dla mnie w tagu svg, stąd opublikowałem
Shivani
Korzystając z naszej strony potwierdzasz, że przeczytałeś(-aś) i rozumiesz nasze zasady używania plików cookie i zasady ochrony prywatności.
Licensed under cc by-sa 3.0 with attribution required.