だからここに私が持っているものがあります:
<path class="..." onmousemove="show_tooltip(event,'very long text
\\\n I would like to linebreak')" onmouseout="hide_tooltip()" d="..."/>
<rect class="tooltip_bg" id="tooltip_bg" ... />
<text class="tooltip" id="tooltip" ...>Tooltip</text>
<script>
<![CDATA[
function show_tooltip(e,text) {
var tt = document.getElementById('tooltip');
var bg = document.getElementById('tooltip_bg');
// set position ...
tt.textContent=text;
bg.setAttribute('width',tt.getBBox().width+10);
bg.setAttribute('height',tt.getBBox().height+6);
// set visibility ...
}
...
これで、alert()を使用していても、非常に長いツールチップテキストに改行がありません。テキストには実際には2行あることがわかります。 (ただし、「\」が含まれていますが、どうすれば削除できますか?)
CDATAをどこでも動作させることができません。
これは、SVG 1.1がサポートするものではありません。 SVG 1.2にはtextArea
要素があり、Wordの自動折り返しがありますが、すべてのブラウザーに実装されているわけではありません。 SVG 2 textArea
の実装は計画していません ですが、 自動折り返しテキスト があります。
ただし、改行が発生する場所がすでにわかっている場合は、テキストを複数の<tspan>
に分割し、それぞれx="0"
およびdy="1.4em"
を使用して、実際のテキスト行をシミュレートできます。例えば:
<g transform="translate(123 456)"><!-- replace with your target upper left corner coordinates -->
<text x="0" y="0">
<tspan x="0" dy="1.2em">very long text</tspan>
<tspan x="0" dy="1.2em">I would like to linebreak</tspan>
</text>
</g>
もちろん、JavaScriptからそれをしたいので、各要素を手動で作成してDOMに挿入する必要があります。
あなたはすでにそれを解決できていると思いますが、誰かが同様の解決策を探しているなら、これは私のために働いた:
g.append('svg:text')
.attr('x', 0)
.attr('y', 30)
.attr('class', 'id')
.append('svg:tspan')
.attr('x', 0)
.attr('dy', 5)
.text(function(d) { return d.name; })
.append('svg:tspan')
.attr('x', 0)
.attr('dy', 20)
.text(function(d) { return d.sname; })
.append('svg:tspan')
.attr('x', 0)
.attr('dy', 20)
.text(function(d) { return d.idcode; })
改行で区切られた3行があります。
Tspanソリューションでは、改行を配置する場所が事前にわからないとしましょう。このNice関数を使用できます。これは http://bl.ocksです。 org/mbostock/7555321
これは、ピクセル単位で指定された幅の長いテキストsvgに対して自動的に改行します。
function wrap(text, width) {
text.each(function() {
var text = d3.select(this),
words = text.text().split(/\s+/).reverse(),
Word,
line = [],
lineNumber = 0,
lineHeight = 1.1, // ems
y = text.attr("y"),
dy = parseFloat(text.attr("dy")),
tspan = text.text(null).append("tspan").attr("x", 0).attr("y", y).attr("dy", dy + "em");
while (Word = words.pop()) {
line.Push(Word);
tspan.text(line.join(" "));
if (tspan.node().getComputedTextLength() > width) {
line.pop();
tspan.text(line.join(" "));
line = [Word];
tspan = text.append("tspan").attr("x", 0).attr("y", y).attr("dy", ++lineNumber * lineHeight + dy + "em").text(Word);
}
}
});
}
私はこれがあなたが望むことをすると思う:
function ShowTooltip(evt, mouseovertext){
// Make tooltip text
var tooltip_text = tt.childNodes.item(1);
var words = mouseovertext.split("\\\n");
var max_length = 0;
for (var i=0; i<3; i++){
tooltip_text.childNodes.item(i).firstChild.data = i<words.length ? words[i] : " ";
length = tooltip_text.childNodes.item(i).getComputedTextLength();
if (length > max_length) {max_length = length;}
}
var x = evt.clientX + 14 + max_length/2;
var y = evt.clientY + 29;
tt.setAttributeNS(null,"transform", "translate(" + x + " " + y + ")")
// Make tooltip background
bg.setAttributeNS(null,"width", max_length+15);
bg.setAttributeNS(null,"height", words.length*15+6);
bg.setAttributeNS(null,"x",evt.clientX+8);
bg.setAttributeNS(null,"y",evt.clientY+14);
// Show everything
tt.setAttributeNS(null,"visibility","visible");
bg.setAttributeNS(null,"visibility","visible");
}
\\\n
でテキストを分割し、それぞれについてtspanに各フラグメントを配置します。次に、テキストの最大長と行数に基づいて、必要なボックスのサイズを計算します。また、3つのtspanを含むようにツールチップテキスト要素を変更する必要があります。
<g id="tooltip" visibility="hidden">
<text><tspan>x</tspan><tspan x="0" dy="15">x</tspan><tspan x="0" dy="15">x</tspan></text>
</g>
これは、3行を超えないことを前提としています。 3行以上が必要な場合は、tspanを追加してforループの長さを増やすことができます。
@stecoによる解決策を少し適用し、依存関係をd3
からjquery
に切り替え、パラメーターとしてテキスト要素のheight
を追加しました
function wrap(text, width, height) {
text.each(function(idx,elem) {
var text = $(elem);
text.attr("dy",height);
var words = text.text().split(/\s+/).reverse(),
Word,
line = [],
lineNumber = 0,
lineHeight = 1.1, // ems
y = text.attr("y"),
dy = parseFloat( text.attr("dy") ),
tspan = text.text(null).append("tspan").attr("x", 0).attr("y", y).attr("dy", dy + "em");
while (Word = words.pop()) {
line.Push(Word);
tspan.text(line.join(" "));
if (elem.getComputedTextLength() > width) {
line.pop();
tspan.text(line.join(" "));
line = [Word];
tspan = text.append("tspan").attr("x", 0).attr("y", y).attr("dy", ++lineNumber * lineHeight + dy + "em").text(Word);
}
}
});
}