mcabber/src/hbuf.c
author mikael
Sat, 16 Apr 2005 10:14:24 +0000
changeset 71 1e9d4949bcfd
child 75 ff119bb11563
permissions -rw-r--r--
[/trunk] Changeset 85 by mikael * New history buffer implementation
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
71
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     1
/*
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     2
 * hbuf.c       -- History buffer implementation
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     3
 * 
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     4
 * Copyright (C) 2005 Mikael Berthe <bmikael@lists.lilotux.net>
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     5
 *
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     6
 * This program is free software; you can redistribute it and/or modify
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     7
 * it under the terms of the GNU General Public License as published by
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     8
 * the Free Software Foundation; either version 2 of the License, or (at
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
     9
 * your option) any later version.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    10
 *
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    11
 * This program is distributed in the hope that it will be useful, but
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    12
 * WITHOUT ANY WARRANTY; without even the implied warranty of
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    14
 * General Public License for more details.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    15
 *
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    16
 * You should have received a copy of the GNU General Public License
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    17
 * along with this program; if not, write to the Free Software
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    18
 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    19
 * USA
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    20
 */
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    21
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    22
#include <string.h>
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    23
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    24
#include "hbuf.h"
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    25
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    26
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    27
/* This is a private structure type */
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    28
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    29
typedef struct {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    30
  char *ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    31
  char *ptr_end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    32
  guchar flags;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    33
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    34
  // XXX This should certainly be a pointer, and be allocated only when needed
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    35
  // (for ex. when HBB_FLAG_PERSISTENT is set).
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    36
  struct { // hbuf_line_info
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    37
    char *ptr_end_alloc;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    38
    char prefix[32];
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    39
  } persist;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    40
} hbuf_block;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    41
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    42
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    43
//  hbuf_add_line(p_hbuf, text, width)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    44
// Add a line to the given buffer.  If width is not null, then lines are
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    45
// wrapped at this length.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    46
//
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    47
// Note 1: Splitting according to width won't work if there are tabs; they
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    48
//         should be expanded before.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    49
// Note 2: width does not include the ending \0.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    50
void hbuf_add_line(GList **p_hbuf, char *text, unsigned int width)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    51
{
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    52
  GList *hbuf = *p_hbuf;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    53
  char *line, *cr, *end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    54
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    55
  if (!text) return;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    56
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    57
  hbuf_block *hbuf_block_elt = g_new0(hbuf_block, 1);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    58
  if (!hbuf) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    59
    hbuf_block_elt->ptr    = g_new(char, HBB_BLOCKSIZE);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    60
    hbuf_block_elt->flags  = HBB_FLAG_ALLOC | HBB_FLAG_PERSISTENT;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    61
    hbuf_block_elt->persist.ptr_end_alloc = hbuf_block_elt->ptr + HBB_BLOCKSIZE;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    62
    *p_hbuf = g_list_append(*p_hbuf, hbuf_block_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    63
  } else {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    64
    hbuf_block *hbuf_b_prev = g_list_last(hbuf)->data;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    65
    hbuf_block_elt->ptr    = hbuf_b_prev->ptr_end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    66
    hbuf_block_elt->flags  = HBB_FLAG_PERSISTENT;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    67
    hbuf_block_elt->persist.ptr_end_alloc = hbuf_b_prev->persist.ptr_end_alloc;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    68
    *p_hbuf = g_list_append(*p_hbuf, hbuf_block_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    69
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    70
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    71
  if (strlen(text) >= HBB_BLOCKSIZE) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    72
    // Too long
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    73
    text = "[ERR:LINE_TOO_LONG]";
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    74
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    75
  if (hbuf_block_elt->ptr + strlen(text) >= hbuf_block_elt->persist.ptr_end_alloc) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    76
    // Too long for the current allocated bloc, we need another one
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    77
    hbuf_block_elt->ptr    = g_new0(char, HBB_BLOCKSIZE);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    78
    hbuf_block_elt->flags  = HBB_FLAG_ALLOC | HBB_FLAG_PERSISTENT;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    79
    hbuf_block_elt->persist.ptr_end_alloc = hbuf_block_elt->ptr + HBB_BLOCKSIZE;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    80
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    81
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    82
  line = hbuf_block_elt->ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    83
  // Ok, now we can copy the text..
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    84
  strcpy(line, text);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    85
  hbuf_block_elt->ptr_end = line + strlen(line) + 1;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    86
  end = hbuf_block_elt->ptr_end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    87
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    88
  // Let's add non-persistent blocs if necessary
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    89
  // - If there are '\n' in the string
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    90
  // - If length > width (and width != 0)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    91
  cr = strchr(line, '\n');
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    92
  while (cr || (width && strlen(line) > width)) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    93
    hbuf_block *hbuf_b_prev = hbuf_block_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    94
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    95
    if (!width || (cr && (cr - line <= (int)width))) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    96
      // Carriage return
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    97
      *cr = 0;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    98
      hbuf_block_elt->ptr_end = cr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
    99
      // Create another persistent block
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   100
      hbuf_block_elt = g_new0(hbuf_block, 1);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   101
      hbuf_block_elt->ptr      = hbuf_b_prev->ptr_end + 1; // == cr+1
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   102
      hbuf_block_elt->ptr_end  = end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   103
      hbuf_block_elt->flags    = HBB_FLAG_PERSISTENT;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   104
      hbuf_block_elt->persist.ptr_end_alloc = hbuf_b_prev->persist.ptr_end_alloc;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   105
      *p_hbuf = g_list_append(*p_hbuf, hbuf_block_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   106
      line = hbuf_block_elt->ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   107
    } else {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   108
      // We need to break where we can find a space char
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   109
      char *br; // break pointer
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   110
      for (br = line + width; br > line && *br != 32 && *br != 9; br--)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   111
        ;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   112
      if (br <= line)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   113
        br = line + width;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   114
      else
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   115
        br++;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   116
      hbuf_block_elt->ptr_end = br;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   117
      // Create another block, non-persistent
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   118
      hbuf_block_elt = g_new0(hbuf_block, 1);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   119
      hbuf_block_elt->ptr      = hbuf_b_prev->ptr_end; // == br
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   120
      hbuf_block_elt->ptr_end  = end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   121
      hbuf_block_elt->flags    = 0;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   122
      hbuf_block_elt->persist.ptr_end_alloc = hbuf_b_prev->persist.ptr_end_alloc;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   123
      *p_hbuf = g_list_append(*p_hbuf, hbuf_block_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   124
      line = hbuf_block_elt->ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   125
    }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   126
    cr = strchr(line, '\n');
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   127
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   128
}
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   129
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   130
//  hbuf_free()
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   131
// Destroys all hbuf list.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   132
void hbuf_free(GList **p_hbuf)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   133
{
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   134
  hbuf_block *hbuf_b_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   135
  GList *hbuf_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   136
  GList *first_elt = g_list_first(*p_hbuf);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   137
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   138
  for (hbuf_elt = first_elt; hbuf_elt; hbuf_elt = g_list_next(hbuf_elt)) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   139
    hbuf_b_elt = (hbuf_block*)(hbuf_elt->data);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   140
    if (hbuf_b_elt->flags & HBB_FLAG_ALLOC) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   141
      g_free(hbuf_b_elt->ptr);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   142
    }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   143
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   144
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   145
  g_list_free(*p_hbuf);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   146
  *p_hbuf = NULL;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   147
}
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   148
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   149
//  hbuf_rebuild()
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   150
// Rebuild all hbuf list, with the new width.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   151
// If width == 0, lines are not wrapped.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   152
void hbuf_rebuild(GList **p_hbuf, unsigned int width)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   153
{
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   154
  GList *first_elt, *curr_elt, *next_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   155
  hbuf_block *hbuf_b_curr, *hbuf_b_next;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   156
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   157
  first_elt = g_list_first(*p_hbuf);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   158
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   159
  // #1 Remove non-persistent blocks (ptr_end should be updated!)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   160
  curr_elt = first_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   161
  while (curr_elt) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   162
    next_elt = g_list_next(curr_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   163
    // Last element?
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   164
    if (!next_elt)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   165
      break;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   166
    hbuf_b_curr = (hbuf_block*)(curr_elt->data);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   167
    hbuf_b_next = (hbuf_block*)(next_elt->data);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   168
    // Is next line not-persistent?
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   169
    if (!(hbuf_b_next->flags & HBB_FLAG_PERSISTENT)) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   170
      hbuf_b_curr->ptr_end = hbuf_b_next->ptr_end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   171
      g_list_delete_link(curr_elt, next_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   172
      next_elt = g_list_next(curr_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   173
    } else 
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   174
      curr_elt = next_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   175
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   176
  // #2 Go back to head and create non-persistent blocks when needed
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   177
  if (width) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   178
    char *line, *end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   179
    curr_elt = first_elt;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   180
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   181
    while (curr_elt) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   182
      hbuf_b_curr = (hbuf_block*)(curr_elt->data);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   183
      line = hbuf_b_curr->ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   184
      if (strlen(line) > width) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   185
        hbuf_block *hbuf_b_prev = hbuf_b_curr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   186
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   187
        // We need to break where we can find a space char
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   188
        char *br; // break pointer
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   189
        for (br = line + width; br > line && *br != 32 && *br != 9; br--)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   190
          ;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   191
        if (br <= line)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   192
          br = line + width;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   193
        else
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   194
          br++;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   195
        end = hbuf_b_curr->ptr_end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   196
        hbuf_b_curr->ptr_end = br;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   197
        // Create another block, non-persistent
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   198
        hbuf_b_curr = g_new0(hbuf_block, 1);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   199
        hbuf_b_curr->ptr      = hbuf_b_prev->ptr_end; // == br
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   200
        hbuf_b_curr->ptr_end  = end;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   201
        hbuf_b_curr->flags    = 0;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   202
        hbuf_b_curr->persist.ptr_end_alloc = hbuf_b_prev->persist.ptr_end_alloc;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   203
        /*
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   204
        // Is there a better way?
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   205
        if (g_list_next(curr_elt))
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   206
          g_list_insert_before(*p_hbuf, curr_elt->next, hbuf_b_curr);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   207
        else
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   208
          *p_hbuf = g_list_append(*p_hbuf, hbuf_b_curr);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   209
        */
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   210
        // This is OK because insert_before(NULL) <==> append()
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   211
        g_list_insert_before(*p_hbuf, curr_elt->next, hbuf_b_curr);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   212
      }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   213
      curr_elt = g_list_next(curr_elt);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   214
    }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   215
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   216
}
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   217
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   218
//  hbuf_get_lines(hbuf, n, where)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   219
// Returns an array of n pointers (for n lines from hbuf)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   220
// (The first line will be the line currently pointed by hbuf)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   221
// Note:The caller should free the array after use.
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   222
char **hbuf_get_lines(GList *hbuf, unsigned int n)
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   223
{
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   224
  unsigned int i;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   225
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   226
  char **array = g_new0(char*, n);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   227
  char **array_elt = array;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   228
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   229
  for (i=0 ; i < n ; i++) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   230
    if (hbuf) {
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   231
      hbuf_block *blk = (hbuf_block*)(hbuf->data);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   232
      int maxlen;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   233
      maxlen = blk->ptr_end - blk->ptr;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   234
      *array_elt++ = g_strndup(blk->ptr, maxlen);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   235
      hbuf = g_list_next(hbuf);
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   236
    } else
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   237
      *array_elt++ = NULL;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   238
  }
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   239
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   240
  return array;
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   241
}
1e9d4949bcfd [/trunk] Changeset 85 by mikael
mikael
parents:
diff changeset
   242