const func string: toPackBits (in string: uncompressed) is func
  result
    var string: compressed is "";
  local
    var integer: pos is 1;
    var integer: pos2 is 1;
    var integer: count is 0;
    var char: ch is ' ';
  begin
    while pos <= length(uncompressed) do
      ch := uncompressed[pos];
      pos2 := succ(pos);
      while pos2 <= length(uncompressed) and ch = uncompressed[pos2] do
        incr(pos2);
      end while;
      if pos2 - pos >= 2 then
        count := pos2 - pos;
        while count > 128 do
          compressed &:= char(129);
          compressed &:= ch;
          count -:= 128;
        end while;
        compressed &:= char(257 - count);
        compressed &:= ch;
        pos := pos2;
      else
        while pos2 < length(uncompressed) and
            (ch <> uncompressed[pos2] or ch <> uncompressed[succ(pos2)]) do
          ch := uncompressed[pos2];
          incr(pos2);
        end while;
        if pos2 < length(uncompressed) then
          decr(pos2);
        elsif pos2 = length(uncompressed) then
          incr(pos2);
        end if;
        count := pos2 - pos;
        while count > 128 do
          compressed &:= char(127);
          compressed &:= uncompressed[pos len 128];
          count -:= 128;
          pos +:= 128;
        end while;
        compressed &:= char(pred(count));
        compressed &:= uncompressed[pos len count];
        pos := pos2;
      end if;
    end while;
  end func;
const func string: fromPackBits (in string: compressed) is func
  result
    var string: uncompressed is "";
  local
    var integer: index is 1;
    var integer: number is 0;
  begin
    while index <= length(compressed) do
      number := ord(compressed[index]);
      if number <= 127 then
        
        uncompressed &:= compressed[succ(index) fixLen succ(number)];
        index +:= number + 2;
      elsif number = 128 then
        
        incr(index);
      else
        
        uncompressed &:= str(compressed[succ(index)]) mult 257 - number;
        index +:= 2;
      end if;
    end while;
  end func;
const func string: toPackBitsPdf (in string: uncompressed) is
  return toPackBits(uncompressed) & "\128;";
const func string: fromPackBitsPdf (in string: compressed) is func
  result
    var string: uncompressed is "";
  local
    const char: endOfData is '\128;';
    var integer: index is 1;
    var integer: number is 0;
  begin
    while index <= length(compressed) and compressed[index] <> endOfData do
      number := ord(compressed[index]);
      if number <= 127 then
        
        uncompressed &:= compressed[succ(index) fixLen succ(number)];
        index +:= number + 2;
      else
        
        uncompressed &:= str(compressed[succ(index)]) mult 257 - number;
        index +:= 2;
      end if;
    end while;
  end func;