smartxml: handling undefined value sent to DocumentNode.sameNode
[fnpeditor.git] / src / smartxml / smartxml.test.js
1 define([
2     'libs/chai',
3     'libs/sinon',
4     './smartxml.js'
5 ], function(chai, sinon, smartxml) {
6     
7 'use strict';
8 /*jshint expr:true */
9 /* global describe, it, beforeEach */
10
11 var expect = chai.expect;
12
13
14 var getDocumentFromXML = function(xml) {
15     return smartxml.documentFromXML(xml);
16 };
17
18 var elementNodeFromParams = function(params) {
19     return smartxml.elementNodeFromXML('<' + params.tag + '></' + params.tag + '>');
20 };
21
22 var elementNodeFromXML = function(xml) {
23     return smartxml.elementNodeFromXML(xml);
24 };
25
26
27 describe('smartxml', function() {
28
29     describe('Basic Document properties', function() {
30         it('exposes its root element', function() {
31             var doc = getDocumentFromXML('<div></div>');
32             expect(doc.root.getTagName()).to.equal('div');
33         });
34
35         it('can resets its content entirely', function() {
36             var doc = getDocumentFromXML('<div></div>');
37
38             expect(doc.root.getTagName()).to.equal('div');
39
40             doc.loadXML('<header></header>');
41             expect(doc.root.getTagName()).to.equal('header');
42         });
43     });
44
45     describe('Basic ElementNode properties', function() {
46         it('exposes node contents', function() {
47             var node = elementNodeFromXML('<node>Some<node>text</node>is here</node>'),
48                 contents = node.contents();
49
50             expect(contents).to.have.length(3);
51             expect(contents[0].nodeType).to.equal(Node.TEXT_NODE, 'text node 1');
52             expect(contents[1].nodeType).to.equal(Node.ELEMENT_NODE, 'element node 1');
53             expect(contents[2].nodeType).to.equal(Node.TEXT_NODE, 'text node 2');
54         });
55
56         describe('Storing custom data', function() {
57             var node;
58
59             beforeEach(function() {
60                 node = elementNodeFromXML('<div></div>');
61             });
62
63             it('can append single value', function() {
64                 node.setData('key', 'value');
65                 expect(node.getData('key')).to.equal('value');
66             });
67
68             it('can overwrite the whole data', function() {
69                 node.setData('key1', 'value1');
70                 node.setData({key2: 'value2'});
71                 expect(node.getData('key2')).to.equal('value2');
72             });
73
74             it('can fetch the whole data at once', function() {
75                 node.setData({key1: 'value1', key2: 'value2'});
76                 expect(node.getData()).to.eql({key1: 'value1', key2: 'value2'});
77             });
78         });
79
80         describe('Changing node tag', function() {
81
82             it('can change tag name', function() {
83                 var node = elementNodeFromXML('<div></div>');
84                 node.setTag('span');
85                 expect(node.getTagName()).to.equal('span');
86             });
87
88             it('emits nodeTagChange event', function() {
89                 var node = elementNodeFromXML('<div></div>'),
90                     spy = sinon.spy();
91
92                 node.document.on('change', spy);
93                 node.setTag('span');
94                 var event = spy.args[0][0];
95
96                 expect(event.type).to.equal('nodeTagChange');
97                 expect(event.meta.node.sameNode(node)).to.be.true;
98                 expect(event.meta.oldTagName).to.equal('div');
99             });
100
101             describe('Implementation specific expectations', function() {
102                 // DOM specifies ElementNode tag as a read-only property, so
103                 // changing it in a seamless way is a little bit tricky. For this reason
104                 // the folowing expectations are required, despite the fact that they actually are
105                 // motivated by implemetation details.
106
107                 it('keeps node in the document', function() {
108                     var doc = getDocumentFromXML('<div><header></header></div>'),
109                         header = doc.root.contents()[0];
110                     header.setTag('span');
111                     expect(header.parent().sameNode(doc.root)).to.be.true;
112                 });
113                 it('keeps custom data', function() {
114                     var node = elementNodeFromXML('<div></div>');
115
116                     node.setData('key', 'value');
117                     node.setTag('header');
118                     
119                     expect(node.getTagName()).to.equal('header');
120                     expect(node.getData()).to.eql({key: 'value'});
121                 });
122
123                 it('can change document root tag name', function() {
124                     var doc = getDocumentFromXML('<div></div>');
125                     doc.root.setTag('span');
126                     expect(doc.root.getTagName()).to.equal('span');
127                 });
128
129                 it('keeps contents', function() {
130                     var node = elementNodeFromXML('<div><div></div></div>');
131                     node.setTag('header');
132                     expect(node.contents()).to.have.length(1);
133                 });
134             });
135
136         describe('Setting node attributes', function() {
137             it('can set node attribute', function() {
138                 var node = elementNodeFromXML('<div></div>');
139
140                 node.setAttr('key', 'value');
141                 expect(node.getAttr('key')).to.equal('value');
142             });
143             it('emits nodeAttrChange event', function() {
144                 var node = elementNodeFromXML('<div key="value1"></div>'),
145                     spy = sinon.spy();
146
147                 node.document.on('change', spy);
148                 node.setAttr('key', 'value2');
149                 var event = spy.args[0][0];
150
151                 expect(event.type).to.equal('nodeAttrChange');
152                 expect(event.meta.node.sameNode(node)).to.be.true;
153                 expect(event.meta.attr).to.equal('key');
154                 expect(event.meta.oldVal).to.equal('value1');
155             });
156         });
157
158         });
159     });
160
161     describe('Basic TextNode properties', function() {
162         it('can have its text set', function() {
163             var node = elementNodeFromXML('<div>Alice</div>'),
164                 textNode = node.contents()[0];
165
166             textNode.setText('Cat');
167             expect(textNode.getText()).to.equal('Cat');
168         });
169
170         it('emits nodeTextChange', function() {
171             var node = elementNodeFromXML('<div>Alice</div>'),
172                 textNode = node.contents()[0],
173                 spy = sinon.spy();
174
175             textNode.document.on('change', spy);
176             textNode.setText('Cat');
177
178             var event = spy.args[0][0];
179             expect(event.type).to.equal('nodeTextChange');
180         });
181     });
182
183     describe('Manipulations', function() {
184
185         it('appends element node to another element node', function() {
186             var node1 = elementNodeFromParams({tag: 'div'}),
187                 node2 = elementNodeFromParams({tag: 'a'});
188             node1.append(node2);
189             expect(node1.contents()[0].sameNode(node2)).to.be.true;
190         });
191
192         it('wraps element node with another element node', function() {
193             var node = elementNodeFromXML('<div></div>'),
194                 wrapper = elementNodeFromXML('<wrapper></wrapper>');
195
196             node.wrapWith(wrapper);
197             expect(node.parent().sameNode(wrapper)).to.be.true;
198         });
199
200         it('wraps text node with element node', function() {
201             var node = elementNodeFromXML('<div>Alice</div>'),
202                 textNode = node.contents()[0],
203                 wrapper = elementNodeFromXML('<wrapper></wrapper>');
204
205             textNode.wrapWith(wrapper);
206             expect(textNode.parent().sameNode(wrapper)).to.be.true;
207             expect(node.contents()).to.have.length(1);
208         });
209
210         it('unwraps element node contents', function() {
211             var node = elementNodeFromXML('<div>Alice <div>has <span>propably</span> a cat</div>!</div>'),
212                 outerDiv = node.contents()[1];
213             
214             outerDiv.unwrapContent();
215
216             expect(node.contents().length).to.equal(3);
217             expect(node.contents()[0].getText()).to.equal('Alice has ');
218             expect(node.contents()[1].getTagName()).to.equal('span');
219             expect(node.contents()[2].getText()).to.equal(' a cat!');
220         });
221
222     });
223
224     describe('Serializing document to WLXML', function() {
225         it('keeps document intact when no changes have been made', function() {
226             var xmlIn = '<section>Alice<div>has</div>a <span class="uri" meta-uri="http://cat.com">cat</span>!</section>',
227                 doc = getDocumentFromXML(xmlIn),
228                 xmlOut = doc.toXML();
229
230             var parser = new DOMParser(),
231                 input = parser.parseFromString(xmlIn, 'application/xml').childNodes[0],
232                 output = parser.parseFromString(xmlOut, 'application/xml').childNodes[0];
233             
234             expect(input.isEqualNode(output)).to.be.true;
235         });
236
237         it('keeps entities intact', function() {
238             var xmlIn = '<section>&lt; &gt;</section>',
239                 doc = getDocumentFromXML(xmlIn),
240                 xmlOut = doc.toXML();
241             expect(xmlOut).to.equal(xmlIn);
242         });
243         it('keeps entities intact when they form html/xml', function() {
244             var xmlIn = '<section>&lt;abc&gt;</section>',
245                 doc = getDocumentFromXML(xmlIn),
246                 xmlOut = doc.toXML();
247             expect(xmlOut).to.equal(xmlIn);
248         });
249     });
250
251 });
252
253 });