/[LeafOK_CVS]/lbbs/src/editor.c
ViewVC logotype

Diff of /lbbs/src/editor.c

Parent Directory Parent Directory | Revision Log Revision Log | View Patch Patch

Revision 1.17 by sysadm, Thu Jun 12 12:53:49 2025 UTC Revision 1.23 by sysadm, Sun Jun 15 13:40:52 2025 UTC
# Line 79  EDITOR_DATA *editor_data_load(const char Line 79  EDITOR_DATA *editor_data_load(const char
79  {  {
80          EDITOR_DATA *p_editor_data;          EDITOR_DATA *p_editor_data;
81          char *p_data_line = NULL;          char *p_data_line = NULL;
82          long line_offsets[MAX_EDITOR_DATA_LINES];          long line_offsets[MAX_EDITOR_DATA_LINES + 1];
83          long current_data_line_length = 0;          long current_data_line_length = 0;
84          long i;          long i;
85    
# Line 96  EDITOR_DATA *editor_data_load(const char Line 96  EDITOR_DATA *editor_data_load(const char
96                  return NULL;                  return NULL;
97          }          }
98    
99          p_editor_data->display_line_total = split_data_lines(p_data, SCREEN_COLS, line_offsets, MAX_EDITOR_DATA_LINES);          p_editor_data->display_line_total = split_data_lines(p_data, SCREEN_COLS, line_offsets, MAX_EDITOR_DATA_LINES + 1);
100    
101          for (i = 0; i < p_editor_data->display_line_total; i++)          for (i = 0; i < p_editor_data->display_line_total; i++)
102          {          {
# Line 126  EDITOR_DATA *editor_data_load(const char Line 126  EDITOR_DATA *editor_data_load(const char
126    
127                  memcpy(p_editor_data->p_display_lines[i], p_data + line_offsets[i], (size_t)p_editor_data->display_line_lengths[i]);                  memcpy(p_editor_data->p_display_lines[i], p_data + line_offsets[i], (size_t)p_editor_data->display_line_lengths[i]);
128                  current_data_line_length += p_editor_data->display_line_lengths[i];                  current_data_line_length += p_editor_data->display_line_lengths[i];
129    
130                    // Trim \n from last line
131                    if (i + 1 == p_editor_data->display_line_total &&
132                            p_editor_data->display_line_lengths[i] > 0 &&
133                            p_editor_data->p_display_lines[i][p_editor_data->display_line_lengths[i] - 1] == '\n')
134                    {
135                            p_editor_data->display_line_lengths[i]--;
136                            current_data_line_length--;
137                    }
138                  p_data_line[current_data_line_length] = '\0';                  p_data_line[current_data_line_length] = '\0';
139          }          }
140    
# Line 218  int editor_data_insert(EDITOR_DATA *p_ed Line 227  int editor_data_insert(EDITOR_DATA *p_ed
227                  return -1;                  return -1;
228          }          }
229    
230            // Validate str
231            if ((str_len == 1 && str[0] <= 0) ||
232                    (str_len == 2 && (str[0] >= 0 || str[1] >= 0)))
233            {
234                    log_error("Invalid input str, len=%d\n", str_len);
235                    return -2;
236            }
237    
238          // Get accurate offset of first character of CJK at offset position          // Get accurate offset of first character of CJK at offset position
239          for (i = 0; i < offset; i++)          for (i = 0; i < offset; i++)
240          {          {
241                  if (p_editor_data->p_display_lines[display_line][i] < 0 || p_editor_data->p_display_lines[display_line][i] > 127) // GBK                  if (p_editor_data->p_display_lines[display_line][i] < 0) // GBK
242                  {                  {
243                          i++;                          i++;
244                  }                  }
# Line 395  int editor_data_insert(EDITOR_DATA *p_ed Line 412  int editor_data_insert(EDITOR_DATA *p_ed
412    
413          if (*p_offset >= p_editor_data->display_line_lengths[*p_display_line])          if (*p_offset >= p_editor_data->display_line_lengths[*p_display_line])
414          {          {
415                  *p_offset -= p_editor_data->display_line_lengths[*p_display_line];                  if (*p_display_line + 1 < p_editor_data->display_line_total)
   
                 if (*p_display_line + 1 >= p_editor_data->display_line_total)  
416                  {                  {
417                          log_error("*p_display_line(%d) >= display_line_total(%d)\n", *p_display_line, p_editor_data->display_line_total);                          *p_offset -= p_editor_data->display_line_lengths[*p_display_line];
418                            (*p_display_line)++;
419                  }                  }
420                  else          }
421    
422            // Prevent the last display line from being over-length
423            if (p_editor_data->display_line_total == MAX_EDITOR_DATA_LINES)
424            {
425                    len = split_line(p_editor_data->p_display_lines[p_editor_data->display_line_total - 1], SCREEN_COLS - 1, &eol, &display_len);
426                    p_editor_data->p_display_lines[p_editor_data->display_line_total - 1][len] = '\0';
427                    p_editor_data->display_line_lengths[p_editor_data->display_line_total - 1] = len;
428                    if (*p_display_line + 1 >= p_editor_data->display_line_total)
429                  {                  {
430                          (*p_display_line)++;                          *p_offset = MIN(*p_offset, len);
431                            *p_display_line = p_editor_data->display_line_total - 1;
432                  }                  }
433          }          }
434    
435          return 0;          return 0;
436  }  }
437    
438  int editor_data_delete(EDITOR_DATA *p_editor_data, long display_line, long offset,  int editor_data_delete(EDITOR_DATA *p_editor_data, long *p_display_line, long *p_offset,
439                                             long *p_last_updated_line)                                             long *p_last_updated_line)
440  {  {
441            long display_line = *p_display_line;
442            long offset = *p_offset;
443          char *p_data_line = NULL;          char *p_data_line = NULL;
444          long len_data_line;          long len_data_line;
445          long offset_data_line;          long offset_data_line;
# Line 431  int editor_data_delete(EDITOR_DATA *p_ed Line 458  int editor_data_delete(EDITOR_DATA *p_ed
458          // Get accurate offset of first character of CJK at offset position          // Get accurate offset of first character of CJK at offset position
459          for (i = 0; i < offset; i++)          for (i = 0; i < offset; i++)
460          {          {
461                  if (p_editor_data->p_display_lines[display_line][i] < 0 || p_editor_data->p_display_lines[display_line][i] > 127) // GBK                  if (p_editor_data->p_display_lines[display_line][i] < 0) // GBK
462                  {                  {
463                          i++;                          i++;
464                  }                  }
# Line 469  int editor_data_delete(EDITOR_DATA *p_ed Line 496  int editor_data_delete(EDITOR_DATA *p_ed
496                  }                  }
497          }          }
498    
499            if (offset_data_line >= len_data_line) // end-of-line
500            {
501                    return 0;
502            }
503    
504          // Check str to be deleted          // Check str to be deleted
505          if (p_data_line[offset_data_line] > 0 && p_data_line[offset_data_line] < 127)          if (p_data_line[offset_data_line] > 0 && p_data_line[offset_data_line] < 127)
506          {          {
507                  str_len = 1;                  str_len = 1;
508          }          }
509          else if (p_data_line[offset_data_line + 1] < 0 || p_data_line[offset_data_line] > 127) // GBK          else if (p_data_line[offset_data_line + 1] < 0) // GBK
510          {          {
511                  str_len = 2;                  str_len = 2;
512          }          }
513          else          else
514          {          {
515                  log_error("Some strange character at display_line %ld, offset %ld: %d %d %d %d\n",                  log_error("Some strange character at display_line %ld, offset %ld: %d %d\n",
516                                    display_line, offset, p_data_line[offset_data_line], p_data_line[offset_data_line + 1],                                    display_line, offset, p_data_line[offset_data_line], p_data_line[offset_data_line + 1]);
                                   p_data_line[offset_data_line + 2], p_data_line[offset_data_line + 3]);  
517                  str_len = 1;                  str_len = 1;
518          }          }
519    
# Line 552  int editor_data_delete(EDITOR_DATA *p_ed Line 583  int editor_data_delete(EDITOR_DATA *p_ed
583    
584          *p_last_updated_line = display_line + MIN(i, split_line_total - 1);          *p_last_updated_line = display_line + MIN(i, split_line_total - 1);
585    
586          if (display_line + i < last_display_line)          if (*p_last_updated_line < last_display_line)
587          {          {
588                  // Remove redundant display line after last_display_line                  // Remove redundant display line after last_display_line
589                  for (j = last_display_line + 1; j < p_editor_data->display_line_total; j++)                  for (j = last_display_line + 1; j < p_editor_data->display_line_total; j++)
590                  {                  {
591                          p_editor_data->p_display_lines[j - (last_display_line - (display_line + i))] = p_editor_data->p_display_lines[j];                          p_editor_data->p_display_lines[j - (last_display_line - *p_last_updated_line)] = p_editor_data->p_display_lines[j];
592                          p_editor_data->display_line_lengths[j - (last_display_line - (display_line + i))] = p_editor_data->display_line_lengths[j];                          p_editor_data->display_line_lengths[j - (last_display_line - *p_last_updated_line)] = p_editor_data->display_line_lengths[j];
593                  }                  }
594    
595                  (p_editor_data->display_line_total) -= (last_display_line - (display_line + i));                  j = p_editor_data->display_line_total;
596                  last_display_line = display_line + i;                  (p_editor_data->display_line_total) -= (last_display_line - *p_last_updated_line);
597                    *p_last_updated_line = MAX(j - 1, *p_last_updated_line);
                 *p_last_updated_line = p_editor_data->display_line_total - 1;  
598          }          }
599    
600            // Return real offset
601            *p_offset = offset;
602    
603          return str_len;          return str_len;
604  }  }
605    
# Line 664  int editor_display(EDITOR_DATA *p_editor Line 697  int editor_display(EDITOR_DATA *p_editor
697                                          input_str[str_len] = (char)(ch - 256);                                          input_str[str_len] = (char)(ch - 256);
698                                          str_len++;                                          str_len++;
699                                  }                                  }
700                                  else                                  else if (str_len > 0)
701                                  {                                  {
702                                            log_error("Received %d character over 127 followed by character less than 127\n", str_len);
703                                          str_len = 0;                                          str_len = 0;
704                                  }                                  }
705    
706                                  if ((ch >= 32 && ch < 127) || (ch > 127 && ch <= 255 && str_len == 2) || // Printable character or GBK                                  if ((ch >= 32 && ch < 127) || (ch > 127 && ch <= 255 && str_len == 2) || // Printable character or GBK
707                                          ch == CR || ch == KEY_ESC)                                                                                       // Special character                                          ch == CR || ch == KEY_ESC)                                                                                       // Special character
708                                  {                                  {
709                                          if (str_len == 0)                                          if (str_len == 0) // ch >= 32 && ch < 127
710                                          {                                          {
711                                                  input_str[0] = (char)ch;                                                  input_str[0] = (char)ch;
712                                                  str_len = 1;                                                  str_len = 1;
# Line 686  int editor_display(EDITOR_DATA *p_editor Line 720  int editor_display(EDITOR_DATA *p_editor
720    
721                                          if (!key_insert) // overwrite                                          if (!key_insert) // overwrite
722                                          {                                          {
723                                                  if (editor_data_delete(p_editor_data, display_line_in, offset_in,                                                  if (editor_data_delete(p_editor_data, &display_line_out, &offset_out,
724                                                                                             &last_updated_line) < 0)                                                                                             &last_updated_line) < 0)
725                                                  {                                                  {
726                                                          log_error("editor_data_delete() error\n");                                                          log_error("editor_data_delete() error\n");
# Line 697  int editor_display(EDITOR_DATA *p_editor Line 731  int editor_display(EDITOR_DATA *p_editor
731                                                                                     input_str, str_len, &last_updated_line) < 0)                                                                                     input_str, str_len, &last_updated_line) < 0)
732                                          {                                          {
733                                                  log_error("editor_data_insert(str_len=%d) error\n", str_len);                                                  log_error("editor_data_insert(str_len=%d) error\n", str_len);
                                                 str_len = 0;  
734                                          }                                          }
735                                          else                                          else
736                                          {                                          {
                                                 str_len = 0;  
   
737                                                  output_end_row = MIN(SCREEN_ROWS - 1, output_current_row + (int)(last_updated_line - line_current));                                                  output_end_row = MIN(SCREEN_ROWS - 1, output_current_row + (int)(last_updated_line - line_current));
738                                                  line_current -= (output_current_row - row_pos);                                                  line_current -= (output_current_row - row_pos);
739                                                  output_current_row = (int)row_pos;                                                  output_current_row = (int)row_pos;
# Line 730  int editor_display(EDITOR_DATA *p_editor Line 761  int editor_display(EDITOR_DATA *p_editor
761                                                  {                                                  {
762                                                          row_pos += (display_line_out - display_line_in);                                                          row_pos += (display_line_out - display_line_in);
763                                                  }                                                  }
764                                                  col_pos = offset_out + 1;                                                  col_pos = offset_out + 1; // Set col_pos to accurate pos
765                                          }                                          }
766    
767                                            str_len = 0;
768                                          continue;                                          continue;
769                                  }                                  }
770                                  else if (ch == KEY_DEL || ch == BACKSPACE) // Del                                  else if (ch == KEY_DEL || ch == BACKSPACE) // Del
771                                  {                                  {
772                                          if (ch == BACKSPACE)                                          if (ch == BACKSPACE)
773                                          {                                          {
774                                                    if (line_current - output_current_row + row_pos <= 0 && col_pos <= 1) // Forbidden
775                                                    {
776                                                            input_ok = 0;
777                                                            continue;
778                                                    }
779    
780                                                  col_pos--;                                                  col_pos--;
781                                                    if (col_pos > 1 &&
782                                                            p_editor_data->p_display_lines[line_current - output_current_row + row_pos][col_pos - 1] < 0) // GBK
783                                                    {
784                                                            col_pos--;
785                                                    }
786    
787                                                  if (col_pos < 1 && line_current - output_current_row + row_pos >= 0)                                                  if (col_pos < 1 && line_current - output_current_row + row_pos >= 0)
788                                                  {                                                  {
789                                                          row_pos--;                                                          row_pos--;
# Line 747  int editor_display(EDITOR_DATA *p_editor Line 791  int editor_display(EDITOR_DATA *p_editor
791                                                  }                                                  }
792                                          }                                          }
793    
794                                          if ((str_len = editor_data_delete(p_editor_data, line_current - output_current_row + row_pos, col_pos - 1,                                          display_line_in = line_current - output_current_row + row_pos;
795                                            offset_in = col_pos - 1;
796                                            display_line_out = display_line_in;
797                                            offset_out = offset_in;
798    
799                                            if ((str_len = editor_data_delete(p_editor_data, &display_line_out, &offset_out,
800                                                                                                            &last_updated_line)) < 0)                                                                                                            &last_updated_line)) < 0)
801                                          {                                          {
802                                                  log_error("editor_data_delete() error\n");                                                  log_error("editor_data_delete() error\n");
803                                          }                                          }
804                                          else                                          else
805                                          {                                          {
806                                                  if (ch == BACKSPACE)                                                  col_pos = offset_out + 1; // Set col_pos to accurate pos
                                                 {  
                                                         for (i = 1; i < str_len; i++)  
                                                         {  
                                                                 col_pos--;  
                                                                 if (col_pos < 1 && line_current - output_current_row + row_pos >= 0)  
                                                                 {  
                                                                         row_pos--;  
                                                                         col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);  
                                                                 }  
                                                         }  
                                                 }  
807    
808                                                  output_end_row = MIN(SCREEN_ROWS - 1, output_current_row + (int)(last_updated_line - line_current));                                                  output_end_row = MIN(SCREEN_ROWS - 1, output_current_row + (int)(last_updated_line - line_current));
809                                                  line_current -= (output_current_row - row_pos);                                                  line_current -= (output_current_row - row_pos);
# Line 789  int editor_display(EDITOR_DATA *p_editor Line 827  int editor_display(EDITOR_DATA *p_editor
827                                                          row_pos += scroll_rows;                                                          row_pos += scroll_rows;
828                                                          output_current_row = screen_begin_row;                                                          output_current_row = screen_begin_row;
829                                                          output_end_row = SCREEN_ROWS - 1;                                                          output_end_row = SCREEN_ROWS - 1;
                                                         clrline(output_current_row, SCREEN_ROWS);  
830                                                  }                                                  }
831    
832                                                    clrline(output_current_row, output_end_row);
833                                          }                                          }
834    
835                                            str_len = 0;
836                                          continue;                                          continue;
837                                  }                                  }
838    
# Line 810  int editor_display(EDITOR_DATA *p_editor Line 850  int editor_display(EDITOR_DATA *p_editor
850                                          break;                                          break;
851                                  case Ctrl('E'): // End of line                                  case Ctrl('E'): // End of line
852                                  case KEY_CTRL_RIGHT:                                  case KEY_CTRL_RIGHT:
853                                            if (line_current - output_current_row + row_pos == p_editor_data->display_line_total - 1) // row_pos at end line
854                                            {
855                                                    // last display line does NOT have \n in the end
856                                                    col_pos = p_editor_data->display_line_lengths[line_current - output_current_row + row_pos] + 1;
857                                                    break;
858                                            }
859                                          col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);                                          col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);
860                                          break;                                          break;
861                                  case Ctrl('T'): // Top of screen                                  case Ctrl('T'): // Top of screen
# Line 819  int editor_display(EDITOR_DATA *p_editor Line 865  int editor_display(EDITOR_DATA *p_editor
865                                          break;                                          break;
866                                  case Ctrl('B'): // Bottom of screen                                  case Ctrl('B'): // Bottom of screen
867                                  case KEY_CTRL_DOWN:                                  case KEY_CTRL_DOWN:
868                                          row_pos = SCREEN_ROWS - 1;                                          if (p_editor_data->display_line_total < screen_row_total)
869                                          col_pos = MIN(col_pos, MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]));                                          {
870                                                    row_pos = p_editor_data->display_line_total;
871                                            }
872                                            else
873                                            {
874                                                    row_pos = SCREEN_ROWS - 1;
875                                            }
876                                            if (line_current + (screen_row_total - (output_current_row - screen_begin_row)) >= p_editor_data->display_line_total) // Reach end
877                                            {
878                                                    // last display line does NOT have \n in the end
879                                                    col_pos = MIN(col_pos, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos] + 1);
880                                            }
881                                            else
882                                            {
883                                                    col_pos = MIN(col_pos, MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]));
884                                            }
885                                          break;                                          break;
886                                  case KEY_INS:                                  case KEY_INS:
887                                          key_insert = !key_insert;                                          key_insert = !key_insert;
# Line 841  int editor_display(EDITOR_DATA *p_editor Line 902  int editor_display(EDITOR_DATA *p_editor
902                                          if (p_editor_data->display_line_total < screen_row_total)                                          if (p_editor_data->display_line_total < screen_row_total)
903                                          {                                          {
904                                                  row_pos = p_editor_data->display_line_total;                                                  row_pos = p_editor_data->display_line_total;
905                                                  col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);                                                  col_pos = p_editor_data->display_line_lengths[line_current - output_current_row + row_pos] + 1;
906                                                  break;                                                  break;
907                                          }                                          }
908                                          line_current = p_editor_data->display_line_total - screen_row_total;                                          line_current = p_editor_data->display_line_total - screen_row_total;
909                                          output_current_row = screen_begin_row;                                          output_current_row = screen_begin_row;
910                                          output_end_row = SCREEN_ROWS - 1;                                          output_end_row = SCREEN_ROWS - 1;
911                                          row_pos = SCREEN_ROWS - 1;                                          row_pos = SCREEN_ROWS - 1;
912                                          col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);                                          col_pos = p_editor_data->display_line_lengths[line_current - output_current_row + row_pos] + 1;
913                                          clrline(output_current_row, SCREEN_ROWS);                                          clrline(output_current_row, SCREEN_ROWS);
914                                          break;                                          break;
915                                  case KEY_LEFT:                                  case KEY_LEFT:
916                                          if (col_pos > 1)                                          if (col_pos > 1)
917                                          {                                          {
918                                                  col_pos--;                                                  col_pos--;
919                                                    if (col_pos > 1 &&
920                                                            p_editor_data->p_display_lines[line_current - output_current_row + row_pos][col_pos - 1] < 0 &&
921                                                            p_editor_data->p_display_lines[line_current - output_current_row + row_pos][col_pos - 2] < 0) // GBK
922                                                    {
923                                                            col_pos--;
924                                                    }
925                                                  break;                                                  break;
926                                          }                                          }
927                                          col_pos = SCREEN_COLS; // continue to KEY_UP                                          col_pos = SCREEN_COLS; // continue to KEY_UP
# Line 882  int editor_display(EDITOR_DATA *p_editor Line 949  int editor_display(EDITOR_DATA *p_editor
949                                  case KEY_RIGHT:                                  case KEY_RIGHT:
950                                          if (col_pos < p_editor_data->display_line_lengths[line_current - output_current_row + row_pos])                                          if (col_pos < p_editor_data->display_line_lengths[line_current - output_current_row + row_pos])
951                                          {                                          {
952                                                    if (p_editor_data->p_display_lines[line_current - output_current_row + row_pos][col_pos - 1] < 0 &&
953                                                            p_editor_data->p_display_lines[line_current - output_current_row + row_pos][col_pos] < 0) // GBK
954                                                    {
955                                                            col_pos++;
956                                                    }
957                                                  col_pos++;                                                  col_pos++;
958                                                  break;                                                  break;
959                                          }                                          }
# Line 893  int editor_display(EDITOR_DATA *p_editor Line 965  int editor_display(EDITOR_DATA *p_editor
965                                                  col_pos = MIN(col_pos, MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]));                                                  col_pos = MIN(col_pos, MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]));
966                                                  break;                                                  break;
967                                          }                                          }
968                                          if (line_current + (screen_row_total - (output_current_row - screen_begin_row)) >= p_editor_data->display_line_total) // Reach end                                          if (line_current - output_current_row + row_pos == p_editor_data->display_line_total - 1) // row_pos at end line
969                                          {                                          {
970                                                  col_pos = MAX(1, p_editor_data->display_line_lengths[line_current - output_current_row + row_pos]);                                                  // last display line does NOT have \n in the end
971                                                    col_pos = p_editor_data->display_line_lengths[line_current - output_current_row + row_pos] + 1;
972                                                  break;                                                  break;
973                                          }                                          }
974                                          line_current += (screen_row_total - (output_current_row - screen_begin_row));                                          line_current += (screen_row_total - (output_current_row - screen_begin_row));


Legend:
Removed lines/characters  
Changed lines/characters
  Added lines/characters

webmaster@leafok.com
ViewVC Help
Powered by ViewVC 1.3.0-beta1