Added support for cell comments in constant_memory mode.

Issue #38
This commit is contained in:
John McNamara 2020-01-10 00:01:11 +00:00
parent 73a47db741
commit ad22c8ad72
5 changed files with 99 additions and 51 deletions

View File

@ -61,7 +61,7 @@ CFLAGS += -DUSE_DOUBLE_FUNCTION
endif endif
# Flags passed to compiler. # Flags passed to compiler.
CFLAGS += -g -O3 -Wall -Wextra -pedantic -ansi CFLAGS += -g -O0 -Wall -Wextra -pedantic -ansi
# Fix for modified zconf.h on Gentoo. # Fix for modified zconf.h on Gentoo.
ifneq (,$(findstring gentoo, $(shell uname -sr))) ifneq (,$(findstring gentoo, $(shell uname -sr)))

View File

@ -795,12 +795,26 @@ _insert_cell_list(struct lxw_table_cells *cell_list,
/* If existing_cell is not NULL, then that cell already existed. */ /* If existing_cell is not NULL, then that cell already existed. */
/* Remove existing_cell and add new one in again. */ /* Remove existing_cell and add new one in again. */
if (existing_cell) { if (existing_cell) {
existing_cell->comment = NULL;
RB_REMOVE(lxw_table_cells, cell_list, existing_cell);
/* Add it in again. */ if (cell->comment) {
RB_INSERT(lxw_table_cells, cell_list, cell); existing_cell->comment = cell->comment;
_free_cell(existing_cell); cell->comment = NULL;
_free_cell(cell);
}
else {
/* Transfer existing cell comment to new cell. */
if (existing_cell->comment) {
cell->comment = existing_cell->comment;
existing_cell->comment = NULL;
}
RB_REMOVE(lxw_table_cells, cell_list, existing_cell);
/* Add it in again. */
RB_INSERT(lxw_table_cells, cell_list, cell);
_free_cell(existing_cell);
}
} }
return; return;
@ -814,7 +828,6 @@ _insert_cell(lxw_worksheet *self, lxw_row_t row_num, lxw_col_t col_num,
lxw_cell *cell) lxw_cell *cell)
{ {
lxw_row *row = _get_row(self, row_num); lxw_row *row = _get_row(self, row_num);
lxw_vml_obj *existing_comment = NULL;
if (!self->optimize) { if (!self->optimize) {
row->data_changed = LXW_TRUE; row->data_changed = LXW_TRUE;
@ -824,15 +837,28 @@ _insert_cell(lxw_worksheet *self, lxw_row_t row_num, lxw_col_t col_num,
if (row) { if (row) {
row->data_changed = LXW_TRUE; row->data_changed = LXW_TRUE;
/* Overwrite an existing cell if necessary. */ /* Overwrite an existing cell or replace comment. */
if (self->array[col_num]) { if (self->array[col_num]) {
existing_comment = self->array[col_num]->comment;
self->array[col_num]->comment = NULL;
_free_cell(self->array[col_num]);
}
self->array[col_num] = cell; if (cell->comment) {
self->array[col_num]->comment = existing_comment; self->array[col_num]->comment = cell->comment;
cell->comment = NULL;
_free_cell(cell);
}
else {
/* Transfer existing cell comment to new cell. */
if (self->array[col_num]->comment) {
cell->comment = self->array[col_num]->comment;
self->array[col_num]->comment = NULL;
}
_free_cell(self->array[col_num]);
self->array[col_num] = cell;
}
}
else {
self->array[col_num] = cell;
}
} }
} }
} }
@ -2558,19 +2584,30 @@ lxw_worksheet_prepare_vml_objects(lxw_worksheet *self,
size_t data_str_len = 0; size_t data_str_len = 0;
size_t used = 0; size_t used = 0;
char *vml_data_id_str; char *vml_data_id_str;
lxw_vml_obj *comment_obj;
RB_FOREACH(row, lxw_table_rows, self->table) { if (self->optimize) {
STAILQ_FOREACH(comment_obj, self->comment_objs, list_pointers) {
/* Calculate the worksheet position of the comment. */
_worksheet_position_vml_object(self, comment_obj);
if (row->has_comments) { comment_count++;
RB_FOREACH(cell, lxw_table_cells, row->cells) { }
if (cell->comment) { }
/* Calculate the worksheet position of the comment. */ else {
_worksheet_position_vml_object(self, cell->comment); RB_FOREACH(row, lxw_table_rows, self->table) {
/* Store comment in a simple list for use by packager. */ if (row->has_comments) {
STAILQ_INSERT_TAIL(self->comment_objs, cell->comment, RB_FOREACH(cell, lxw_table_cells, row->cells) {
list_pointers); if (cell->comment) {
comment_count++; /* Calculate the worksheet position of the comment. */
_worksheet_position_vml_object(self, cell->comment);
/* Store comment in a list for use by packager. */
STAILQ_INSERT_TAIL(self->comment_objs, cell->comment,
list_pointers);
comment_count++;
}
} }
} }
} }
@ -5086,13 +5123,6 @@ worksheet_write_comment_opt(lxw_worksheet *self,
lxw_vml_obj *comment; lxw_vml_obj *comment;
uint8_t data_changed = LXW_FALSE; uint8_t data_changed = LXW_FALSE;
if (self->optimize) {
LXW_WARN("worksheet_write_comment/opt(): "
"Not supported in 'constant_memory' mode.");
return LXW_ERROR_FEATURE_NOT_SUPPORTED;
}
err = _check_dimensions(self, row_num, col_num, LXW_FALSE, LXW_FALSE); err = _check_dimensions(self, row_num, col_num, LXW_FALSE, LXW_FALSE);
if (err) if (err)
return err; return err;
@ -5109,40 +5139,30 @@ worksheet_write_comment_opt(lxw_worksheet *self,
comment->text = lxw_strdup(text); comment->text = lxw_strdup(text);
GOTO_LABEL_ON_MEM_ERROR(comment->text, mem_error); GOTO_LABEL_ON_MEM_ERROR(comment->text, mem_error);
row = lxw_worksheet_find_row(self, row_num); row = _get_row(self, row_num);
if (row) if (row)
data_changed = row->data_changed; data_changed = row->data_changed;
cell = lxw_worksheet_find_cell_in_row(row, col_num); /* Set user and default parameters for the comment. */
if (cell) {
free(cell->comment);
}
else {
/* If there isn't an existing cell we use a new blank cell. */
cell = _new_blank_cell(row_num, col_num, NULL);
_insert_cell(self, row_num, col_num, cell);
}
comment->row = row_num; comment->row = row_num;
comment->col = col_num; comment->col = col_num;
/* Set user and default parameters for the comment. */
_get_comment_params(comment, options); _get_comment_params(comment, options);
cell = _new_blank_cell(row_num, col_num, NULL);
cell->comment = comment; cell->comment = comment;
_insert_cell(self, row_num, col_num, cell);
if (!row) { row = _get_row(self, row_num);
row = lxw_worksheet_find_row(self, row_num); row->data_changed = data_changed;
row->data_changed = LXW_FALSE;
}
else {
row->data_changed = data_changed;
}
row->has_comments = LXW_TRUE; row->has_comments = LXW_TRUE;
self->has_vml = LXW_TRUE; self->has_vml = LXW_TRUE;
self->has_comments = LXW_TRUE; self->has_comments = LXW_TRUE;
/* Store comment in a simple list for use by packager. */
if (self->optimize)
STAILQ_INSERT_TAIL(self->comment_objs, comment, list_pointers);
return LXW_NO_ERROR; return LXW_NO_ERROR;
mem_error: mem_error:

View File

@ -0,0 +1,25 @@
/*****************************************************************************
* Test cases for libxlsxwriter.
*
* Test to compare output against Excel files.
*
* Copyright 2014-2019, John McNamara, jmcnamara@cpan.org
*
*/
#include "xlsxwriter.h"
int main() {
lxw_workbook_options options = {.constant_memory = LXW_TRUE};
lxw_workbook *workbook = workbook_new_opt("test_optimize13.xlsx", &options);
lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL);
worksheet_write_string(worksheet, CELL("A1"), "Foo", NULL);
worksheet_write_comment(worksheet, CELL("B2"), "Some text");
worksheet_set_comments_author(worksheet, "John");
return workbook_close(workbook);
}

View File

@ -33,6 +33,9 @@ class TestCompareXLSXFiles(base_test_class.XLSXBaseTest):
# Skip some of the XlsxWriter tests until the required functionality is ported. # Skip some of the XlsxWriter tests until the required functionality is ported.
def test_optimize13(self):
self.run_exe_test('test_optimize13')
def test_optimize21(self): def test_optimize21(self):
self.run_exe_test('test_optimize21') self.run_exe_test('test_optimize21')

Binary file not shown.