如何检查svg路径是否具有与数组中的值匹配的类,如果是,则添加新类

时间:2018-06-06 10:46:26

标签: javascript jquery svg path leaflet

我有一个数组和一些svg path元素(我正在使用leaflet map)。我需要检查路径的类是否与我的数组中的某个值匹配,如果是,则向其中添加一个类fadeIn

var foundNations = ["usa", "France", "Italy"];
document.querySelectorAll('path').forEach(path => {
  if (foundNations.includes(path.className)) {
   path.className.add('fadeIn');
    console.log(path.className);
  }
});
(function($) {
    var map = L.map('map').setView([45.4655171, 12.7700794], 2);
    map.fitWorld().zoomIn();
    L.tileLayer('https://api.tiles.mapbox.com/v4/{id}/{z}/{x}/{y}.png?access_token=pk.eyJ1IjoibWFwYm94IiwiYSI6ImNpejY4NXVycTA2emYycXBndHRqcmZ3N3gifQ.rJcFIG214AriISLbB6B5aw', {
      attribution: 'Map data &copy; <a href="http://openstreetmap.org">OpenStreetMap</a> contributors, ' + '<a href="http://creativecommons.org/licenses/by-sa/2.0/">CC-BY-SA</a>, ' + 'Imagery © <a href="http://mapbox.com">Mapbox</a>',
      id: 'mapbox.light'
    }).addTo(map);

    var mapHalfHeight = map.getSize().y / 2,
    container = map.zoomControl.getContainer(),
    containerHalfHeight = parseInt(container.offsetHeight / 2),
    containerTop = mapHalfHeight - containerHalfHeight + 'px';

    container.style.position = 'absolute';
    container.style.top = containerTop;
    map.scrollWheelZoom.disable();
    var southWest = L.latLng(-89.98155760646617, -180),
    northEast = L.latLng(89.99346179538875, 180);
    var bounds = L.latLngBounds(southWest, northEast);
    map.setMaxBounds(bounds);
    map.on('drag', function() {
      map.panInsideBounds(bounds, { animate: false });
    });

  // get color depending on population density value
    function getColor(d) {
      return d > 1000 ? '#800026' :
        d > 500  ? '#BD0026' :
        d > 200  ? '#E31A1C' :
        d > 100  ? '#FC4E2A' :
        d > 50   ? '#FD8D3C' :
        d > 20   ? '#FEB24C' :
        d > 10   ? '#FED976' :
              '#FFEDA0';
    }

    function style(feature) {
      return {
        weight: 1,
        opacity: 1,
        color: '#ffffff',
        dashArray: '',
        fillOpacity: 0,
        fillColor : '#FF0080',
        className: feature.properties.name
      };
    }

    var geojson;

    function selectNation(e) {

    }


   function onEachFeature(feature, layer) {
      layer.on({
        click: selectNation
      });
    }

    geojson = L.geoJson(statesData, {
      style: style,
      onEachFeature: onEachFeature
    }).addTo(map);
})( jQuery );
#map {
  width: 100vw;
  height: 100vh;
}

.fadeIn {
  fill: red;
}
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<script src="https://unpkg.com/leaflet@1.3.1/dist/leaflet.js"></script>
<link href="https://unpkg.com/leaflet@1.3.1/dist/leaflet.css" rel="stylesheet"/>
<script src="https://www.jikupin.com/world.json"></script>

<div id='map'></div>

1 个答案:

答案 0 :(得分:2)

你的错误是微妙的。您的开头是Array String

var nationList = ["usa", "france", "italy"];

然后你正在调用String.prototype.includes,将path.className作为参数传递。

if (foundNations.includes(path.className)) { path.classList.add('fadeIn')

在那里,您隐含假设 path.classNameString。但是,令人惊讶的是,它不是String,而是SVGAnimatedString

console.log(path.className)
> [object SVGAnimatedString] {
   animVal: "germany",
   baseVal: "germany"
  }

是的,在某些边缘情况下,可以在动画期间修改SVG元素的类名。

您可能想要做的是使用baseVal property of the SVGAnimatedStrings

console.log(typeof path.className.baseVal)
> "string"

现在一切都应该与你期望的方式更紧密地合作:

if (foundNations.includes(path.className.baseVal)) {
  path.classList.add('fadeIn')
}
console.log(path.className.baseVal);
> "spain fadeIn"

由于另一个假设,你有第二个问题。您假设 path.className只包含一个类名,但是according to the documentation,强调我的:

  

cName是一个字符串变量,表示当前元素的类或空格分隔的类

事实上,如果您使用浏览器中提供的开发人员工具来检查SVG元素,您会看到类似的内容......

  

<path class="Italy leaflet-interactive" stroke="#ffffff" ....></path>

因此,在这种情况下,您假设className.baseVal将成为字符串"Italy",但实际上,它取值为"Italy leaflet-interactive"

这里的方法是使用Element.classList来遍历类名 s ,看看中的是否与给定的相匹配 基。

此外,我认为这是XY problem的一个实例。我认为你不想问

  

如何检查SVG路径是否具有匹配foo

的类

而是

  

当要素与foo匹配时,如何在Leaflet中对SVG多边形进行符号化?

因为我认为将支票移入style回调函数更为优雅,例如:

geojson = L.geoJson(statesData, {
  style: function(feature){ 
    var polygonClassName = feature.properties.name;
    if (nationList.contains(feature.properties.name)) {
        polygonClassName += ' fadeIn';
    }
    return {
      weight: 1,
      opacity: 1,
      color: '#ffffff',
      dashArray: '',
      fillOpacity: 0,
      fillColor : '#FF0080',
      className: polygonClassName
    };
  },
  onEachFeature: onEachFeature
}).addTo(map);

Leaflet提供了方便的功能,如L.Path.setStyle,它隐藏了直接处理选择器和SVG类的复杂性,只要你保持对L.Polygon实例的引用(在这种情况下,你可以在onEachFeature回调中完成。