From a341901d9441547942e9b57f9b02cc83a35d4a92 Mon Sep 17 00:00:00 2001 From: Project Nayuki Date: Thu, 17 Aug 2017 20:03:19 +0000 Subject: [PATCH] Simplified Python code by making BitBuffer subclass list, updated internal usages. --- python/qrcodegen.py | 50 +++++++++++++++------------------------------ 1 file changed, 16 insertions(+), 34 deletions(-) diff --git a/python/qrcodegen.py b/python/qrcodegen.py index e65957a..7001588 100644 --- a/python/qrcodegen.py +++ b/python/qrcodegen.py @@ -120,18 +120,18 @@ class QrCode(object): for seg in segs: bb.append_bits(seg.get_mode().get_mode_bits(), 4) bb.append_bits(seg.get_num_chars(), seg.get_mode().num_char_count_bits(version)) - bb.append_all(seg) + bb.extend(seg._bitdata) # Add terminator and pad up to a byte if applicable - bb.append_bits(0, min(4, datacapacitybits - bb.bit_length())) - bb.append_bits(0, -bb.bit_length() % 8) # Note: Python's modulo on negative numbers behaves better than C family languages + bb.append_bits(0, min(4, datacapacitybits - len(bb))) + bb.append_bits(0, -len(bb) % 8) # Note: Python's modulo on negative numbers behaves better than C family languages # Pad with alternate bytes until data capacity is reached for padbyte in itertools.cycle((0xEC, 0x11)): - if bb.bit_length() >= datacapacitybits: + if len(bb) >= datacapacitybits: break bb.append_bits(padbyte, 8) - assert bb.bit_length() % 8 == 0 + assert len(bb) % 8 == 0 # Create the QR Code symbol return QrCode(None, bb.get_bytes(), mask, version, ecl) @@ -619,7 +619,7 @@ class QrSegment(object): bb = _BitBuffer() for b in data: bb.append_bits(b, 8) - return QrSegment(QrSegment.Mode.BYTE, len(data), bb.get_bits()) + return QrSegment(QrSegment.Mode.BYTE, len(data), bb) @staticmethod @@ -633,7 +633,7 @@ class QrSegment(object): rem = len(digits) % 3 if rem > 0: # 1 or 2 digits remaining bb.append_bits(int(digits[-rem : ]), rem * 3 + 1) - return QrSegment(QrSegment.Mode.NUMERIC, len(digits), bb.get_bits()) + return QrSegment(QrSegment.Mode.NUMERIC, len(digits), bb) @staticmethod @@ -649,7 +649,7 @@ class QrSegment(object): bb.append_bits(temp, 11) if len(text) % 2 > 0: # 1 character remaining bb.append_bits(QrSegment._ALPHANUMERIC_ENCODING_TABLE[text[-1]], 6) - return QrSegment(QrSegment.Mode.ALPHANUMERIC, len(text), bb.get_bits()) + return QrSegment(QrSegment.Mode.ALPHANUMERIC, len(text), bb) @staticmethod @@ -685,7 +685,7 @@ class QrSegment(object): bb.append_bits(assignval, 21) else: raise ValueError("ECI assignment value out of range") - return QrSegment(QrSegment.Mode.ECI, 0, bb.get_bits()) + return QrSegment(QrSegment.Mode.ECI, 0, bb) # ---- Constructor ---- @@ -721,7 +721,7 @@ class QrSegment(object): # Fail if segment length value doesn't fit in the length field's bit-width if seg.get_num_chars() >= (1 << ccbits): return None - result += 4 + ccbits + len(seg.get_bits()) + result += 4 + ccbits + len(seg._bitdata) return result @@ -828,25 +828,13 @@ class _ReedSolomonGenerator(object): -class _BitBuffer(object): - """An appendable sequence of bits. Bits are packed in big endian within a byte.""" - - def __init__(self): - """Creates an empty bit buffer (length 0).""" - self.data = [] # Only contains 0s and 1s - - def bit_length(self): - """Returns the number of bits in the buffer, which is a non-negative value.""" - return len(self.data) - - def get_bits(self): - """Returns a copy of all bits.""" - return list(self.data) +class _BitBuffer(list): + """An appendable sequence of bits (0's and 1's).""" def get_bytes(self): - """Returns a copy of all bytes, padding up to the nearest byte.""" - result = [0] * ((len(self.data) + 7) // 8) - for (i, bit) in enumerate(self.data): + """Returns a copy of all bytes, padding up to the nearest byte. Bits are packed in big endian within a byte.""" + result = [0] * ((len(self) + 7) // 8) + for (i, bit) in enumerate(self): result[i >> 3] |= bit << (7 - (i & 7)) return result @@ -854,10 +842,4 @@ class _BitBuffer(object): """Appends the given number of bits of the given value to this sequence. This requires 0 <= val < 2^n.""" if n < 0 or val >> n != 0: raise ValueError("Value out of range") - self.data.extend(((val >> i) & 1) for i in reversed(range(n))) - - def append_all(self, seg): - """Appends the data of the given segment to this bit buffer.""" - if not isinstance(seg, QrSegment): - raise TypeError("QrSegment expected") - self.data.extend(seg.get_bits()) + self.extend(((val >> i) & 1) for i in reversed(range(n)))