如何在SVG文件中构建滑块功能?



我有一个带有彩色条和句柄的滑块的SVG图像(请参阅下面的代码片段)。我想让这个SVG图像具有交互性,以便查看器可以沿着滑块将手柄移动到11个不同的位置。一切都需要在<svg></svg>标签内;我不能使用外部HTML或脚本。SVG文件将被添加到HTML网页的<object>标记中,以保持交互性。

SVG文件中包含了11个不可见的<rect>元素,作为11个滑块位置的每个点击框,以及手柄组上的transform="translate(0 0)"标记。点击其中一个命中框应该通过更新transform参数的x值将句柄移动到相应的位置;在按住鼠标按钮的情况下,将鼠标从一边移动到另一边应该会将手柄移动到与鼠标x坐标对应的位置,即使鼠标垂直移动到hitbox的范围之外。

我知道你可以把JavaScript在SVG文件,但我不是很熟悉JS,所以我不知道如何处理这个。JS是最好的方法吗?如果是,我如何使用它来获得所需的功能?

<svg id="Slider-Image" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="240" height="84" viewBox="0 0 240 84">
<defs>
<style>
.hitbox {
cursor: pointer;
opacity: 0;
}
</style>
</defs>
<g id="Slider-BG">
<rect id="BG-Fill" width="240" height="84" fill="#999"/>
<polygon id="Bevel-Top" points="9 29 14 35 226 35 231 29 9 29" fill="#887"/>
<polygon id="Bevel-Right" points="226 35 226 63 231 69 231 29 226 35" fill="#aba"/>
<polygon id="Bevel-Left" points="9 29 9 69 14 63 14 35 9 29" fill="#baa"/>
<polygon id="Bevel-Bottom" points="9 69 231 69 226 63 14 63 9 69" fill="#cdd"/>
</g>
<g id="Slider-Bar">
<rect id="Slider-Bar-BG" x="14" y="35" width="212" height="28"/>
<rect id="Slider-Bar-Red" x="15" y="38" width="18" height="19" fill="#f22"/>
<rect id="Slider-Bar-Yellow" x="35" y="38" width="18" height="19" fill="#ec0"/>
<rect id="Slider-Bar-Green" x="53" y="38" width="171" height="19" fill="#9e1"/>
<line id="Slider-Bar-Divider-1" x1="53.5" y1="38" x2="53.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-2" x1="72.5" y1="38" x2="72.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-3" x1="91.5" y1="38" x2="91.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-4" x1="110.5" y1="38" x2="110.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-5" x1="129.5" y1="38" x2="129.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-6" x1="148.5" y1="38" x2="148.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-7" x1="167.5" y1="38" x2="167.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-8" x1="186.5" y1="38" x2="186.5" y2="57" stroke="#000" opacity="0.5"/>
<line id="Slider-Bar-Divider-9" x1="205.5" y1="38" x2="205.5" y2="57" stroke="#000" opacity="0.5"/>
</g>
<g id="Slider-Handle" transform="translate(0 0)">
<polygon id="Handle-Body" points="31 66 57 66 57 46 46 35 42 35 31 46 31 66" fill="#aaa"/>
<rect id="Handle-Center" x="42" y="35" width="4" height="31" fill="#ddd"/>
<rect id="Handle-Pointer" x="43" y="34" width="2" height="16" fill="#111"/>
</g>
<g id="Slider-Positions">
<rect id="Slider-Position-Minus-20" class="hitbox" x="15" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-0" class="hitbox" x="34" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-19" class="hitbox" x="53" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-38" class="hitbox" x="72" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-57" class="hitbox" x="91" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-76" class="hitbox" x="110" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-95" class="hitbox" x="129" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-114" class="hitbox" x="148" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-133" class="hitbox" x="167" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-152" class="hitbox" x="186" y="29" width="19" height="40"/>
<rect id="Slider-Position-Plus-171" class="hitbox" x="205" y="29" width="19" height="40"/>
</g>
</svg>

是这样的。当它是一个独立的文档时,代码应该看起来有点不同,但是你明白了。

console.log(Math.ceil((200 - 53) / 19) * 19);
<svg id="Slider-Image" xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink" width="240" height="84" viewBox="0 0 240 84">
<defs>
<style>
#Slider-Hitbox {
cursor: pointer;
opacity: 0;
}
.divider {
stroke: #000;
opacity: 0.5;
}
</style>
</defs>
<g id="Slider-BG">
<rect id="BG-Fill" width="240" height="84" fill="#999"/>
<polygon id="Bevel-Top" points="9 29 14 35 226 35 231 29 9 29" fill="#887"/>
<polygon id="Bevel-Right" points="226 35 226 63 231 69 231 29 226 35" fill="#aba"/>
<polygon id="Bevel-Left" points="9 29 9 69 14 63 14 35 9 29" fill="#baa"/>
<polygon id="Bevel-Bottom" points="9 69 231 69 226 63 14 63 9 69" fill="#cdd"/>
</g>
<g id="Slider-Bar">
<rect id="Slider-Bar-BG" x="14" y="35" width="212" height="28"/>
<rect id="Slider-Bar-Red" x="15" y="38" width="18" height="19" fill="#f22"/>
<rect id="Slider-Bar-Yellow" x="35" y="38" width="18" height="19" fill="#ec0"/>
<rect id="Slider-Bar-Green" x="53" y="38" width="171" height="19" fill="#9e1"/>
<line id="Slider-Bar-Divider-1" class="divider" x1="53.5" y1="38" x2="53.5" y2="57"/>
<line id="Slider-Bar-Divider-2" class="divider" x1="72.5" y1="38" x2="72.5" y2="57"/>
<line id="Slider-Bar-Divider-3" class="divider" x1="91.5" y1="38" x2="91.5" y2="57"/>
<line id="Slider-Bar-Divider-4" class="divider" x1="110.5" y1="38" x2="110.5" y2="57"/>
<line id="Slider-Bar-Divider-5" class="divider" x1="129.5" y1="38" x2="129.5" y2="57"/>
<line id="Slider-Bar-Divider-6" class="divider" x1="148.5" y1="38" x2="148.5" y2="57"/>
<line id="Slider-Bar-Divider-7" class="divider" x1="167.5" y1="38" x2="167.5" y2="57"/>
<line id="Slider-Bar-Divider-8" class="divider" x1="186.5" y1="38" x2="186.5" y2="57"/>
<line id="Slider-Bar-Divider-9" class="divider" x1="205.5" y1="38" x2="205.5" y2="57"/>
</g>
<g id="Slider-Handle" transform="translate(0 0)">
<polygon id="Handle-Body" points="31 66 57 66 57 46 46 35 42 35 31 46 31 66" fill="#aaa"/>
<rect id="Handle-Center" x="42" y="35" width="4" height="31" fill="#ddd"/>
<rect id="Handle-Pointer" x="43" y="34" width="2" height="16" fill="#111"/>
</g>
<rect id="Slider-Hitbox" x="13" y="34" width="214" height="30"/>
<script>//<![CDATA[
var down = false;
var translateX = 0;
const toSVGPoint = (svg, x, y) => {
let p = new DOMPoint(x, y);
return p.matrixTransform(svg.getScreenCTM().inverse());
};
document.getElementById("Slider-Hitbox").addEventListener("mousedown", e => {
down = true;
let p = toSVGPoint(document.getElementById("Slider-Image"), e.clientX, e.clientY);
let translateX = Math.ceil((p.x-53) / 19) * 19;
if(translateX < 0) translateX = -20;
if(translateX > 171) translateX = 171;
document.getElementById("Slider-Handle").setAttribute("transform", `translate(${translateX} 0)`);
});
document.getElementById("Slider-Image").addEventListener("mousemove", e => {
if(down) {
let p = toSVGPoint(document.getElementById("Slider-Image"), e.clientX, e.clientY);
let translateX = Math.ceil((p.x-53) / 19) * 19;
if(translateX < 0) translateX = -20;
if(translateX > 171) translateX = 171;
document.getElementById("Slider-Handle").setAttribute("transform", `translate(${translateX} 0)`);
}
});
document.getElementById("Slider-Image").addEventListener("mouseup", e => {
down = false;
});
//]]></script>
</svg>

这是整个SVG使用JavaScript运行在一个对象元素:

<object width="240" data=""></object>

最新更新