emitterstate.cpp 7.62 KB
Newer Older
1
#include "emitterstate.h"
2
#include "yaml-cpp/exceptions.h"
3
#include <limits>
4
5
6

namespace YAML
{
Jesse Beder's avatar
Jesse Beder committed
7
	EmitterState::EmitterState(): m_isGood(true), m_curIndent(0), m_hasAnchor(false), m_hasTag(false), m_hasNonContent(false)
8
9
	{
		// set default global manipulators
10
		m_charset.set(EmitNonAscii);
11
12
13
14
15
16
17
18
19
20
21
		m_strFmt.set(Auto);
		m_boolFmt.set(TrueFalseBool);
		m_boolLengthFmt.set(LongBool);
		m_boolCaseFmt.set(LowerCase);
		m_intFmt.set(Dec);
		m_indent.set(2);
		m_preCommentIndent.set(2);
		m_postCommentIndent.set(1);
		m_seqFmt.set(Block);
		m_mapFmt.set(Block);
		m_mapKeyFmt.set(Auto);
22
23
        m_floatPrecision.set(6);
        m_doublePrecision.set(15);
24
25
26
27
28
29
30
31
32
33
34
	}
	
	EmitterState::~EmitterState()
	{
	}

	// SetLocalValue
	// . We blindly tries to set all possible formatters to this value
	// . Only the ones that make sense will be accepted
	void EmitterState::SetLocalValue(EMITTER_MANIP value)
	{
35
36
37
38
39
40
41
42
43
		SetOutputCharset(value, FmtScope::Local);
		SetStringFormat(value, FmtScope::Local);
		SetBoolFormat(value, FmtScope::Local);
		SetBoolCaseFormat(value, FmtScope::Local);
		SetBoolLengthFormat(value, FmtScope::Local);
		SetIntFormat(value, FmtScope::Local);
		SetFlowType(GroupType::Seq, value, FmtScope::Local);
		SetFlowType(GroupType::Map, value, FmtScope::Local);
		SetMapKeyFormat(value, FmtScope::Local);
44
45
	}
	
Jesse Beder's avatar
Jesse Beder committed
46
47
48
49
50
51
52
53
54
55
    void EmitterState::SetAnchor()
    {
        m_hasAnchor = true;
    }
        
    void EmitterState::SetTag()
    {
        m_hasTag = true;
    }

Jesse Beder's avatar
Jesse Beder committed
56
57
58
59
60
    void EmitterState::SetNonContent()
    {
        m_hasNonContent = true;
    }

61
    void EmitterState::StartedNode()
Jesse Beder's avatar
Jesse Beder committed
62
63
64
65
66
67
    {
        if(!m_groups.empty())
            m_groups.top().childCount++;
        
        m_hasAnchor = false;
        m_hasTag = false;
Jesse Beder's avatar
Jesse Beder committed
68
        m_hasNonContent = false;
Jesse Beder's avatar
Jesse Beder committed
69
70
    }

71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
    EmitterNodeType::value EmitterState::NextGroupType(GroupType::value type) const
    {
        if(type == GroupType::Seq) {
            if(GetFlowType(type) == Block)
                return EmitterNodeType::BlockSeq;
            else
                return EmitterNodeType::FlowSeq;
        } else {
            if(GetFlowType(type) == Block)
                return EmitterNodeType::BlockMap;
            else
                return EmitterNodeType::FlowMap;
        }
        
        // can't happen
        assert(false);
        return EmitterNodeType::None;
    }
    
90
    void EmitterState::StartedScalar()
Jesse Beder's avatar
Jesse Beder committed
91
    {
92
        StartedNode();
Jesse Beder's avatar
Jesse Beder committed
93
94
    }

95
	void EmitterState::StartedGroup(GroupType::value type)
96
	{
97
        StartedNode();
Jesse Beder's avatar
Jesse Beder committed
98

Jesse Beder's avatar
Jesse Beder committed
99
100
		const int lastGroupIndent = (m_groups.empty() ? 0 : m_groups.top().indent);
		m_curIndent += lastGroupIndent;
101
		
102
		std::auto_ptr<Group> pGroup(new Group(type));
103
104
105
106
107
		
		// transfer settings (which last until this group is done)
		pGroup->modifiedSettings = m_modifiedSettings;

		// set up group
108
109
110
111
        if(GetFlowType(type) == Block)
            pGroup->flowType = FlowType::Block;
        else
            pGroup->flowType = FlowType::Flow;
112
113
		pGroup->indent = GetIndent();

114
		m_groups.push(pGroup);
115
116
	}
	
117
	void EmitterState::EndedGroup(GroupType::value type)
118
119
120
121
122
123
	{
		if(m_groups.empty())
			return SetError(ErrorMsg::UNMATCHED_GROUP_TAG);
		
		// get rid of the current group
		{
124
			std::auto_ptr<Group> pFinishedGroup = m_groups.pop();
125
126
127
128
129
			if(pFinishedGroup->type != type)
				return SetError(ErrorMsg::UNMATCHED_GROUP_TAG);
		}

		// reset old settings
130
		unsigned lastIndent = (m_groups.empty() ? 0 : m_groups.top().indent);
131
132
133
134
135
136
137
		assert(m_curIndent >= lastIndent);
		m_curIndent -= lastIndent;
		
		// some global settings that we changed may have been overridden
		// by a local setting we just popped, so we need to restore them
		m_globalModifiedSettings.restore();
	}
138
139
140
141
142
143
144
145
146
147

    EmitterNodeType::value EmitterState::CurGroupNodeType() const
    {
        if(m_groups.empty())
            return EmitterNodeType::None;
        
        return m_groups.top().NodeType();
    }

    GroupType::value EmitterState::CurGroupType() const
148
	{
149
        return m_groups.empty() ? GroupType::None : m_groups.top().type;
150
151
	}
	
Jesse Beder's avatar
Jesse Beder committed
152
    FlowType::value EmitterState::CurGroupFlowType() const
153
	{
154
        return m_groups.empty() ? FlowType::None : m_groups.top().flowType;
155
156
	}

Jesse Beder's avatar
Jesse Beder committed
157
158
159
160
161
    int EmitterState::CurGroupIndent() const
    {
        return m_groups.empty() ? 0 : m_groups.top().indent;
    }

162
163
164
165
166
    std::size_t EmitterState::CurGroupChildCount() const
    {
        return m_groups.empty() ? 0 : m_groups.top().childCount;
    }

167
168
169
170
	void EmitterState::ClearModifiedSettings()
	{
		m_modifiedSettings.clear();
	}
171

172
	bool EmitterState::SetOutputCharset(EMITTER_MANIP value, FmtScope::value scope)
173
174
175
176
177
178
179
180
181
182
	{
		switch(value) {
			case EmitNonAscii:
			case EscapeNonAscii:
				_Set(m_charset, value, scope);
				return true;
			default:
				return false;
		}
	}
183
	
184
	bool EmitterState::SetStringFormat(EMITTER_MANIP value, FmtScope::value scope)
185
186
187
188
189
190
191
192
193
194
195
196
197
	{
		switch(value) {
			case Auto:
			case SingleQuoted:
			case DoubleQuoted:
			case Literal:
				_Set(m_strFmt, value, scope);
				return true;
			default:
				return false;
		}
	}
	
198
	bool EmitterState::SetBoolFormat(EMITTER_MANIP value, FmtScope::value scope)
199
200
201
202
203
204
205
206
207
208
209
210
	{
		switch(value) {
			case OnOffBool:
			case TrueFalseBool:
			case YesNoBool:
				_Set(m_boolFmt, value, scope);
				return true;
			default:
				return false;
		}
	}

211
	bool EmitterState::SetBoolLengthFormat(EMITTER_MANIP value, FmtScope::value scope)
212
213
214
215
216
217
218
219
220
221
222
	{
		switch(value) {
			case LongBool:
			case ShortBool:
				_Set(m_boolLengthFmt, value, scope);
				return true;
			default:
				return false;
		}
	}

223
	bool EmitterState::SetBoolCaseFormat(EMITTER_MANIP value, FmtScope::value scope)
224
225
226
227
228
229
230
231
232
233
234
235
	{
		switch(value) {
			case UpperCase:
			case LowerCase:
			case CamelCase:
				_Set(m_boolCaseFmt, value, scope);
				return true;
			default:
				return false;
		}
	}

236
	bool EmitterState::SetIntFormat(EMITTER_MANIP value, FmtScope::value scope)
237
238
239
240
241
242
243
244
245
246
247
248
	{
		switch(value) {
			case Dec:
			case Hex:
			case Oct:
				_Set(m_intFmt, value, scope);
				return true;
			default:
				return false;
		}
	}

249
	bool EmitterState::SetIndent(unsigned value, FmtScope::value scope)
250
251
252
253
254
255
256
257
	{
		if(value == 0)
			return false;
		
		_Set(m_indent, value, scope);
		return true;
	}

258
	bool EmitterState::SetPreCommentIndent(unsigned value, FmtScope::value scope)
259
260
261
262
263
264
265
266
	{
		if(value == 0)
			return false;
		
		_Set(m_preCommentIndent, value, scope);
		return true;
	}
	
267
	bool EmitterState::SetPostCommentIndent(unsigned value, FmtScope::value scope)
268
269
270
271
272
273
274
275
	{
		if(value == 0)
			return false;
		
		_Set(m_postCommentIndent, value, scope);
		return true;
	}

276
	bool EmitterState::SetFlowType(GroupType::value groupType, EMITTER_MANIP value, FmtScope::value scope)
277
278
279
280
	{
		switch(value) {
			case Block:
			case Flow:
281
				_Set(groupType == GroupType::Seq ? m_seqFmt : m_mapFmt, value, scope);
282
283
284
285
286
287
				return true;
			default:
				return false;
		}
	}

288
	EMITTER_MANIP EmitterState::GetFlowType(GroupType::value groupType) const
289
290
	{
		// force flow style if we're currently in a flow
Jesse Beder's avatar
Jesse Beder committed
291
		if(CurGroupFlowType() == FlowType::Flow)
292
293
			return Flow;
		
294
295
		// otherwise, go with what's asked of us
		return (groupType == GroupType::Seq ? m_seqFmt.get() : m_mapFmt.get());
296
297
	}
	
298
	bool EmitterState::SetMapKeyFormat(EMITTER_MANIP value, FmtScope::value scope)
299
300
301
302
303
304
305
306
307
308
	{
		switch(value) {
			case Auto:
			case LongKey:
				_Set(m_mapKeyFmt, value, scope);
				return true;
			default:
				return false;
		}
	}
309

310
    bool EmitterState::SetFloatPrecision(int value, FmtScope::value scope)
311
    {
312
        if(value < 0 || value > std::numeric_limits<float>::digits10)
313
314
315
316
317
            return false;
        _Set(m_floatPrecision, value, scope);
        return true;
    }
    
318
    bool EmitterState::SetDoublePrecision(int value, FmtScope::value scope)
319
    {
320
        if(value < 0 || value > std::numeric_limits<double>::digits10)
321
322
323
324
            return false;
        _Set(m_doublePrecision, value, scope);
        return true;
    }
325
326
}