editbuf.h
上传用户:sichengcw
上传日期:2009-02-17
资源大小:202k
文件大小:7k
源码类别:

STL

开发平台:

Visual C++

  1. /* This is part of libio/iostream, providing -*- C++ -*- input/output.
  2. Copyright (C) 1993 Free Software Foundation
  3. This file is part of the GNU IO Library.  This library is free
  4. software; you can redistribute it and/or modify it under the
  5. terms of the GNU General Public License as published by the
  6. Free Software Foundation; either version 2, or (at your option)
  7. any later version.
  8. This library is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this library; see the file COPYING.  If not, write to the Free
  14. Software Foundation, 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
  15. As a special exception, if you link this library with files
  16. compiled with a GNU compiler to produce an executable, this does not cause
  17. the resulting executable to be covered by the GNU General Public License.
  18. This exception does not however invalidate any other reasons why
  19. the executable file might be covered by the GNU General Public License.
  20. Written by Per Bothner (bothner@cygnus.com). */
  21. #ifndef _EDITBUF_H
  22. #define _EDITBUF_H
  23. #ifdef __GNUG__
  24. #pragma interface
  25. #endif
  26. #include <stdio.h>
  27. #include <fstream.h>
  28. extern "C++" {
  29. typedef unsigned long mark_pointer;
  30. // At some point, it might be nice to parameterize this code
  31. // in terms of buf_char.
  32. typedef /*unsigned*/ char buf_char;
  33. // Logical pos from start of buffer (does not count gap).
  34. typedef long buf_index;
  35. // Pos from start of buffer, possibly including gap_size.
  36. typedef long buf_offset; 
  37. #if 0
  38. struct buf_cookie {
  39.     FILE *file;
  40.     struct edit_string *str;
  41.     struct buf_cookie *next;
  42.     buf_index tell();
  43. };
  44. #endif
  45. struct edit_buffer;
  46. struct edit_mark;
  47. // A edit_string is defined as the region between the 'start' and 'end' marks.
  48. // Normally (always?) 'start->insert_before()' should be false,
  49. // and 'end->insert_before()' should be true.
  50. struct edit_string {
  51.     struct edit_buffer *buffer; // buffer that 'start' and 'end' belong to
  52.     struct edit_mark *start, *end;
  53.     int length() const; // count of buf_chars currently in string
  54.     edit_string(struct edit_buffer *b,
  55.       struct edit_mark *ms, struct edit_mark *me)
  56. { buffer = b; start = ms; end = me; }
  57. /* Make a fresh, contiguous copy of the data in STR.
  58.    Assign length of STR to *LENP.
  59.    (Output has extra NUL at out[*LENP].) */
  60.     buf_char *copy_bytes(int *lenp) const;
  61. //    FILE *open_file(char *mode);
  62.     void assign(struct edit_string *src); // copy bytes from src to this
  63. };
  64. struct edit_streambuf : public streambuf {
  65.     friend edit_buffer;
  66.     edit_string *str;
  67.     edit_streambuf* next; // Chain of edit_streambuf's for a edit_buffer.
  68.     short _mode;
  69.     edit_streambuf(edit_string* bstr, int mode);
  70.     ~edit_streambuf();
  71.     virtual int underflow();
  72.     virtual int overflow(int c = EOF);
  73.     virtual streampos seekoff(streamoff, _seek_dir, int mode=ios::in|ios::out);
  74.     void flush_to_buffer();
  75.     void flush_to_buffer(edit_buffer* buffer);
  76.     int _inserting;
  77.     int inserting() { return _inserting; }
  78.     void inserting(int i) { _inserting = i; }
  79. //    int delete_chars(int count, char* cut_buf); Not implemented.
  80.     int truncate();
  81.     int is_reading() { return gptr() != NULL; }
  82.     buf_char* current() { return is_reading() ? gptr() : pptr(); }
  83.     void set_current(char *p, int is_reading);
  84.   protected:
  85.     void disconnect_gap_from_file(edit_buffer* buffer);
  86. };
  87. // A 'edit_mark' indicates a position in a buffer.
  88. // It is "attached" the text (rather than the offset).
  89. // There are two kinds of mark, which have different behavior
  90. // when text is inserted at the mark:
  91. // If 'insert_before()' is true the mark will be adjusted to be
  92. // *after* the new text.
  93. struct edit_mark {
  94.     struct edit_mark *chain;
  95.     mark_pointer _pos;
  96.     inline int insert_before() { return _pos & 1; }
  97.     inline unsigned long index_in_buffer(struct edit_buffer *)
  98. { return _pos >> 1; }
  99.     inline buf_char *ptr(struct edit_buffer *buf);
  100.     buf_index tell();
  101.     edit_mark() { }
  102.     edit_mark(struct edit_string *str, long delta);
  103.     edit_buffer *buffer();
  104.     ~edit_mark();
  105. };
  106. // A 'edit_buffer' consists of a sequence of buf_chars (the data),
  107. // a list of edit_marks pointing into the data, and a list of FILEs
  108. // also pointing into the data.
  109. // A 'edit_buffer' coerced to a edit_string is the string of
  110. // all the buf_chars in the buffer.
  111. // This implementation uses a conventional buffer gap (as in Emacs).
  112. // The gap start is defined by de-referencing a (buf_char**).
  113. // This is because sometimes a FILE is inserting into the buffer,
  114. // so rather than having each putc adjust the gap, we use indirection
  115. // to have the gap be defined as the write pointer of the FILE.
  116. // (This assumes that putc adjusts a pointer (as in GNU's libc), not an index.)
  117. struct edit_buffer {
  118.     buf_char *data; /* == emacs buffer_text.p1+1 */
  119.     buf_char *_gap_start;
  120.     edit_streambuf* _writer; // If non-NULL, currently writing stream
  121.     inline buf_char *gap_start()
  122. { return _writer ? _writer->pptr() : _gap_start; }
  123.     buf_offset __gap_end_pos; // size of part 1 + size of gap
  124.     /* int gap; implicit: buf_size - size1 - size2 */
  125.     int buf_size;
  126.     struct edit_streambuf *files;
  127.     struct edit_mark start_mark;
  128.     struct edit_mark end_mark;
  129.     edit_buffer();
  130.     inline buf_offset gap_end_pos() { return __gap_end_pos; }
  131.     inline struct edit_mark *start_marker() { return &start_mark; }
  132.     inline struct edit_mark *end_marker() { return &end_mark; }
  133. /* these should be protected, ultimately */
  134.     buf_index tell(edit_mark*);
  135.     buf_index tell(buf_char*);
  136.     inline buf_char *gap_end() { return data + gap_end_pos(); }
  137.     inline int gap_size() { return gap_end() - gap_start(); }
  138.     inline int size1() { return gap_start() - data; }
  139.     inline int size2() { return buf_size - gap_end_pos(); }
  140.     inline struct edit_mark * mark_list() { return &start_mark; }
  141.     void make_gap (buf_offset);
  142.     void move_gap (buf_offset pos);
  143.     void move_gap (buf_char *pos) { move_gap(pos - data); }
  144.     void gap_left (int pos);
  145.     void gap_right (int pos);
  146.     void adjust_markers(mark_pointer low, mark_pointer high,
  147. int amount, buf_char *old_data);
  148.     void delete_range(buf_index from, buf_index to);
  149.     void delete_range(struct edit_mark *start, struct edit_mark *end);
  150. };
  151. extern buf_char * bstr_copy(struct edit_string *str, int *lenp);
  152. // Convert a edit_mark to a (buf_char*)
  153. inline buf_char *edit_mark::ptr(struct edit_buffer *buf)
  154. { return buf->data + index_in_buffer(buf); }
  155. inline void edit_streambuf::flush_to_buffer()
  156. {
  157.     edit_buffer* buffer = str->buffer;
  158.     if (buffer->_writer == this) flush_to_buffer(buffer);
  159. }
  160. } // extern "C++"
  161. #endif /* !_EDITBUF_H*/