123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441 |
- window.onload = function () {
- var paper = Raphael("holder");
-
-
- var text = "Betty Botter bought some butter but, she said, the butter's bitter. If I put it in my batter, it will make my batter bitter. But a bit of better butter will make my batter better. So, she bought a bit of butter, better than her bitter butter, and she put it in her batter, and the batter was not bitter. It was better Betty Botter bought a bit better butter.";
- var font = {font: "11px Arial", "font-style":"italic", opacity: 1, "fill": LABEL_COLOR, stroke: LABEL_COLOR, "stroke-width":.3};
- var font = {font: "11px Arial", opacity: 1, "fill": LABEL_COLOR};
- var boxWidth = 100
-
- var AttributedStringIterator = function(text){
-
- text = text.replace(/(\s)+/, " ");
- this.text = this.rtrim(text);
-
- this.beginIndex = 0;
- this.endIndex = this.text.length;
- this.currentIndex = this.beginIndex;
-
-
- var i = 0;
- var string = this.text;
- var fullPos = 0;
-
-
- this.startWordOffsets = [];
- this.startWordOffsets.push(fullPos);
-
-
- while (i<1000) {
- var pos = string.search(/[ \t\n\f-\.\,]/);
- if (pos == -1)
- break;
-
-
- fullPos += pos;
- string = string.substr(pos);
-
-
-
- var pos = string.search(/[^ \t\n\f-\.\,]/);
- if (pos == -1)
- break;
-
-
- fullPos += pos;
- string = string.substr(pos);
-
-
- this.startWordOffsets.push(fullPos);
-
- i++;
- }
-
-
- };
- AttributedStringIterator.prototype = {
- getEndIndex: function(pos){
- if (typeof(pos) == "undefined")
- return this.endIndex;
-
- var string = this.text.substr(pos, this.endIndex - pos);
-
- var posEndOfLine = string.search(/[\n]/);
- if (posEndOfLine == -1)
- return this.endIndex;
- else
- return pos + posEndOfLine;
- },
- getBeginIndex: function(){
- return this.beginIndex;
- },
- isWhitespace: function(pos){
- var str = this.text[pos];
- var whitespaceChars = " \t\n\f";
-
- return (whitespaceChars.indexOf(str) != -1);
- },
- isNewLine: function(pos){
- var str = this.text[pos];
- var whitespaceChars = "\n";
-
- return (whitespaceChars.indexOf(str) != -1);
- },
- preceding: function(pos){
-
- for(var i in this.startWordOffsets) {
- var startWordOffset = this.startWordOffsets[i];
- if (pos < startWordOffset && i>0) {
-
-
- return this.startWordOffsets[i-1];
- }
- }
-
-
- return this.startWordOffsets[i];
- },
- following: function(pos){
-
- for(var i in this.startWordOffsets) {
- var startWordOffset = this.startWordOffsets[i];
- if (pos < startWordOffset && i>0) {
-
-
- return this.startWordOffsets[i];
- }
- }
-
-
- return this.startWordOffsets[i];
- },
- ltrim: function(str){
- var patt2=/^\s+/g;
- return str.replace(patt2, "");
- },
- rtrim: function(str){
- var patt2=/\s+$/g;
- return str.replace(patt2, "");
- },
- getLayout: function(start, limit){
- return this.text.substr(start, limit - start);
- },
- getCharAtPos: function(pos) {
- return this.text[pos];
- }
- };
-
-
-
-
- var LineBreakMeasurer = function(paper, text, fontAttrs){
- this.paper = paper;
- this.text = new AttributedStringIterator(text);
- this.fontAttrs = fontAttrs;
-
- if (this.text.getEndIndex() - this.text.getBeginIndex() < 1) {
- throw {message: "Text must contain at least one character.", code: "IllegalArgumentException"};
- }
-
-
- this.limit = this.text.getEndIndex();
- this.pos = this.start = this.text.getBeginIndex();
-
- this.rafaelTextObject = this.paper.text(100, 100, this.text.text).attr(fontAttrs).attr("text-anchor", "start");
- this.svgTextObject = this.rafaelTextObject[0];
- };
- LineBreakMeasurer.prototype = {
- nextOffset: function(wrappingWidth, offsetLimit, requireNextWord) {
-
- var nextOffset = this.pos;
- if (this.pos < this.limit) {
- if (offsetLimit <= this.pos) {
- throw {message: "offsetLimit must be after current position", code: "IllegalArgumentException"};
- }
-
- var charAtMaxAdvance = this.getLineBreakIndex(this.pos, wrappingWidth);
-
-
-
- if (charAtMaxAdvance == this.limit) {
- nextOffset = this.limit;
-
- } else if (this.text.isNewLine(charAtMaxAdvance)) {
- console.log("isNewLine");
- nextOffset = charAtMaxAdvance+1;
- } else if (this.text.isWhitespace(charAtMaxAdvance)) {
-
-
- nextOffset = this.text.following(charAtMaxAdvance);
- } else {
-
-
- nextOffset = this.text.preceding(charAtMaxAdvance);
-
- if (nextOffset <= this.pos) {
- nextOffset = Math.max(this.pos+1, charAtMaxAdvance);
- }
- }
- }
- if (nextOffset > offsetLimit) {
- nextOffset = offsetLimit;
- }
-
-
- return nextOffset;
- },
- nextLayout: function(wrappingWidth) {
-
- if (this.pos < this.limit) {
- var requireNextWord = false;
- var layoutLimit = this.nextOffset(wrappingWidth, this.limit, requireNextWord);
-
- if (layoutLimit == this.pos) {
-
- return null;
- }
- var result = this.text.getLayout(this.pos, layoutLimit);
-
-
-
-
-
-
-
-
- this.pos = layoutLimit;
-
-
- return result;
- } else {
-
- return null;
- }
- },
- getLineBreakIndex: function(pos, wrappingWidth) {
-
-
-
- var bb = this.rafaelTextObject.getBBox();
-
- var charNum = -1;
- try {
- var svgPoint = this.svgTextObject.getStartPositionOfChar(pos);
-
- svgPoint.x = svgPoint.x + wrappingWidth;
-
-
-
-
-
- charNum = this.svgTextObject.getCharNumAtPosition(svgPoint);
- } catch (e){
- console.warn("getStartPositionOfChar error, pos:" + pos);
-
- }
-
- if (charNum == -1) {
-
- return this.text.getEndIndex(pos);
- } else {
-
- var newLineIndex = this.text.getEndIndex(pos);
- if (newLineIndex < charNum ) {
- console.log("newLineIndex <= charNum, newLineIndex:"+newLineIndex+", charNum:"+charNum, "\"" + this.text.text.substr(newLineIndex+1).replace(/\n/g, "↵") + "\"");
-
-
- return newLineIndex;
- }
-
-
- var charAtMaxAdvance = this.text.getCharAtPos(charNum);
-
-
- return charNum;
- }
- },
- getPosition: function() {
- return this.pos;
- }
- };
-
-
-
-
- function drawMultilineText(text, x, y, boxWidth, boxHeight, options) {
- var TEXT_PADDING = 3;
- var width = boxWidth - (2 * TEXT_PADDING);
- if (boxHeight)
- var height = boxHeight - (2 * TEXT_PADDING);
-
- var layouts = [];
-
- var measurer = new LineBreakMeasurer(paper, text, font);
- var lineHeight = measurer.rafaelTextObject.getBBox().height;
- console.log("text: ", text.replace(/\n/g, "↵"));
-
- if (height) {
- var availableLinesCount = parseInt(height/lineHeight);
- console.log("availableLinesCount: " + availableLinesCount);
- }
-
- var i = 1;
- while (measurer.getPosition() < measurer.text.getEndIndex()) {
- var layout = measurer.nextLayout(width);
-
-
- if (layout != null) {
- if (!availableLinesCount || i < availableLinesCount) {
- layouts.push(layout);
- } else {
- layouts.push(fitTextToWidth(layout + "...", boxWidth));
- break;
- }
- }
- i++;
- };
- console.log(layouts);
-
- measurer.rafaelTextObject.attr({"text": layouts.join("\n")});
-
-
- if (options)
- measurer.rafaelTextObject.attr({"text-anchor": options["text-anchor"]});
-
- var bb = measurer.rafaelTextObject.getBBox();
-
- if (options["vertical-align"] == "top")
- measurer.rafaelTextObject.attr({"y": y + bb.height/2 + TEXT_PADDING});
- else
- measurer.rafaelTextObject.attr({"y": y + height/2});
-
-
- if (measurer.rafaelTextObject.attr("text-anchor") == "middle" )
- measurer.rafaelTextObject.attr("x", x + boxWidth/2 + TEXT_PADDING/2);
- else if (measurer.rafaelTextObject.attr("text-anchor") == "end" )
- measurer.rafaelTextObject.attr("x", x + boxWidth + TEXT_PADDING/2);
- else
- measurer.rafaelTextObject.attr("x", x + boxWidth/2 - bb.width/2 + TEXT_PADDING/2);
-
- var boxStyle = {stroke: Color.LightSteelBlue2, "stroke-width": 1.0, "stroke-dasharray": "- "};
-
-
-
- var textAreaCX = x + boxWidth/2;
- var textAreaCY = y + height/2;
- var dotLeftTop = paper.ellipse(x, y, 3, 3).attr({"stroke-width": 0, fill: Color.LightSteelBlue, stroke: "none"});
- var dotCenter = paper.ellipse(textAreaCX, textAreaCY, 3, 3).attr({fill: Color.LightSteelBlue2, stroke: "none"});
-
- var boxStyle = {stroke: Color.LightSteelBlue2, "stroke-width": 1.0, "stroke-dasharray": "- "};
- var rect = paper.rect(x+.5, y + .5, boxWidth, boxHeight).attr(boxStyle);
- }
-
-
-
-
-
-
-
- var fitTextToWidth = function(original, width) {
- var text = original;
-
- var attr = {font: "11px Arial", opacity: 0};
-
-
- var dots = paper.text(0, 0, "...").attr(attr).hide();
- var dotsBB = dots.getBBox();
-
- var maxWidth = width - dotsBB.width;
-
- var textElement = paper.text(0, 0, text).attr(attr).hide();
- var bb = textElement.getBBox();
-
-
- while (bb.width > maxWidth && text.length > 0) {
- text = text.substring(0, text.length - 1);
- textElement.attr({"text": text});
- bb = textElement.getBBox();
- }
-
- textElement.remove();
-
- if (text != original) {
- text = text + "...";
- }
- return text;
- }
-
-
- var x=100, y=90, height=20;
- var options = {"text-anchor": "middle", "boxHeight": 150, "vertical-align": "top"};
- var options = {"boxHeight": 150, "vertical-align": "top"};
- drawMultilineText(text, x, y, 150, 100, options);
- };
|