rangy-rails 1.3alpha.780.0 → 1.3alpha.804.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -6,7 +6,493 @@
6
6
  *
7
7
  * Copyright 2013, Tim Down
8
8
  * Licensed under the MIT license.
9
- * Version: 1.3alpha.780M
10
- * Build date: 17 May 2013
9
+ * Version: 1.3alpha.804
10
+ * Build date: 8 December 2013
11
11
  */
12
- rangy.createModule("Highlighter",["ClassApplier"],function(a,b){function f(a,b){return a.characterRange.start-b.characterRange.start}function j(a,b){this.type=a,this.converterCreator=b}function k(a,b){i[a]=new j(a,b)}function l(a){var b=i[a];if(b instanceof j)return b.create();throw new Error("Highlighter type '"+a+"' is not valid")}function m(a,b){this.start=a,this.end=b}function o(a,b,c,d,e,f){e?(this.id=e,h=Math.max(h,e+1)):this.id=h++,this.characterRange=b,this.doc=a,this.classApplier=c,this.converter=d,this.containerElementId=f||null,this.applied=!1}function p(a,b){b=b||"textContent",this.doc=a||document,this.classAppliers={},this.highlights=[],this.converter=l(b)}var c=a.dom,d=c.arrayContains,e=c.getBody,g=[].forEach?function(a,b){a.forEach(b)}:function(a,b){for(var c=0,d=a.length;c<d;++c)b(a[c])},h=1,i={};j.prototype.create=function(){var a=this.converterCreator();return a.type=this.type,a},a.registerHighlighterType=k,m.prototype={intersects:function(a){return this.start<a.end&&this.end>a.start},union:function(a){return new m(Math.min(this.start,a.start),Math.max(this.end,a.end))},intersection:function(a){return new m(Math.max(this.start,a.start),Math.min(this.end,a.end))},toString:function(){return"[CharacterRange("+this.start+", "+this.end+")]"}},m.fromCharacterRange=function(a){return new m(a.start,a.end)};var n={rangeToCharacterRange:function(a,b){var c=a.getBookmark(b);return new m(c.start,c.end)},characterRangeToRange:function(b,c,d){var e=a.createRange(b);return e.moveToBookmark({start:c.start,end:c.end,containerNode:d}),e},serializeSelection:function(a,b){var c=a.getAllRanges(),d=c.length,e=[],f=d==1&&a.isBackward();for(var g=0,h=c.length;g<h;++g)e[g]={characterRange:this.rangeToCharacterRange(c[g],b),backward:f};return e},restoreSelection:function(a,b,c){a.removeAllRanges();var d=a.win.document;for(var e=0,f=b.length,g,h,i;e<f;++e)h=b[e],i=h.characterRange,g=this.characterRangeToRange(d,h.characterRange,c),a.addRange(g,h.backward)}};k("textContent",function(){return n}),k("TextRange",function(){var b;return function(){if(!b){var c=a.modules.TextRange;if(!c)throw new Error("TextRange module is missing.");if(!c.supported)throw new Error("TextRange module is present but not supported.");b={rangeToCharacterRange:function(a,b){return m.fromCharacterRange(a.toCharacterRange(b))},characterRangeToRange:function(b,c,d){var e=a.createRange(b);return e.selectCharacters(d,c.start,c.end),e},serializeSelection:function(a,b){return a.saveCharacterRanges(b)},restoreSelection:function(a,b,c){a.restoreCharacterRanges(c,b)}}}return b}}()),o.prototype={getContainerElement:function(){return this.containerElementId?this.doc.getElementById(this.containerElementId):e(this.doc)},getRange:function(){return this.converter.characterRangeToRange(this.doc,this.characterRange,this.getContainerElement())},fromRange:function(a){this.characterRange=this.converter.rangeToCharacterRange(a,this.getContainerElement())},getText:function(){return this.getRange().toString()},containsElement:function(a){return this.getRange().containsNodeContents(a.firstChild)},unapply:function(){this.classApplier.undoToRange(this.getRange()),this.applied=!1},apply:function(){this.classApplier.applyToRange(this.getRange()),this.applied=!0},getHighlightElements:function(){return this.classApplier.getElementsWithClassIntersectingRange(this.getRange())},toString:function(){return"[Highlight(ID: "+this.id+", class: "+this.classApplier.cssClass+", character range: "+this.characterRange.start+" - "+this.characterRange.end+")]"}},p.prototype={addClassApplier:function(a){this.classAppliers[a.cssClass]=a},getHighlightForElement:function(a){var b=this.highlights;for(var c=0,d=b.length;c<d;++c)if(b[c].containsElement(a))return b[c];return null},removeHighlights:function(a){for(var b=0,c=this.highlights.length,e;b<c;++b)e=this.highlights[b],d(a,e)&&(e.unapply(),this.highlights.splice(b--,1))},removeAllHighlights:function(){this.removeHighlights(this.highlights)},getIntersectingHighlights:function(a){var b=[],c=this.highlights,e=this.converter;return g(a,function(a){g(c,function(c){a.intersectsRange(c.getRange())&&!d(b,c)&&b.push(c)})}),b},highlightCharacterRanges:function(b,c,d){var e,f,h,i=this.highlights,j=this.converter,k=this.doc,l=[],n=this.classAppliers[b];d=d||null;var p,q,r;d&&(p=this.doc.getElementById(d),p&&(q=a.createRange(this.doc),q.selectNodeContents(p),r=new m(0,q.toString().length),q.detach()));var s,t,u,v;for(e=0,f=c.length;e<f;++e){s=c[e],v=!1,r&&(s=s.intersection(r));for(h=0;h<i.length;++h)d==i[h].containerElementId&&(t=i[h].characterRange,t.intersects(s)&&(l.push(i[h]),i[h]=new o(k,t.union(s),n,j,null,d)));v||i.push(new o(k,s,n,j,null,d))}g(l,function(a){a.unapply()});var w=[];return g(i,function(a){a.applied||(a.apply(),w.push(a))}),w},highlightRanges:function(b,c,d){var f=[],h=this.converter,i=d?d.id:null,j;return d&&(j=a.createRange(d),j.selectNodeContents(d)),g(c,function(a){var b=d?j.intersection(a):a;f.push(h.rangeToCharacterRange(b,d||e(a.getDocument())))}),this.highlightCharacterRanges(f,c,i)},highlightSelection:function(b,c,d){var f=this.converter;c=c||a.getSelection();var h=this.classAppliers[b],i=this.highlights,j=c.win.document,k=d?j.getElementById(d):e(j);if(!h)throw new Error("No class applier found for class '"+b+"'");var l=f.serializeSelection(c,k),n=[];g(l,function(a){n.push(m.fromCharacterRange(a.characterRange))});var o=this.highlightCharacterRanges(b,n,d);return f.restoreSelection(c,l,k),o},unhighlightSelection:function(b){b=b||a.getSelection();var c=this.getIntersectingHighlights(b.getAllRanges());this.removeHighlights(c),b.removeAllRanges()},selectionOverlapsHighlight:function(b){return b=b||a.getSelection(),this.getIntersectingHighlights(b.getAllRanges()).length>0},serialize:function(a){var b=this.highlights;b.sort(f);var c=["type:"+this.converter.type];return g(b,function(b){var d=b.characterRange,e=[d.start,d.end,b.id,b.classApplier.cssClass,b.containerElementId];a&&a.serializeHighlightText&&e.push(b.getText()),c.push(e.join("$"))}),c.join("|")},deserialize:function(a){var b=a.split("|"),c=[],d=b[0],f,g,h,i=!1;if(!d||!(f=/^type:(\w+)$/.exec(d)))throw new Error("Serialized highlights are invalid.");g=f[1],g!=this.converter.type&&(h=l(g),i=!0),b.shift();var j,k,n,p,q;for(var r=b.length,s;r-->0;)s=b[r].split("$"),n=new m(+s[0],+s[1]),p=s[4]||null,q=p?this.doc.getElementById(p):e(this.doc),i&&(n=this.converter.rangeToCharacterRange(h.characterRangeToRange(this.doc,n,q),q)),j=this.classAppliers[s[3]],k=new o(this.doc,n,j,this.converter,parseInt(s[2]),p),k.apply(),c.push(k);this.highlights=c}},a.Highlighter=p,a.createHighlighter=function(a,b){return new p(a,b)}})
12
+ rangy.createModule("Highlighter", ["ClassApplier"], function(api, module) {
13
+ var dom = api.dom;
14
+ var contains = dom.arrayContains;
15
+ var getBody = dom.getBody;
16
+
17
+ // Puts highlights in order, last in document first.
18
+ function compareHighlights(h1, h2) {
19
+ return h1.characterRange.start - h2.characterRange.start;
20
+ }
21
+
22
+ var forEach = [].forEach ?
23
+ function(arr, func) {
24
+ arr.forEach(func);
25
+ } :
26
+ function(arr, func) {
27
+ for (var i = 0, len = arr.length; i < len; ++i) {
28
+ func( arr[i] );
29
+ }
30
+ };
31
+
32
+ var nextHighlightId = 1;
33
+
34
+ /*----------------------------------------------------------------------------------------------------------------*/
35
+
36
+ var highlighterTypes = {};
37
+
38
+ function HighlighterType(type, converterCreator) {
39
+ this.type = type;
40
+ this.converterCreator = converterCreator;
41
+ }
42
+
43
+ HighlighterType.prototype.create = function() {
44
+ var converter = this.converterCreator();
45
+ converter.type = this.type;
46
+ return converter;
47
+ };
48
+
49
+ function registerHighlighterType(type, converterCreator) {
50
+ highlighterTypes[type] = new HighlighterType(type, converterCreator);
51
+ }
52
+
53
+ function getConverter(type) {
54
+ var highlighterType = highlighterTypes[type];
55
+ if (highlighterType instanceof HighlighterType) {
56
+ return highlighterType.create();
57
+ } else {
58
+ throw new Error("Highlighter type '" + type + "' is not valid");
59
+ }
60
+ }
61
+
62
+ api.registerHighlighterType = registerHighlighterType;
63
+
64
+ /*----------------------------------------------------------------------------------------------------------------*/
65
+
66
+ function CharacterRange(start, end) {
67
+ this.start = start;
68
+ this.end = end;
69
+ }
70
+
71
+ CharacterRange.prototype = {
72
+ intersects: function(charRange) {
73
+ return this.start < charRange.end && this.end > charRange.start;
74
+ },
75
+
76
+ union: function(charRange) {
77
+ return new CharacterRange(Math.min(this.start, charRange.start), Math.max(this.end, charRange.end));
78
+ },
79
+
80
+ intersection: function(charRange) {
81
+ return new CharacterRange(Math.max(this.start, charRange.start), Math.min(this.end, charRange.end));
82
+ },
83
+
84
+ toString: function() {
85
+ return "[CharacterRange(" + this.start + ", " + this.end + ")]";
86
+ }
87
+ };
88
+
89
+ CharacterRange.fromCharacterRange = function(charRange) {
90
+ return new CharacterRange(charRange.start, charRange.end);
91
+ };
92
+
93
+ /*----------------------------------------------------------------------------------------------------------------*/
94
+
95
+ var textContentConverter = {
96
+ rangeToCharacterRange: function(range, containerNode) {
97
+ var bookmark = range.getBookmark(containerNode);
98
+ return new CharacterRange(bookmark.start, bookmark.end);
99
+ },
100
+
101
+ characterRangeToRange: function(doc, characterRange, containerNode) {
102
+ var range = api.createRange(doc);
103
+ range.moveToBookmark({
104
+ start: characterRange.start,
105
+ end: characterRange.end,
106
+ containerNode: containerNode
107
+ });
108
+
109
+ return range;
110
+ },
111
+
112
+ serializeSelection: function(selection, containerNode) {
113
+ var ranges = selection.getAllRanges(), rangeCount = ranges.length;
114
+ var rangeInfos = [];
115
+
116
+ var backward = rangeCount == 1 && selection.isBackward();
117
+
118
+ for (var i = 0, len = ranges.length; i < len; ++i) {
119
+ rangeInfos[i] = {
120
+ characterRange: this.rangeToCharacterRange(ranges[i], containerNode),
121
+ backward: backward
122
+ };
123
+ }
124
+
125
+ return rangeInfos;
126
+ },
127
+
128
+ restoreSelection: function(selection, savedSelection, containerNode) {
129
+ selection.removeAllRanges();
130
+ var doc = selection.win.document;
131
+ for (var i = 0, len = savedSelection.length, range, rangeInfo, characterRange; i < len; ++i) {
132
+ rangeInfo = savedSelection[i];
133
+ characterRange = rangeInfo.characterRange;
134
+ range = this.characterRangeToRange(doc, rangeInfo.characterRange, containerNode);
135
+ selection.addRange(range, rangeInfo.backward);
136
+ }
137
+ }
138
+ };
139
+
140
+ registerHighlighterType("textContent", function() {
141
+ return textContentConverter;
142
+ });
143
+
144
+ /*----------------------------------------------------------------------------------------------------------------*/
145
+
146
+ // Lazily load the TextRange-based converter so that the dependency is only checked when required.
147
+ registerHighlighterType("TextRange", (function() {
148
+ var converter;
149
+
150
+ return function() {
151
+ if (!converter) {
152
+ // Test that textRangeModule exists and is supported
153
+ var textRangeModule = api.modules.TextRange;
154
+ if (!textRangeModule) {
155
+ throw new Error("TextRange module is missing.");
156
+ } else if (!textRangeModule.supported) {
157
+ throw new Error("TextRange module is present but not supported.");
158
+ }
159
+
160
+ converter = {
161
+ rangeToCharacterRange: function(range, containerNode) {
162
+ return CharacterRange.fromCharacterRange( range.toCharacterRange(containerNode) );
163
+ },
164
+
165
+ characterRangeToRange: function(doc, characterRange, containerNode) {
166
+ var range = api.createRange(doc);
167
+ range.selectCharacters(containerNode, characterRange.start, characterRange.end);
168
+ return range;
169
+ },
170
+
171
+ serializeSelection: function(selection, containerNode) {
172
+ return selection.saveCharacterRanges(containerNode);
173
+ },
174
+
175
+ restoreSelection: function(selection, savedSelection, containerNode) {
176
+ selection.restoreCharacterRanges(containerNode, savedSelection);
177
+ }
178
+ };
179
+ }
180
+
181
+ return converter;
182
+ };
183
+ })());
184
+
185
+ /*----------------------------------------------------------------------------------------------------------------*/
186
+
187
+ function Highlight(doc, characterRange, classApplier, converter, id, containerElementId) {
188
+ if (id) {
189
+ this.id = id;
190
+ nextHighlightId = Math.max(nextHighlightId, id + 1);
191
+ } else {
192
+ this.id = nextHighlightId++;
193
+ }
194
+ this.characterRange = characterRange;
195
+ this.doc = doc;
196
+ this.classApplier = classApplier;
197
+ this.converter = converter;
198
+ this.containerElementId = containerElementId || null;
199
+ this.applied = false;
200
+ }
201
+
202
+ Highlight.prototype = {
203
+ getContainerElement: function() {
204
+ return this.containerElementId ? this.doc.getElementById(this.containerElementId) : getBody(this.doc);
205
+ },
206
+
207
+ getRange: function() {
208
+ return this.converter.characterRangeToRange(this.doc, this.characterRange, this.getContainerElement());
209
+ },
210
+
211
+ fromRange: function(range) {
212
+ this.characterRange = this.converter.rangeToCharacterRange(range, this.getContainerElement());
213
+ },
214
+
215
+ getText: function() {
216
+ return this.getRange().toString();
217
+ },
218
+
219
+ containsElement: function(el) {
220
+ return this.getRange().containsNodeContents(el.firstChild);
221
+ },
222
+
223
+ unapply: function() {
224
+ this.classApplier.undoToRange(this.getRange());
225
+ this.applied = false;
226
+ },
227
+
228
+ apply: function() {
229
+ this.classApplier.applyToRange(this.getRange());
230
+ this.applied = true;
231
+ },
232
+
233
+ getHighlightElements: function() {
234
+ return this.classApplier.getElementsWithClassIntersectingRange(this.getRange());
235
+ },
236
+
237
+ toString: function() {
238
+ return "[Highlight(ID: " + this.id + ", class: " + this.classApplier.cssClass + ", character range: " +
239
+ this.characterRange.start + " - " + this.characterRange.end + ")]";
240
+ }
241
+ };
242
+
243
+ /*----------------------------------------------------------------------------------------------------------------*/
244
+
245
+ function Highlighter(doc, type) {
246
+ type = type || "textContent";
247
+ this.doc = doc || document;
248
+ this.classAppliers = {};
249
+ this.highlights = [];
250
+ this.converter = getConverter(type);
251
+ }
252
+
253
+ Highlighter.prototype = {
254
+ addClassApplier: function(classApplier) {
255
+ this.classAppliers[classApplier.cssClass] = classApplier;
256
+ },
257
+
258
+ getHighlightForElement: function(el) {
259
+ var highlights = this.highlights;
260
+ for (var i = 0, len = highlights.length; i < len; ++i) {
261
+ if (highlights[i].containsElement(el)) {
262
+ return highlights[i];
263
+ }
264
+ }
265
+ return null;
266
+ },
267
+
268
+ removeHighlights: function(highlights) {
269
+ for (var i = 0, len = this.highlights.length, highlight; i < len; ++i) {
270
+ highlight = this.highlights[i];
271
+ if (contains(highlights, highlight)) {
272
+ highlight.unapply();
273
+ this.highlights.splice(i--, 1);
274
+ }
275
+ }
276
+ },
277
+
278
+ removeAllHighlights: function() {
279
+ this.removeHighlights(this.highlights);
280
+ },
281
+
282
+ getIntersectingHighlights: function(ranges) {
283
+ // Test each range against each of the highlighted ranges to see whether they overlap
284
+ var intersectingHighlights = [], highlights = this.highlights, converter = this.converter;
285
+ forEach(ranges, function(range) {
286
+ //var selCharRange = converter.rangeToCharacterRange(range);
287
+ forEach(highlights, function(highlight) {
288
+ if (range.intersectsRange( highlight.getRange() ) && !contains(intersectingHighlights, highlight)) {
289
+ intersectingHighlights.push(highlight);
290
+ }
291
+ });
292
+ });
293
+
294
+ return intersectingHighlights;
295
+ },
296
+
297
+ highlightCharacterRanges: function(className, charRanges, containerElementId) {
298
+ var i, len, j;
299
+ var highlights = this.highlights;
300
+ var converter = this.converter;
301
+ var doc = this.doc;
302
+ var highlightsToRemove = [];
303
+ var classApplier = this.classAppliers[className];
304
+ containerElementId = containerElementId || null;
305
+
306
+ var containerElement, containerElementRange, containerElementCharRange;
307
+ if (containerElementId) {
308
+ containerElement = this.doc.getElementById(containerElementId);
309
+ if (containerElement) {
310
+ containerElementRange = api.createRange(this.doc);
311
+ containerElementRange.selectNodeContents(containerElement);
312
+ containerElementCharRange = new CharacterRange(0, containerElementRange.toString().length);
313
+ containerElementRange.detach();
314
+ }
315
+ }
316
+
317
+ var charRange, highlightCharRange, merged;
318
+ for (i = 0, len = charRanges.length; i < len; ++i) {
319
+ charRange = charRanges[i];
320
+ merged = false;
321
+
322
+ // Restrict character range to container element, if it exists
323
+ if (containerElementCharRange) {
324
+ charRange = charRange.intersection(containerElementCharRange);
325
+ }
326
+
327
+ // Check for intersection with existing highlights. For each intersection, create a new highlight
328
+ // which is the union of the highlight range and the selected range
329
+ for (j = 0; j < highlights.length; ++j) {
330
+ if (containerElementId == highlights[j].containerElementId) {
331
+ highlightCharRange = highlights[j].characterRange;
332
+
333
+ if (highlightCharRange.intersects(charRange)) {
334
+ // Replace the existing highlight in the list of current highlights and add it to the list for
335
+ // removal
336
+ highlightsToRemove.push(highlights[j]);
337
+ highlights[j] = new Highlight(doc, highlightCharRange.union(charRange), classApplier, converter, null, containerElementId);
338
+ }
339
+ }
340
+ }
341
+
342
+ if (!merged) {
343
+ highlights.push( new Highlight(doc, charRange, classApplier, converter, null, containerElementId) );
344
+ }
345
+ }
346
+
347
+ // Remove the old highlights
348
+ forEach(highlightsToRemove, function(highlightToRemove) {
349
+ highlightToRemove.unapply();
350
+ });
351
+
352
+ // Apply new highlights
353
+ var newHighlights = [];
354
+ forEach(highlights, function(highlight) {
355
+ if (!highlight.applied) {
356
+ highlight.apply();
357
+ newHighlights.push(highlight);
358
+ }
359
+ });
360
+
361
+ return newHighlights;
362
+ },
363
+
364
+ highlightRanges: function(className, ranges, containerElement) {
365
+ var selCharRanges = [];
366
+ var converter = this.converter;
367
+ var containerElementId = containerElement ? containerElement.id : null;
368
+ var containerElementRange;
369
+ if (containerElement) {
370
+ containerElementRange = api.createRange(containerElement);
371
+ containerElementRange.selectNodeContents(containerElement);
372
+ }
373
+
374
+ forEach(ranges, function(range) {
375
+ var scopedRange = containerElement ? containerElementRange.intersection(range) : range;
376
+ selCharRanges.push( converter.rangeToCharacterRange(scopedRange, containerElement || getBody(range.getDocument())) );
377
+ });
378
+
379
+ return this.highlightCharacterRanges(selCharRanges, ranges, containerElementId);
380
+ },
381
+
382
+ highlightSelection: function(className, selection, containerElementId) {
383
+ var converter = this.converter;
384
+ selection = selection || api.getSelection();
385
+ var classApplier = this.classAppliers[className];
386
+ var doc = selection.win.document;
387
+ var containerElement = containerElementId ? doc.getElementById(containerElementId) : getBody(doc);
388
+
389
+ if (!classApplier) {
390
+ throw new Error("No class applier found for class '" + className + "'");
391
+ }
392
+
393
+ // Store the existing selection as character ranges
394
+ var serializedSelection = converter.serializeSelection(selection, containerElement);
395
+
396
+ // Create an array of selected character ranges
397
+ var selCharRanges = [];
398
+ forEach(serializedSelection, function(rangeInfo) {
399
+ selCharRanges.push( CharacterRange.fromCharacterRange(rangeInfo.characterRange) );
400
+ });
401
+
402
+ var newHighlights = this.highlightCharacterRanges(className, selCharRanges, containerElementId);
403
+
404
+ // Restore selection
405
+ converter.restoreSelection(selection, serializedSelection, containerElement);
406
+
407
+ return newHighlights;
408
+ },
409
+
410
+ unhighlightSelection: function(selection) {
411
+ selection = selection || api.getSelection();
412
+ var intersectingHighlights = this.getIntersectingHighlights( selection.getAllRanges() );
413
+ this.removeHighlights(intersectingHighlights);
414
+ selection.removeAllRanges();
415
+ return intersectingHighlights;
416
+ },
417
+
418
+ getHighlightsInSelection: function(selection) {
419
+ selection = selection || api.getSelection();
420
+ return this.getIntersectingHighlights(selection.getAllRanges());
421
+ },
422
+
423
+ selectionOverlapsHighlight: function(selection) {
424
+ return this.getHighlightsInSelection(selection).length > 0;
425
+ },
426
+
427
+ serialize: function(options) {
428
+ var highlights = this.highlights;
429
+ highlights.sort(compareHighlights);
430
+ var serializedHighlights = ["type:" + this.converter.type];
431
+
432
+ forEach(highlights, function(highlight) {
433
+ var characterRange = highlight.characterRange;
434
+ var parts = [
435
+ characterRange.start,
436
+ characterRange.end,
437
+ highlight.id,
438
+ highlight.classApplier.cssClass,
439
+ highlight.containerElementId
440
+ ];
441
+ if (options && options.serializeHighlightText) {
442
+ parts.push(highlight.getText());
443
+ }
444
+ serializedHighlights.push( parts.join("$") );
445
+ });
446
+
447
+ return serializedHighlights.join("|");
448
+ },
449
+
450
+ deserialize: function(serialized) {
451
+ var serializedHighlights = serialized.split("|");
452
+ var highlights = [];
453
+
454
+ var firstHighlight = serializedHighlights[0];
455
+ var regexResult;
456
+ var serializationType, serializationConverter, convertType = false;
457
+ if ( firstHighlight && (regexResult = /^type:(\w+)$/.exec(firstHighlight)) ) {
458
+ serializationType = regexResult[1];
459
+ if (serializationType != this.converter.type) {
460
+ serializationConverter = getConverter(serializationType);
461
+ convertType = true;
462
+ }
463
+ serializedHighlights.shift();
464
+ } else {
465
+ throw new Error("Serialized highlights are invalid.");
466
+ }
467
+
468
+ var classApplier, highlight, characterRange, containerElementId, containerElement;
469
+
470
+ for (var i = serializedHighlights.length, parts; i-- > 0; ) {
471
+ parts = serializedHighlights[i].split("$");
472
+ characterRange = new CharacterRange(+parts[0], +parts[1]);
473
+ containerElementId = parts[4] || null;
474
+ containerElement = containerElementId ? this.doc.getElementById(containerElementId) : getBody(this.doc);
475
+
476
+ // Convert to the current Highlighter's type, if different from the serialization type
477
+ if (convertType) {
478
+ characterRange = this.converter.rangeToCharacterRange(
479
+ serializationConverter.characterRangeToRange(this.doc, characterRange, containerElement),
480
+ containerElement
481
+ );
482
+ }
483
+
484
+ classApplier = this.classAppliers[parts[3]];
485
+ highlight = new Highlight(this.doc, characterRange, classApplier, this.converter, parseInt(parts[2]), containerElementId);
486
+ highlight.apply();
487
+ highlights.push(highlight);
488
+ }
489
+ this.highlights = highlights;
490
+ }
491
+ };
492
+
493
+ api.Highlighter = Highlighter;
494
+
495
+ api.createHighlighter = function(doc, rangeCharacterOffsetConverterType) {
496
+ return new Highlighter(doc, rangeCharacterOffsetConverterType);
497
+ };
498
+ });