Skip to content

Commit f0cb7e6

Browse files
authored
Merge pull request #86 from tomtiao/feat/support-empty-string-title
Updated: allows empty string as topic title
2 parents d4674da + 0c8fa7b commit f0cb7e6

File tree

6 files changed

+80
-16
lines changed

6 files changed

+80
-16
lines changed

src/core/base.ts

+5-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
import * as TreeModel from 'tree-model';
22
import { Node } from 'tree-model';
33

4+
import { isString } from '../utils/common';
5+
46
const v4 = require('uuid/v4');
57
const Debug = require('debug');
68

@@ -96,11 +98,12 @@ export default class Base {
9698
protected getConflictedComponentId(
9799
options: ConflictedOnDifferentBranchOptions & ConflictedOnSameBranchOptions
98100
): string | null {
99-
if (options.title && options.parentId) {
101+
const validString = isString(options.title);
102+
if (validString && options.parentId) {
100103
return this.different(options);
101104
}
102105

103-
if (options.title && options.customId) {
106+
if (validString && options.customId) {
104107
return this.identical(options);
105108
}
106109

src/core/topic.ts

+5-4
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ import {
55
import { SummaryOptions } from '../abstracts/summary.abstract';
66
import { Summary } from './summary';
77
import { Note } from './note';
8-
import { isEmpty, isObject, isRuntime } from '../utils/common';
8+
import { isEmpty, isObject, isRuntime, isString } from '../utils/common';
99

1010
import * as Model from '../common/model';
1111
import * as Core from 'xmind-model';
@@ -73,7 +73,7 @@ export class Topic extends Base implements AbstractTopic {
7373
public add(topic: Model.Topic & {
7474
customId?: string | number, parentId?: string
7575
} = {} as any, options?: { index: number }): Topic {
76-
if (!topic.title || typeof topic.title !== 'string') {
76+
if (!isString(topic.title)) {
7777
throw new Error('topic.title should be a valid string');
7878
}
7979
topic.id = topic.id || this.id;
@@ -183,7 +183,8 @@ export class Topic extends Base implements AbstractTopic {
183183
public cid(title?: string, dependencies: {
184184
parentId?: number | string, customId?: number | string
185185
} = {}): string | null {
186-
if (title && dependencies) {
186+
const validTitle = isString(title);
187+
if (validTitle && dependencies) {
187188
if (dependencies.parentId) {
188189
return this.getConflictedComponentId({
189190
title, parentId: dependencies.parentId
@@ -196,7 +197,7 @@ export class Topic extends Base implements AbstractTopic {
196197
}
197198
}
198199

199-
if (title && typeof title === 'string') {
200+
if (validTitle) {
200201
return this.findComponentIdBy(title);
201202
}
202203
return this.lastId;

src/utils/common.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -35,5 +35,9 @@ const isRuntime = function(): boolean {
3535
return typeof global === 'object' && typeof window === 'undefined';
3636
};
3737

38-
export {isEmpty, isObject, isRuntime};
38+
const isString = function(v: any): v is string {
39+
return typeof v === 'string';
40+
};
41+
42+
export {isEmpty, isObject, isRuntime, isString};
3943

test/functional/index.test.ts

+13-7
Original file line numberDiff line numberDiff line change
@@ -44,12 +44,13 @@ describe('# Functional Test', () => {
4444

4545
describe('# Topic', () => {
4646
it('should be a lots of main topic added to rootTopic', done => {
47-
const topics = ['main topic 1', 'main topic 2', 'main topic 3', 'main topic 4'];
47+
const topics = ['main topic 1', 'main topic 2', 'main topic 3', 'main topic 4', ''];
4848
const {workbook, topic} = getComponents();
4949
topic.add({title: 'main topic 1'})
5050
.add({title: 'main topic 2'})
5151
.add({title: 'main topic 3'})
52-
.add({title: 'main topic 4'});
52+
.add({title: 'main topic 4'})
53+
.add({title: ''});
5354

5455
const children = workbook.toJSON()[0].rootTopic.children.attached;
5556
for (let i = 0; i < children.length; i++) {
@@ -72,11 +73,12 @@ describe('# Functional Test', () => {
7273
.on(topic.cid())
7374
.add({title: 'subtopic 1'})
7475
.add({title: 'subtopic 2'})
75-
.add({title: 'subtopic 3'});
76+
.add({title: 'subtopic 3'})
77+
.add({title: ''});
7678

7779
const subtopics = workbook.toJSON()[0].rootTopic.children.attached[0].children.attached;
7880
for(let i = 0; i < subtopics.length; i++) {
79-
expect(subtopics[i].title.startsWith('subtopic')).to.be.true;
81+
expect(subtopics[i].title.startsWith('subtopic') || subtopics[i].title === '').to.be.true;
8082
expect(subtopics[i].id).to.not.be.empty;
8183
}
8284
done();
@@ -97,14 +99,15 @@ describe('# Functional Test', () => {
9799
const subTopic2Id = topic.cid();
98100

99101
topic.add({title: 'subtopic 3'});
102+
topic.add({title: ''});
100103
topic.destroy(subTopic2Id);
101104
// Do nothing if you have to remove topic twice
102105
topic.destroy(subTopic2Id);
103106

104107
const subtopics = workbook.toJSON()[0].rootTopic.children.attached[0].children.attached;
105-
expect(subtopics.length).to.be.eq(2);
108+
expect(subtopics.length).to.be.eq(3);
106109
for(let i = 0; i < subtopics.length; i++) {
107-
expect(subtopics[i].title.startsWith('subtopic')).to.be.true;
110+
expect(subtopics[i].title.startsWith('subtopic') || subtopics[i].title === '').to.be.true;
108111
expect(subtopics[i].id).to.not.be.empty;
109112
}
110113
done();
@@ -159,9 +162,11 @@ describe('# Functional Test', () => {
159162
topic
160163
.add({title: 'main topic 1'})
161164
.add({title: 'main topic 2'})
162-
.add({title: 'main topic 3'});
165+
.add({title: 'main topic 3'})
166+
.add({title: ''});
163167

164168
topic.destroy(topic.cid('main topic 2'));
169+
topic.destroy(topic.cid(''));
165170
zip.save().then(async status => {
166171
expect(status).to.be.true;
167172
const p = getBuildTemporaryPath('default.xmind');
@@ -173,6 +178,7 @@ describe('# Functional Test', () => {
173178
const {attached} = map.rootTopic.children;
174179
expect(attached.length).to.be.eq(2);
175180
expect(attached.find(child => child.title === topic.cid('main topic 2'))).to.be.undefined;
181+
expect(attached.find(child => child.title === topic.cid(''))).to.be.undefined;
176182
fs.unlinkSync(p);
177183
done();
178184
});

test/units/common.test.ts

+18-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import { isEmpty, isObject } from '../../src/utils/common';
1+
import { isEmpty, isObject, isString } from '../../src/utils/common';
22
import { expect } from 'chai';
33

44

@@ -54,4 +54,21 @@ describe('# Common Utils Test', function() {
5454
expect(isObject(() => {})).to.eq(true);
5555
done();
5656
});
57+
58+
it('isString(), should return true if given any string', done => {
59+
expect(isString('')).to.eq(true);
60+
expect(isString('foo')).to.eq(true);
61+
done();
62+
})
63+
64+
it('isString(), should return false if given any non string value', done => {
65+
expect(isString(42)).to.eq(false);
66+
expect(isString(null)).to.eq(false);
67+
expect(isString(undefined)).to.eq(false);
68+
expect(isString({})).to.eq(false);
69+
expect(isString(() => {})).to.eq(false);
70+
expect(isString([])).to.eq(false);
71+
expect(isString(new String)).to.eq(false);
72+
done();
73+
})
5774
});

test/units/topic.test.ts

+34-1
Original file line numberDiff line numberDiff line change
@@ -26,7 +26,7 @@ describe('# Topic Unit Test', () => {
2626
}
2727
});
2828

29-
it('should be failed to add topic with empty title', done => {
29+
it('should be failed to add topic with non string title', done => {
3030
const {topic} = getComponents();
3131
try {
3232
topic.on().add();
@@ -36,6 +36,18 @@ describe('# Topic Unit Test', () => {
3636
}
3737
});
3838

39+
it('should be able to add topic with empty title', done => {
40+
const {topic} = getComponents();
41+
42+
topic.on().add({ title: '' });
43+
44+
const topics = topic.cids();
45+
expect(topics).to.have.property(topic.cid(''));
46+
expect(topics).to.have.property(topic.cid('Central Topic'));
47+
expect(topics).to.have.property(topic.rootTopicId);
48+
done();
49+
});
50+
3951
it('should be failed if call .on() with an invalid componentId', done => {
4052
try {
4153
const {topic} = getComponents();
@@ -226,11 +238,32 @@ describe('# Topic Unit Test', () => {
226238
expect(topic.cid('abc', { parentId: a2})).to.eq(abc1);
227239
expect(topic.cid('abc', { parentId: a1})).to.eq(abc2);
228240

241+
topic.add({ title: 'main topic - 1', customId: 3 });
242+
const a3 = topic.cid();
243+
topic.add({ title: 'main topic - 1', customId: 4 });
244+
const a4 = topic.cid();
245+
expect(topic.cid('main topic - 1', { customId: 3 })).to.eq(a3);
246+
expect(topic.cid('main topic - 1', { customId: 4 })).to.eq(a4);
247+
248+
topic.on(a4).add({title: ''});
249+
const emptyString1 = topic.cid();
250+
topic.on(a3).add({title: ''});
251+
const emptyString2 = topic.cid();
252+
253+
expect(topic.cid('', { parentId: a4 })).to.eq(emptyString1);
254+
expect(topic.cid('', { parentId: a3 })).to.eq(emptyString2);
255+
229256
topic.destroy(topic.cid('abc', { parentId: a1}));
230257
topic.destroy(topic.cid('abc', { parentId: a2}));
231258

232259
expect(topic.cid('abc', { parentId: a1})).to.eq(null);
233260
expect(topic.cid('abc', { parentId: a2})).to.eq(null);
261+
262+
topic.destroy(topic.cid('', { parentId: a3}));
263+
topic.destroy(topic.cid('', { parentId: a4}));
264+
265+
expect(topic.cid('', { parentId: a4})).to.eq(null);
266+
expect(topic.cid('', { parentId: a3})).to.eq(null);
234267
done();
235268
});
236269
});

0 commit comments

Comments
 (0)