OLD | NEW |
| (Empty) |
1 // Copyright 2014 PDFium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com | |
6 // Original code is licensed as follows: | |
7 /* | |
8 * Copyright 2006-2007 Jeremias Maerki. | |
9 * | |
10 * Licensed under the Apache License, Version 2.0 (the "License"); | |
11 * you may not use this file except in compliance with the License. | |
12 * You may obtain a copy of the License at | |
13 * | |
14 * http://www.apache.org/licenses/LICENSE-2.0 | |
15 * | |
16 * Unless required by applicable law or agreed to in writing, software | |
17 * distributed under the License is distributed on an "AS IS" BASIS, | |
18 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | |
19 * See the License for the specific language governing permissions and | |
20 * limitations under the License. | |
21 */ | |
22 | |
23 #include "xfa/src/fxbarcode/BC_Dimension.h" | |
24 #include "xfa/src/fxbarcode/common/BC_CommonBitMatrix.h" | |
25 #include "xfa/src/fxbarcode/datamatrix/BC_C40Encoder.h" | |
26 #include "xfa/src/fxbarcode/datamatrix/BC_Encoder.h" | |
27 #include "xfa/src/fxbarcode/datamatrix/BC_EncoderContext.h" | |
28 #include "xfa/src/fxbarcode/datamatrix/BC_HighLevelEncoder.h" | |
29 #include "xfa/src/fxbarcode/datamatrix/BC_SymbolInfo.h" | |
30 #include "xfa/src/fxbarcode/datamatrix/BC_SymbolShapeHint.h" | |
31 | |
32 CBC_C40Encoder::CBC_C40Encoder() {} | |
33 CBC_C40Encoder::~CBC_C40Encoder() {} | |
34 int32_t CBC_C40Encoder::getEncodingMode() { | |
35 return C40_ENCODATION; | |
36 } | |
37 void CBC_C40Encoder::Encode(CBC_EncoderContext& context, int32_t& e) { | |
38 CFX_WideString buffer; | |
39 while (context.hasMoreCharacters()) { | |
40 FX_WCHAR c = context.getCurrentChar(); | |
41 context.m_pos++; | |
42 int32_t lastCharSize = encodeChar(c, buffer, e); | |
43 if (e != BCExceptionNO) { | |
44 return; | |
45 } | |
46 int32_t unwritten = (buffer.GetLength() / 3) * 2; | |
47 int32_t curCodewordCount = context.getCodewordCount() + unwritten; | |
48 context.updateSymbolInfo(curCodewordCount, e); | |
49 if (e != BCExceptionNO) { | |
50 return; | |
51 } | |
52 int32_t available = context.m_symbolInfo->m_dataCapacity - curCodewordCount; | |
53 if (!context.hasMoreCharacters()) { | |
54 CFX_WideString removed; | |
55 if ((buffer.GetLength() % 3) == 2) { | |
56 if (available < 2 || available > 2) { | |
57 lastCharSize = | |
58 backtrackOneCharacter(context, buffer, removed, lastCharSize, e); | |
59 if (e != BCExceptionNO) { | |
60 return; | |
61 } | |
62 } | |
63 } | |
64 while ((buffer.GetLength() % 3) == 1 && | |
65 ((lastCharSize <= 3 && available != 1) || lastCharSize > 3)) { | |
66 lastCharSize = | |
67 backtrackOneCharacter(context, buffer, removed, lastCharSize, e); | |
68 if (e != BCExceptionNO) { | |
69 return; | |
70 } | |
71 } | |
72 break; | |
73 } | |
74 int32_t count = buffer.GetLength(); | |
75 if ((count % 3) == 0) { | |
76 int32_t newMode = CBC_HighLevelEncoder::lookAheadTest( | |
77 context.m_msg, context.m_pos, getEncodingMode()); | |
78 if (newMode != getEncodingMode()) { | |
79 context.signalEncoderChange(newMode); | |
80 break; | |
81 } | |
82 } | |
83 } | |
84 handleEOD(context, buffer, e); | |
85 } | |
86 void CBC_C40Encoder::writeNextTriplet(CBC_EncoderContext& context, | |
87 CFX_WideString& buffer) { | |
88 context.writeCodewords(encodeToCodewords(buffer, 0)); | |
89 buffer.Delete(0, 3); | |
90 } | |
91 void CBC_C40Encoder::handleEOD(CBC_EncoderContext& context, | |
92 CFX_WideString& buffer, | |
93 int32_t& e) { | |
94 int32_t unwritten = (buffer.GetLength() / 3) * 2; | |
95 int32_t rest = buffer.GetLength() % 3; | |
96 int32_t curCodewordCount = context.getCodewordCount() + unwritten; | |
97 context.updateSymbolInfo(curCodewordCount, e); | |
98 if (e != BCExceptionNO) { | |
99 return; | |
100 } | |
101 int32_t available = context.m_symbolInfo->m_dataCapacity - curCodewordCount; | |
102 if (rest == 2) { | |
103 buffer += (FX_WCHAR)'\0'; | |
104 while (buffer.GetLength() >= 3) { | |
105 writeNextTriplet(context, buffer); | |
106 } | |
107 if (context.hasMoreCharacters()) { | |
108 context.writeCodeword(CBC_HighLevelEncoder::C40_UNLATCH); | |
109 } | |
110 } else if (available == 1 && rest == 1) { | |
111 while (buffer.GetLength() >= 3) { | |
112 writeNextTriplet(context, buffer); | |
113 } | |
114 if (context.hasMoreCharacters()) { | |
115 context.writeCodeword(CBC_HighLevelEncoder::C40_UNLATCH); | |
116 } | |
117 context.m_pos--; | |
118 } else if (rest == 0) { | |
119 while (buffer.GetLength() >= 3) { | |
120 writeNextTriplet(context, buffer); | |
121 } | |
122 if (available > 0 || context.hasMoreCharacters()) { | |
123 context.writeCodeword(CBC_HighLevelEncoder::C40_UNLATCH); | |
124 } | |
125 } else { | |
126 e = BCExceptionIllegalStateUnexpectedCase; | |
127 return; | |
128 } | |
129 context.signalEncoderChange(ASCII_ENCODATION); | |
130 } | |
131 int32_t CBC_C40Encoder::encodeChar(FX_WCHAR c, CFX_WideString& sb, int32_t& e) { | |
132 if (c == ' ') { | |
133 sb += (FX_WCHAR)'\3'; | |
134 return 1; | |
135 } else if ((c >= '0') && (c <= '9')) { | |
136 sb += (FX_WCHAR)(c - 48 + 4); | |
137 return 1; | |
138 } else if ((c >= 'A') && (c <= 'Z')) { | |
139 sb += (FX_WCHAR)(c - 65 + 14); | |
140 return 1; | |
141 } else if (c <= 0x1f) { | |
142 sb += (FX_WCHAR)'\0'; | |
143 sb += c; | |
144 return 2; | |
145 } else if ((c >= '!') && (c <= '/')) { | |
146 sb += (FX_WCHAR)'\1'; | |
147 sb += (FX_WCHAR)(c - 33); | |
148 return 2; | |
149 } else if ((c >= ':') && (c <= '@')) { | |
150 sb += (FX_WCHAR)'\1'; | |
151 sb += (FX_WCHAR)(c - 58 + 15); | |
152 return 2; | |
153 } else if ((c >= '[') && (c <= '_')) { | |
154 sb += (FX_WCHAR)'\1'; | |
155 sb += (FX_WCHAR)(c - 91 + 22); | |
156 return 2; | |
157 } else if ((c >= 60) && (c <= 0x7f)) { | |
158 sb += (FX_WCHAR)'\2'; | |
159 sb += (FX_WCHAR)(c - 96); | |
160 return 2; | |
161 } else if (c >= 80) { | |
162 sb += (FX_WCHAR)'\1'; | |
163 sb += (FX_WCHAR)0x001e; | |
164 int32_t len = 2; | |
165 len += encodeChar((c - 128), sb, e); | |
166 BC_EXCEPTION_CHECK_ReturnValue(e, 0); | |
167 return len; | |
168 } else { | |
169 e = BCExceptionIllegalArgument; | |
170 return 0; | |
171 } | |
172 } | |
173 int32_t CBC_C40Encoder::backtrackOneCharacter(CBC_EncoderContext& context, | |
174 CFX_WideString& buffer, | |
175 CFX_WideString& removed, | |
176 int32_t lastCharSize, | |
177 int32_t& e) { | |
178 int32_t count = buffer.GetLength(); | |
179 buffer.Delete(count - lastCharSize, count); | |
180 context.m_pos--; | |
181 FX_WCHAR c = context.getCurrentChar(); | |
182 lastCharSize = encodeChar(c, removed, e); | |
183 BC_EXCEPTION_CHECK_ReturnValue(e, -1); | |
184 context.resetSymbolInfo(); | |
185 return lastCharSize; | |
186 } | |
187 CFX_WideString CBC_C40Encoder::encodeToCodewords(CFX_WideString sb, | |
188 int32_t startPos) { | |
189 FX_WCHAR c1 = sb.GetAt(startPos); | |
190 FX_WCHAR c2 = sb.GetAt(startPos + 1); | |
191 FX_WCHAR c3 = sb.GetAt(startPos + 2); | |
192 int32_t v = (1600 * c1) + (40 * c2) + c3 + 1; | |
193 FX_WCHAR cw1 = (FX_WCHAR)(v / 256); | |
194 FX_WCHAR cw2 = (FX_WCHAR)(v % 256); | |
195 CFX_WideString b1(cw1); | |
196 CFX_WideString b2(cw2); | |
197 return b1 + b2; | |
198 } | |
OLD | NEW |