diff --git a/dev/release/fix_example_docs.pl b/dev/release/fix_example_docs.pl index d03baf80..3b5cd241 100644 --- a/dev/release/fix_example_docs.pl +++ b/dev/release/fix_example_docs.pl @@ -39,6 +39,7 @@ my @examples = ( [ 'defined_name.c', 'Example of how to create defined names' ], [ 'outline.c', 'Example of grouping and outlines' ], [ 'outline_collapsed.c', 'Example of grouping and collapsed outlines' ], + [ 'background.c', 'Example of how to set the background image for a worksheet' ], [ 'tab_colors.c', 'Example of how to set worksheet tab colors' ], [ 'diagonal_border.c', 'Example of how to set a worksheet cell diagonal border.' ], [ 'hide_sheet.c', 'Example of hiding a worksheet' ], diff --git a/docs/images/background.png b/docs/images/background.png new file mode 100644 index 00000000..fc28c654 Binary files /dev/null and b/docs/images/background.png differ diff --git a/docs/src/examples.dox b/docs/src/examples.dox index d1307706..77e64b38 100644 --- a/docs/src/examples.dox +++ b/docs/src/examples.dox @@ -456,7 +456,7 @@ Example of how to generate Excel outlines and grouping. - +
@ref outline.c "<< outline.c"@ref tab_colors.c "tab_colors.c >>"@ref background.c "background.c >>"
@@ -468,11 +468,27 @@ mainly on collapsed outlines. -@example tab_colors.c +@example background.c + + +
@ref outline_collapsed.c "<< outline_collapsed.c"@ref tab_colors.c "tab_colors.c >>"
+ +Example of how to set the background image for a worksheet. + +@image html background.png + + + + +@example tab_colors.c + + + +
@ref background.c "<< background.c" @ref diagonal_border.c "diagonal_border.c >>"
diff --git a/docs/src/examples.txt b/docs/src/examples.txt index 0acf93fa..aac04418 100644 --- a/docs/src/examples.txt +++ b/docs/src/examples.txt @@ -213,6 +213,13 @@ mainly on collapsed outlines. @image html outline2.png +############################################################## +@example background.c + +Example of how to set the background image for a worksheet. + +@image html background.png + ############################################################## @example tab_colors.c diff --git a/examples/background.c b/examples/background.c new file mode 100644 index 00000000..b3ee8dcb --- /dev/null +++ b/examples/background.c @@ -0,0 +1,20 @@ +/* + * An example of setting a worksheet background image with libxlsxwriter. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("background.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_set_background(worksheet, "logo.png"); + + workbook_close(workbook); + + return 0; +} diff --git a/include/xlsxwriter/workbook.h b/include/xlsxwriter/workbook.h index 1a605b9f..83ff58bc 100644 --- a/include/xlsxwriter/workbook.h +++ b/include/xlsxwriter/workbook.h @@ -287,6 +287,7 @@ typedef struct lxw_workbook { struct lxw_chartsheet_names *chartsheet_names; struct lxw_image_md5s *image_md5s; struct lxw_image_md5s *header_image_md5s; + struct lxw_image_md5s *background_md5s; struct lxw_charts *charts; struct lxw_charts *ordered_charts; struct lxw_formats *formats; diff --git a/include/xlsxwriter/worksheet.h b/include/xlsxwriter/worksheet.h index e6d26258..9532acb8 100644 --- a/include/xlsxwriter/worksheet.h +++ b/include/xlsxwriter/worksheet.h @@ -1360,6 +1360,7 @@ typedef struct lxw_object_properties { double y_dpi; lxw_chart *chart; uint8_t is_duplicate; + uint8_t is_background; char *md5; char *image_position; uint8_t decorative; @@ -1752,9 +1753,11 @@ typedef struct lxw_worksheet { uint8_t has_vml; uint8_t has_comments; uint8_t has_header_vml; + uint8_t has_background_image; lxw_rel_tuple *external_vml_comment_link; lxw_rel_tuple *external_comment_link; lxw_rel_tuple *external_vml_header_link; + lxw_rel_tuple *external_background_link; char *comment_author; char *vml_data_id_str; char *vml_header_id_str; @@ -1784,6 +1787,7 @@ typedef struct lxw_worksheet { lxw_object_properties *footer_left_object_props; lxw_object_properties *footer_center_object_props; lxw_object_properties *footer_right_object_props; + lxw_object_properties *background_image; STAILQ_ENTRY (lxw_worksheet) list_pointers; @@ -3148,6 +3152,70 @@ lxw_error worksheet_insert_image_buffer_opt(lxw_worksheet *worksheet, size_t image_size, lxw_image_options *options); +/** + * @brief Set the background image for a worksheet. + * + * @param worksheet Pointer to a lxw_worksheet instance to be updated. + * @param filename The image filename, with path if required. + * + * @return A #lxw_error code. + * + * The `%worksheet_set_background()` function can be used to set the + * background image for a worksheet: + * + * @code + * worksheet_set_background(worksheet, "logo.png"); + * @endcode + * + * @image html background.png + * + * The ``set_background()`` method supports all the image formats supported by + * `worksheet_insert_image()`. + * + * Some people use this method to add a watermark background to their + * document. However, Microsoft recommends using a header image [to set a + * watermark][watermark]. The choice of method depends on whether you want the + * watermark to be visible in normal viewing mode or just when the file is + * printed. In XlsxWriter you can get the header watermark effect using + * `worksheet_set_header()`: + * + * @code + * lxw_header_footer_options header_options = {.image_center = "watermark.png"}; + * worksheet_set_header_opt(worksheet, "&C&G", &header_options); + * @endcode + * + * [watermark]:https://support.microsoft.com/en-us/office/add-a-watermark-in-excel-a372182a-d733-484e-825c-18ddf3edf009 + * + */ +lxw_error worksheet_set_background(lxw_worksheet *worksheet, + const char *filename); + +/** + * @brief Set the background image for a worksheet, from a buffer. + * + * @param worksheet Pointer to a lxw_worksheet instance to be updated. + * @param image_buffer Pointer to an array of bytes that holds the image data. + * @param image_size The size of the array of bytes. + * + * @return A #lxw_error code. + * + * This function can be used to insert a background image into a worksheet + * from a memory buffer: + * + * @code + * worksheet_set_background_buffer(worksheet, image_buffer, image_size); + * @endcode + * + * The buffer should be a pointer to an array of unsigned char data with a + * specified size. + * + * See `worksheet_set_background()` for more details. + * + */ +lxw_error worksheet_set_background_buffer(lxw_worksheet *worksheet, + const unsigned char *image_buffer, + size_t image_size); + /** * @brief Insert a chart object into a worksheet. * @@ -4845,6 +4913,10 @@ void lxw_worksheet_prepare_header_image(lxw_worksheet *worksheet, uint32_t image_ref_id, lxw_object_properties *object_props); +void lxw_worksheet_prepare_background(lxw_worksheet *worksheet, + uint32_t image_ref_id, + lxw_object_properties *object_props); + void lxw_worksheet_prepare_chart(lxw_worksheet *worksheet, uint32_t chart_ref_id, uint32_t drawing_id, lxw_object_properties *object_props, diff --git a/src/packager.c b/src/packager.c index 3e21cc50..681eaa91 100644 --- a/src/packager.c +++ b/src/packager.c @@ -1132,6 +1132,7 @@ _write_worksheet_rels_file(lxw_packager *self) STAILQ_EMPTY(worksheet->external_drawing_links) && !worksheet->external_vml_header_link && !worksheet->external_vml_comment_link && + !worksheet->external_background_link && !worksheet->external_comment_link) continue; @@ -1163,6 +1164,11 @@ _write_worksheet_rels_file(lxw_packager *self) lxw_add_worksheet_relationship(rels, rel->type, rel->target, rel->target_mode); + rel = worksheet->external_background_link; + if (rel) + lxw_add_worksheet_relationship(rels, rel->type, rel->target, + rel->target_mode); + rel = worksheet->external_comment_link; if (rel) lxw_add_worksheet_relationship(rels, rel->type, rel->target, diff --git a/src/workbook.c b/src/workbook.c index 1c3e5269..ecafc16f 100644 --- a/src/workbook.c +++ b/src/workbook.c @@ -244,6 +244,20 @@ lxw_workbook_free(lxw_workbook *workbook) free(workbook->header_image_md5s); } + if (workbook->background_md5s) { + for (image_md5 = RB_MIN(lxw_image_md5s, workbook->background_md5s); + image_md5; image_md5 = next_image_md5) { + + next_image_md5 = + RB_NEXT(lxw_image_md5s, workbook->image_md5, image_md5); + RB_REMOVE(lxw_image_md5s, workbook->background_md5s, image_md5); + free(image_md5->md5); + free(image_md5); + } + + free(workbook->background_md5s); + } + lxw_hash_free(workbook->used_xf_formats); lxw_hash_free(workbook->used_dxf_formats); lxw_sst_free(workbook->sst); @@ -1001,6 +1015,25 @@ _add_chart_cache_data(lxw_workbook *self) } } +/* + * Store the image types used in the workbook to update the content types. + */ +STATIC void +_store_image_type(lxw_workbook *self, uint8_t image_type) +{ + if (image_type == LXW_IMAGE_PNG) + self->has_png = LXW_TRUE; + + if (image_type == LXW_IMAGE_JPEG) + self->has_jpeg = LXW_TRUE; + + if (image_type == LXW_IMAGE_BMP) + self->has_bmp = LXW_TRUE; + + if (image_type == LXW_IMAGE_GIF) + self->has_gif = LXW_TRUE; +} + /* * Iterate through the worksheets and set up any chart or image drawings. */ @@ -1032,26 +1065,58 @@ _prepare_drawings(lxw_workbook *self) if (STAILQ_EMPTY(worksheet->image_props) && STAILQ_EMPTY(worksheet->chart_data) - && !worksheet->has_header_vml) { + && !worksheet->has_header_vml && !worksheet->has_background_image) { continue; } drawing_id++; + /* Prepare background images. */ + if (worksheet->has_background_image) { + + object_props = worksheet->background_image; + + _store_image_type(self, object_props->image_type); + + /* Check for duplicate images and only store the first instance. */ + if (object_props->md5) { + tmp_image_md5.md5 = object_props->md5; + found_duplicate_image = RB_FIND(lxw_image_md5s, + self->background_md5s, + &tmp_image_md5); + } + + if (found_duplicate_image) { + ref_id = found_duplicate_image->id; + object_props->is_duplicate = LXW_TRUE; + } + else { + image_ref_id++; + ref_id = image_ref_id; + +#ifndef USE_NO_MD5 + new_image_md5 = calloc(1, sizeof(lxw_image_md5)); +#endif + if (new_image_md5 && object_props->md5) { + new_image_md5->id = ref_id; + new_image_md5->md5 = lxw_strdup(object_props->md5); + + RB_INSERT(lxw_image_md5s, self->background_md5s, + new_image_md5); + } + } + + lxw_worksheet_prepare_background(worksheet, ref_id, object_props); + } + /* Prepare worksheet images. */ STAILQ_FOREACH(object_props, worksheet->image_props, list_pointers) { - if (object_props->image_type == LXW_IMAGE_PNG) - self->has_png = LXW_TRUE; + /* Ignore background image added above. */ + if (object_props->is_background) + continue; - if (object_props->image_type == LXW_IMAGE_JPEG) - self->has_jpeg = LXW_TRUE; - - if (object_props->image_type == LXW_IMAGE_BMP) - self->has_bmp = LXW_TRUE; - - if (object_props->image_type == LXW_IMAGE_GIF) - self->has_gif = LXW_TRUE; + _store_image_type(self, object_props->image_type); /* Check for duplicate images and only store the first instance. */ if (object_props->md5) { @@ -1102,17 +1167,7 @@ _prepare_drawings(lxw_workbook *self) if (!object_props) continue; - if (object_props->image_type == LXW_IMAGE_PNG) - self->has_png = LXW_TRUE; - - if (object_props->image_type == LXW_IMAGE_JPEG) - self->has_jpeg = LXW_TRUE; - - if (object_props->image_type == LXW_IMAGE_BMP) - self->has_bmp = LXW_TRUE; - - if (object_props->image_type == LXW_IMAGE_GIF) - self->has_gif = LXW_TRUE; + _store_image_type(self, object_props->image_type); /* Check for duplicate images and only store the first instance. */ if (object_props->md5) { @@ -1730,11 +1785,16 @@ workbook_new_opt(const char *filename, lxw_workbook_options *options) GOTO_LABEL_ON_MEM_ERROR(workbook->image_md5s, mem_error); RB_INIT(workbook->image_md5s); - /* Add the image MD5 tree. */ + /* Add the header image MD5 tree. */ workbook->header_image_md5s = calloc(1, sizeof(struct lxw_image_md5s)); GOTO_LABEL_ON_MEM_ERROR(workbook->header_image_md5s, mem_error); RB_INIT(workbook->header_image_md5s); + /* Add the background image MD5 tree. */ + workbook->background_md5s = calloc(1, sizeof(struct lxw_image_md5s)); + GOTO_LABEL_ON_MEM_ERROR(workbook->background_md5s, mem_error); + RB_INIT(workbook->background_md5s); + /* Add the charts list. */ workbook->charts = calloc(1, sizeof(struct lxw_charts)); GOTO_LABEL_ON_MEM_ERROR(workbook->charts, mem_error); diff --git a/src/worksheet.c b/src/worksheet.c index 1ba630aa..37ea0b57 100644 --- a/src/worksheet.c +++ b/src/worksheet.c @@ -634,6 +634,7 @@ lxw_worksheet_free(lxw_worksheet *worksheet) _free_relationship(worksheet->external_vml_comment_link); _free_relationship(worksheet->external_comment_link); _free_relationship(worksheet->external_vml_header_link); + _free_relationship(worksheet->external_background_link); if (worksheet->array) { for (col = 0; col < LXW_COL_MAX; col++) { @@ -2858,6 +2859,44 @@ mem_error: } } +/* + * Set up background image. + */ +void +lxw_worksheet_prepare_background(lxw_worksheet *self, + uint32_t image_ref_id, + lxw_object_properties *object_props) +{ + lxw_rel_tuple *relationship = NULL; + char filename[LXW_FILENAME_LENGTH]; + + STAILQ_INSERT_TAIL(self->image_props, object_props, list_pointers); + + relationship = calloc(1, sizeof(lxw_rel_tuple)); + RETURN_VOID_ON_MEM_ERROR(relationship); + + relationship->type = lxw_strdup("/image"); + GOTO_LABEL_ON_MEM_ERROR(relationship->type, mem_error); + + lxw_snprintf(filename, 32, "../media/image%d.%s", image_ref_id, + object_props->extension); + + relationship->target = lxw_strdup(filename); + GOTO_LABEL_ON_MEM_ERROR(relationship->target, mem_error); + + self->external_background_link = relationship; + + return; + +mem_error: + if (relationship) { + free(relationship->type); + free(relationship->target); + free(relationship->target_mode); + free(relationship); + } +} + /* * Set up chart/drawings. */ @@ -3329,7 +3368,7 @@ _process_jpeg(lxw_object_properties *image_props) if (!feof(stream)) { fseek_err = fseek(stream, offset, SEEK_CUR); if (fseek_err) - goto file_error; + break; } } @@ -4720,6 +4759,30 @@ _worksheet_write_legacy_drawing_hf(lxw_worksheet *self) } +/* + * Write the element. + */ +STATIC void +_worksheet_write_picture(lxw_worksheet *self) +{ + struct xml_attribute_list attributes; + struct xml_attribute *attribute; + char r_id[LXW_MAX_ATTRIBUTE_LENGTH]; + + if (!self->has_background_image) + return; + else + self->rel_count++; + + lxw_snprintf(r_id, LXW_ATTR_32, "rId%d", self->rel_count); + LXW_INIT_ATTRIBUTES(); + LXW_PUSH_ATTRIBUTES_STR("r:id", r_id); + + lxw_xml_empty_tag(self->file, "picture", &attributes); + + LXW_FREE_ATTRIBUTES(); +} + /* * Write the element. */ @@ -6709,6 +6772,9 @@ lxw_worksheet_assemble_xml_file(lxw_worksheet *self) /* Write the legacyDrawingHF element. */ _worksheet_write_legacy_drawing_hf(self); + /* Write the picture element. */ + _worksheet_write_picture(self); + /* Write the extLst element. */ _worksheet_write_ext_list(self); @@ -8771,7 +8837,7 @@ worksheet_set_default_row(lxw_worksheet *self, double height, } /* - * Insert an image into the worksheet. + * Insert an image with options into the worksheet. */ lxw_error worksheet_insert_image_opt(lxw_worksheet *self, @@ -8864,6 +8930,9 @@ worksheet_insert_image(lxw_worksheet *self, return worksheet_insert_image_opt(self, row_num, col_num, filename, NULL); } +/* + * Insert an image buffer, with options, into the worksheet. + */ lxw_error worksheet_insert_image_buffer_opt(lxw_worksheet *self, lxw_row_t row_num, @@ -8955,6 +9024,9 @@ worksheet_insert_image_buffer_opt(lxw_worksheet *self, } } +/* + * Insert an image buffer into the worksheet. + */ lxw_error worksheet_insert_image_buffer(lxw_worksheet *self, lxw_row_t row_num, @@ -8966,6 +9038,129 @@ worksheet_insert_image_buffer(lxw_worksheet *self, image_buffer, image_size, NULL); } +/* + * Set an image as a worksheet background. + */ +lxw_error +worksheet_set_background(lxw_worksheet *self, const char *filename) +{ + FILE *image_stream; + lxw_object_properties *object_props; + + if (!filename) { + LXW_WARN("worksheet_set_background(): " + "filename must be specified."); + return LXW_ERROR_NULL_PARAMETER_IGNORED; + } + + /* Check that the image file exists and can be opened. */ + image_stream = lxw_fopen(filename, "rb"); + if (!image_stream) { + LXW_WARN_FORMAT1("worksheet_set_background(): " + "file doesn't exist or can't be opened: %s.", + filename); + return LXW_ERROR_PARAMETER_VALIDATION; + } + + /* Create a new object to hold the image properties. */ + object_props = calloc(1, sizeof(lxw_object_properties)); + if (!object_props) { + fclose(image_stream); + return LXW_ERROR_MEMORY_MALLOC_FAILED; + } + + /* Copy other options or set defaults. */ + object_props->filename = lxw_strdup(filename); + object_props->stream = image_stream; + object_props->is_background = LXW_TRUE; + + if (_get_image_properties(object_props) == LXW_NO_ERROR) { + _free_object_properties(self->background_image); + self->background_image = object_props; + self->has_background_image = LXW_TRUE; + fclose(image_stream); + return LXW_NO_ERROR; + } + else { + _free_object_properties(object_props); + fclose(image_stream); + return LXW_ERROR_IMAGE_DIMENSIONS; + } +} + +/* + * Set an image buffer as a worksheet background. + */ +lxw_error +worksheet_set_background_buffer(lxw_worksheet *self, + const unsigned char *image_buffer, + size_t image_size) +{ + FILE *image_stream; + lxw_object_properties *object_props; + + if (!image_size) { + LXW_WARN("worksheet_set_background(): " "size must be non-zero."); + return LXW_ERROR_NULL_PARAMETER_IGNORED; + } + + /* Write the image buffer to a file (preferably in memory) so we can read + * the dimensions like an ordinary file. */ +#ifdef USE_FMEMOPEN + image_stream = fmemopen(NULL, image_size, "w+b"); +#else + image_stream = lxw_tmpfile(self->tmpdir); +#endif + + if (!image_stream) + return LXW_ERROR_CREATING_TMPFILE; + + if (fwrite(image_buffer, 1, image_size, image_stream) != image_size) { + fclose(image_stream); + return LXW_ERROR_CREATING_TMPFILE; + } + + rewind(image_stream); + + /* Create a new object to hold the image properties. */ + object_props = calloc(1, sizeof(lxw_object_properties)); + if (!object_props) { + fclose(image_stream); + return LXW_ERROR_MEMORY_MALLOC_FAILED; + } + + /* Store the image data in the properties structure. */ + object_props->image_buffer = calloc(1, image_size); + if (!object_props->image_buffer) { + _free_object_properties(object_props); + fclose(image_stream); + return LXW_ERROR_MEMORY_MALLOC_FAILED; + } + else { + memcpy(object_props->image_buffer, image_buffer, image_size); + object_props->image_buffer_size = image_size; + object_props->is_image_buffer = LXW_TRUE; + } + + /* Copy other options or set defaults. */ + object_props->filename = lxw_strdup("image_buffer"); + object_props->stream = image_stream; + object_props->is_background = LXW_TRUE; + + if (_get_image_properties(object_props) == LXW_NO_ERROR) { + _free_object_properties(self->background_image); + self->background_image = object_props; + self->has_background_image = LXW_TRUE; + fclose(image_stream); + return LXW_NO_ERROR; + } + else { + _free_object_properties(object_props); + fclose(image_stream); + return LXW_ERROR_IMAGE_DIMENSIONS; + } +} + /* * Insert an chart into the worksheet. */ diff --git a/test/functional/src/images/logo.jpg b/test/functional/src/images/logo.jpg new file mode 100644 index 00000000..76310461 Binary files /dev/null and b/test/functional/src/images/logo.jpg differ diff --git a/test/functional/src/test_background01.c b/test/functional/src/test_background01.c new file mode 100644 index 00000000..17f0eeff --- /dev/null +++ b/test/functional/src/test_background01.c @@ -0,0 +1,20 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background01.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_insert_image(worksheet, CELL("E9"), "images/logo.jpg"); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background02.c b/test/functional/src/test_background02.c new file mode 100644 index 00000000..42a5f9be --- /dev/null +++ b/test/functional/src/test_background02.c @@ -0,0 +1,20 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background02.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_set_background(worksheet, "images/logo.jpg"); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background03.c b/test/functional/src/test_background03.c new file mode 100644 index 00000000..82a2f916 --- /dev/null +++ b/test/functional/src/test_background03.c @@ -0,0 +1,21 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background03.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_insert_image(worksheet, CELL("E9"), "images/logo.jpg"); + worksheet_set_background(worksheet, "images/logo.jpg"); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background04.c b/test/functional/src/test_background04.c new file mode 100644 index 00000000..8129de7e --- /dev/null +++ b/test/functional/src/test_background04.c @@ -0,0 +1,22 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background04.xlsx"); + lxw_worksheet *worksheet1 = workbook_add_worksheet(workbook, NULL); + lxw_worksheet *worksheet2 = workbook_add_worksheet(workbook, NULL); + + worksheet_set_background(worksheet1, "images/logo.jpg"); + worksheet_set_background(worksheet2, "images/logo.jpg"); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background05.c b/test/functional/src/test_background05.c new file mode 100644 index 00000000..86554f94 --- /dev/null +++ b/test/functional/src/test_background05.c @@ -0,0 +1,22 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background05.xlsx"); + lxw_worksheet *worksheet1 = workbook_add_worksheet(workbook, NULL); + lxw_worksheet *worksheet2 = workbook_add_worksheet(workbook, NULL); + + worksheet_set_background(worksheet1, "images/logo.jpg"); + worksheet_set_background(worksheet2, "images/red.jpg"); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background06.c b/test/functional/src/test_background06.c new file mode 100644 index 00000000..0f2bad0c --- /dev/null +++ b/test/functional/src/test_background06.c @@ -0,0 +1,24 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background06.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_insert_image(worksheet, CELL("E9"), "images/logo.jpg"); + worksheet_set_background(worksheet, "images/logo.jpg"); + + lxw_header_footer_options header_options = {.image_center = "images/blue.jpg"}; + worksheet_set_header_opt(worksheet, "&C&G", &header_options); + + return workbook_close(workbook); +} diff --git a/test/functional/src/test_background07.c b/test/functional/src/test_background07.c new file mode 100644 index 00000000..52ab7307 --- /dev/null +++ b/test/functional/src/test_background07.c @@ -0,0 +1,28 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +int main() { + + lxw_workbook *workbook = workbook_new("test_background07.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_insert_image(worksheet, CELL("E9"), "images/logo.jpg"); + worksheet_set_background(worksheet, "images/logo.jpg"); + + lxw_header_footer_options header_options = {.image_center = "images/blue.jpg"}; + worksheet_set_header_opt(worksheet, "&C&G", &header_options); + + 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); +} diff --git a/test/functional/src/test_background52.c b/test/functional/src/test_background52.c new file mode 100644 index 00000000..bdef48a3 --- /dev/null +++ b/test/functional/src/test_background52.c @@ -0,0 +1,368 @@ +/***************************************************************************** + * Test cases for libxlsxwriter. + * + * Test to compare output against Excel files. + * + * Copyright 2014-2021, John McNamara, jmcnamara@cpan.org + * + */ + +#include "xlsxwriter.h" + +unsigned char image_buffer[] = { + 0xff, 0xd8, 0xff, 0xe0, 0x00, 0x10, 0x4a, 0x46, 0x49, 0x46, 0x00, 0x01, + 0x01, 0x00, 0x00, 0x01, 0x00, 0x01, 0x00, 0x00, 0xff, 0xdb, 0x00, 0x43, + 0x00, 0x06, 0x04, 0x05, 0x06, 0x05, 0x04, 0x06, 0x06, 0x05, 0x06, 0x07, + 0x07, 0x06, 0x08, 0x0a, 0x10, 0x0a, 0x0a, 0x09, 0x09, 0x0a, 0x14, 0x0e, + 0x0f, 0x0c, 0x10, 0x17, 0x14, 0x18, 0x18, 0x17, 0x14, 0x16, 0x16, 0x1a, + 0x1d, 0x25, 0x1f, 0x1a, 0x1b, 0x23, 0x1c, 0x16, 0x16, 0x20, 0x2c, 0x20, + 0x23, 0x26, 0x27, 0x29, 0x2a, 0x29, 0x19, 0x1f, 0x2d, 0x30, 0x2d, 0x28, + 0x30, 0x25, 0x28, 0x29, 0x28, 0xff, 0xdb, 0x00, 0x43, 0x01, 0x07, 0x07, + 0x07, 0x0a, 0x08, 0x0a, 0x13, 0x0a, 0x0a, 0x13, 0x28, 0x1a, 0x16, 0x1a, + 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, + 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, + 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, + 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, 0x28, + 0x28, 0x28, 0xff, 0xc0, 0x00, 0x11, 0x08, 0x00, 0x50, 0x00, 0xc8, 0x03, + 0x01, 0x22, 0x00, 0x02, 0x11, 0x01, 0x03, 0x11, 0x01, 0xff, 0xc4, 0x00, + 0x1c, 0x00, 0x00, 0x02, 0x02, 0x03, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x05, 0x04, 0x06, 0x02, 0x03, + 0x07, 0x01, 0x08, 0xff, 0xc4, 0x00, 0x49, 0x10, 0x00, 0x01, 0x03, 0x03, + 0x01, 0x03, 0x08, 0x07, 0x04, 0x06, 0x05, 0x0d, 0x00, 0x00, 0x00, 0x00, + 0x01, 0x02, 0x03, 0x04, 0x00, 0x05, 0x11, 0x06, 0x12, 0x21, 0x31, 0x07, + 0x13, 0x15, 0x41, 0x51, 0x55, 0x94, 0xd1, 0x14, 0x36, 0x61, 0x71, 0x74, + 0x81, 0xb2, 0x32, 0x91, 0xb1, 0xc1, 0x22, 0x33, 0x34, 0x72, 0xa1, 0xa2, + 0x16, 0x23, 0x52, 0x73, 0xc2, 0x24, 0x35, 0x42, 0x43, 0x54, 0x62, 0x64, + 0x65, 0x82, 0x92, 0x93, 0xd2, 0xf0, 0xff, 0xc4, 0x00, 0x1b, 0x01, 0x00, + 0x02, 0x03, 0x01, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, + 0x00, 0x00, 0x00, 0x00, 0x01, 0x03, 0x04, 0x05, 0x06, 0x02, 0x07, 0xff, + 0xc4, 0x00, 0x2b, 0x11, 0x00, 0x02, 0x02, 0x01, 0x03, 0x03, 0x03, 0x03, + 0x04, 0x03, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x01, 0x02, 0x03, + 0x11, 0x04, 0x12, 0x21, 0x05, 0x13, 0x31, 0x06, 0x41, 0x51, 0x61, 0x81, + 0xe1, 0x22, 0x71, 0x91, 0xc1, 0x52, 0xa1, 0xd1, 0xff, 0xda, 0x00, 0x0c, + 0x03, 0x01, 0x00, 0x02, 0x11, 0x03, 0x11, 0x00, 0x3f, 0x00, 0xe8, 0x9a, + 0x7a, 0xc9, 0x6a, 0x55, 0x82, 0xd8, 0xa5, 0x5b, 0x20, 0x95, 0x18, 0xad, + 0x12, 0x4c, 0x74, 0x64, 0x9d, 0x81, 0xec, 0xa6, 0x1d, 0x07, 0x69, 0xee, + 0xb8, 0x1e, 0x1d, 0x1e, 0x54, 0x69, 0xcf, 0x57, 0xad, 0x7f, 0x0a, 0xd7, + 0xd0, 0x29, 0x8d, 0x00, 0x2e, 0xe8, 0x3b, 0x4f, 0x75, 0xc0, 0xf0, 0xe8, + 0xf2, 0xa3, 0xa0, 0xed, 0x3d, 0xd7, 0x03, 0xc3, 0xa3, 0xca, 0x98, 0xd5, + 0x2b, 0x94, 0x4b, 0xb4, 0xc8, 0xca, 0x87, 0x6e, 0xb7, 0x2d, 0x4d, 0xbb, + 0x27, 0xed, 0x29, 0x27, 0x04, 0x8c, 0xe0, 0x00, 0x7a, 0xb7, 0xd4, 0x95, + 0x56, 0xed, 0x92, 0x8a, 0x23, 0xb6, 0xc5, 0x54, 0x5c, 0x98, 0xfd, 0xfb, + 0x65, 0x85, 0x8f, 0xd7, 0xc2, 0xb5, 0xb7, 0xfb, 0xed, 0x36, 0x3f, 0x11, + 0x51, 0x83, 0x7a, 0x58, 0xab, 0x64, 0x22, 0xc8, 0x55, 0xd8, 0x03, 0x59, + 0xaa, 0x7e, 0xbc, 0xe4, 0xf6, 0x6e, 0x97, 0xb5, 0x31, 0x71, 0x91, 0x39, + 0xa9, 0x61, 0xc7, 0x03, 0x6e, 0x84, 0xa4, 0x82, 0x85, 0x10, 0x4f, 0x12, + 0x7f, 0x48, 0x6e, 0xe3, 0xbb, 0xdd, 0x54, 0x23, 0x57, 0xab, 0xd0, 0xc2, + 0x71, 0xdd, 0x19, 0x64, 0xa5, 0x3d, 0x6c, 0xe1, 0x2d, 0xb2, 0x8e, 0x0e, + 0xee, 0xcd, 0xaa, 0xc6, 0xf8, 0x25, 0x98, 0x16, 0xd7, 0x00, 0xfe, 0xc3, + 0x28, 0x3f, 0x80, 0xad, 0x9d, 0x07, 0x69, 0xee, 0xb8, 0x1e, 0x1d, 0x1e, + 0x55, 0xc1, 0x98, 0x90, 0xf4, 0x67, 0x92, 0xec, 0x77, 0x56, 0xd3, 0xa9, + 0xde, 0x14, 0x85, 0x60, 0x8a, 0xec, 0xda, 0x0a, 0xf8, 0xed, 0xf2, 0xce, + 0x57, 0x2b, 0x06, 0x53, 0x0a, 0xe6, 0xdc, 0x50, 0xdd, 0xb5, 0xbb, 0x21, + 0x58, 0xff, 0x00, 0xee, 0x15, 0x06, 0xa3, 0x48, 0xea, 0x5b, 0x93, 0xca, + 0x26, 0xa3, 0x54, 0xad, 0x7b, 0x5a, 0xc3, 0x19, 0xf4, 0x1d, 0xa7, 0xba, + 0xe0, 0x78, 0x74, 0x79, 0x51, 0xd0, 0x76, 0x9e, 0xeb, 0x81, 0xe1, 0xd1, + 0xe5, 0x4c, 0x68, 0xaa, 0x85, 0xb1, 0x77, 0x41, 0xda, 0x7b, 0xae, 0x07, + 0x87, 0x47, 0x95, 0x1d, 0x07, 0x69, 0xee, 0xb8, 0x1e, 0x1d, 0x1e, 0x54, + 0xc6, 0xbc, 0x5a, 0xd2, 0x81, 0x95, 0xa8, 0x24, 0x76, 0x93, 0x8a, 0x00, + 0x5f, 0xd0, 0x76, 0x9e, 0xeb, 0x81, 0xe1, 0xd1, 0xe5, 0x47, 0x41, 0xda, + 0x7b, 0xae, 0x07, 0x87, 0x47, 0x95, 0x31, 0x1b, 0xc6, 0xea, 0x32, 0x33, + 0x8c, 0xef, 0xa0, 0x05, 0xdd, 0x07, 0x69, 0xee, 0xb8, 0x1e, 0x1d, 0x1e, + 0x54, 0x74, 0x1d, 0xa7, 0xba, 0xe0, 0x78, 0x74, 0x79, 0x53, 0x1a, 0x81, + 0x7e, 0x96, 0xe4, 0x0b, 0x34, 0xd9, 0x4c, 0x6c, 0x97, 0x5a, 0x68, 0xad, + 0x3b, 0x43, 0x23, 0x22, 0x80, 0x31, 0xe8, 0x3b, 0x4f, 0x75, 0xc0, 0xf0, + 0xe8, 0xf2, 0xa8, 0x36, 0xd6, 0x34, 0xf5, 0xc9, 0xe9, 0x4d, 0xc3, 0xb7, + 0xc2, 0x70, 0xc6, 0x56, 0xc3, 0x87, 0xd1, 0x52, 0x00, 0x3b, 0xf7, 0x6f, + 0x1e, 0xca, 0x97, 0xa5, 0xa6, 0x3d, 0x3e, 0xc1, 0x0e, 0x54, 0xa5, 0x05, + 0x3c, 0xea, 0x49, 0x51, 0x03, 0x1d, 0x67, 0xaa, 0xab, 0xfc, 0x9d, 0xfe, + 0xd9, 0xa8, 0x3e, 0x2b, 0xf3, 0x55, 0x00, 0x58, 0xfa, 0x0e, 0xd3, 0xdd, + 0x70, 0x3c, 0x3a, 0x3c, 0xa8, 0xe8, 0x3b, 0x4f, 0x75, 0xc0, 0xf0, 0xe8, + 0xf2, 0xa6, 0x34, 0x50, 0x02, 0xee, 0x83, 0xb4, 0xf7, 0x5c, 0x0f, 0x0e, + 0x8f, 0x2a, 0x3a, 0x0e, 0xd3, 0xdd, 0x70, 0x3c, 0x3a, 0x3c, 0xa9, 0x82, + 0x56, 0x95, 0x12, 0x12, 0xa0, 0x48, 0xdc, 0x40, 0x3c, 0x2b, 0xda, 0x00, + 0x5d, 0xd0, 0x76, 0x9e, 0xeb, 0x81, 0xe1, 0xd1, 0xe5, 0x47, 0x41, 0xda, + 0x7b, 0xae, 0x07, 0x87, 0x47, 0x95, 0x4f, 0x5b, 0x88, 0x6f, 0x1b, 0x6b, + 0x4a, 0x73, 0xc3, 0x27, 0x15, 0x95, 0x00, 0x57, 0xb5, 0x0d, 0x92, 0xd4, + 0x9b, 0x05, 0xcd, 0x49, 0xb6, 0x41, 0x0a, 0x11, 0x5d, 0x20, 0x88, 0xe8, + 0xc8, 0x3b, 0x07, 0xd9, 0x45, 0x30, 0xd4, 0x7e, 0xaf, 0x5d, 0x3e, 0x15, + 0xdf, 0xa0, 0xd1, 0x40, 0x06, 0x9c, 0xf5, 0x7a, 0xd7, 0xf0, 0xad, 0x7d, + 0x02, 0x98, 0xd2, 0xed, 0x39, 0xea, 0xf5, 0xaf, 0xe1, 0x5a, 0xfa, 0x05, + 0x29, 0xd7, 0x3a, 0x81, 0x76, 0x68, 0x6d, 0xb5, 0x10, 0x8f, 0x4b, 0x7f, + 0x3b, 0x2a, 0x3b, 0xf6, 0x12, 0x38, 0x9c, 0x7e, 0x15, 0xee, 0xba, 0xe5, + 0x64, 0x94, 0x23, 0xe5, 0x9e, 0x2c, 0xb2, 0x35, 0xc5, 0xca, 0x5e, 0x0b, + 0x3d, 0x50, 0xf9, 0x43, 0xb1, 0xdc, 0xae, 0xb7, 0x28, 0xce, 0xdb, 0xe3, + 0x17, 0x90, 0x86, 0xb6, 0x49, 0x0b, 0x48, 0xc1, 0xda, 0x27, 0xac, 0xd7, + 0x3f, 0x95, 0x36, 0x54, 0xa7, 0x0a, 0xe4, 0x48, 0x79, 0xd5, 0x93, 0x9c, + 0xad, 0x64, 0xd4, 0x72, 0xe2, 0xff, 0x00, 0xb6, 0xaf, 0xbe, 0xb5, 0xa9, + 0xd0, 0xca, 0xa9, 0x6f, 0x52, 0xe7, 0xf6, 0xfc, 0x99, 0x57, 0x6b, 0x63, + 0x6c, 0x76, 0xb8, 0xf1, 0xfb, 0xfe, 0x0b, 0x95, 0xee, 0x16, 0xb7, 0xbd, + 0xc6, 0x62, 0x3d, 0xd7, 0xd2, 0x24, 0xb2, 0xc6, 0xf4, 0x25, 0x6e, 0xb7, + 0xb8, 0xf6, 0x9c, 0x1d, 0xe7, 0x7f, 0x13, 0xbe, 0xaa, 0x97, 0x6b, 0x4c, + 0xeb, 0x4b, 0x88, 0x45, 0xc1, 0x82, 0xca, 0x9c, 0x1b, 0x49, 0x05, 0x40, + 0xe4, 0x7c, 0x89, 0xa8, 0xaa, 0x71, 0x7f, 0xdb, 0x57, 0xdf, 0x5a, 0xd4, + 0xa5, 0x2b, 0xed, 0x12, 0x7d, 0xe6, 0xac, 0xc6, 0x32, 0x8f, 0x1c, 0x63, + 0xe8, 0xbf, 0x25, 0x69, 0x4a, 0x32, 0xe7, 0x9c, 0xfd, 0x5f, 0xe0, 0xc1, + 0x55, 0xd3, 0x39, 0x1f, 0xfd, 0x96, 0xe7, 0xfb, 0xe8, 0xfc, 0x0d, 0x73, + 0x35, 0x57, 0x4b, 0xe4, 0x83, 0xf6, 0x4b, 0x9f, 0xef, 0xa3, 0xf0, 0x35, + 0x47, 0xaa, 0xdd, 0xd8, 0xd2, 0x4e, 0xdc, 0x67, 0x6a, 0xcf, 0xf0, 0x5b, + 0xe9, 0xf0, 0xdf, 0xa8, 0x8c, 0x7e, 0x4e, 0x87, 0x45, 0x60, 0x6b, 0x24, + 0x9c, 0x8a, 0xe4, 0x3a, 0x6f, 0x5b, 0xaf, 0x5d, 0x67, 0x69, 0xc7, 0x6c, + 0xbd, 0xb9, 0xce, 0x4e, 0x8f, 0x51, 0xa2, 0x95, 0x31, 0xdd, 0x9c, 0xa2, + 0x25, 0xea, 0x43, 0x91, 0x2c, 0xf3, 0xa4, 0x32, 0x40, 0x75, 0xa6, 0x16, + 0xe2, 0x09, 0x19, 0xc1, 0x09, 0x24, 0x55, 0x53, 0x4a, 0xd8, 0xa0, 0xde, + 0xec, 0xcc, 0x5c, 0xaf, 0x09, 0x76, 0x6c, 0xb7, 0xca, 0xf6, 0x94, 0xeb, + 0xaa, 0xc0, 0xc2, 0x88, 0xc0, 0x00, 0xee, 0x1b, 0xaa, 0xcb, 0xa9, 0x7d, + 0x5c, 0xba, 0x7c, 0x2b, 0xbf, 0x41, 0xa5, 0x9c, 0x9d, 0x3a, 0xda, 0xf4, + 0x94, 0x34, 0x21, 0x69, 0x52, 0xd0, 0x56, 0x16, 0x90, 0x72, 0x52, 0x4a, + 0xd4, 0x77, 0xf6, 0x6e, 0xad, 0xb2, 0x90, 0xaa, 0xdc, 0x55, 0xa6, 0xb5, + 0xa2, 0x2d, 0x2c, 0xad, 0x66, 0xdb, 0x31, 0x1b, 0x6d, 0xb6, 0xa5, 0x67, + 0x9b, 0x51, 0xcf, 0x0c, 0xfb, 0x52, 0x47, 0xcc, 0x56, 0xfb, 0x72, 0x94, + 0x79, 0x51, 0xba, 0x02, 0xa2, 0x42, 0x62, 0x80, 0x01, 0x3c, 0x37, 0x37, + 0x51, 0x6e, 0x4a, 0x17, 0x4e, 0x52, 0xe0, 0x37, 0x14, 0xed, 0xa6, 0x13, + 0x60, 0xba, 0xa4, 0xf0, 0x49, 0x04, 0x93, 0xf8, 0x81, 0xef, 0xad, 0xb1, + 0x9d, 0x44, 0x6e, 0x54, 0xe6, 0xf3, 0xea, 0x08, 0xf4, 0x88, 0xe9, 0x4b, + 0x64, 0xee, 0xda, 0x3b, 0x28, 0xff, 0x00, 0xd4, 0xd0, 0x03, 0xad, 0x79, + 0xbb, 0x49, 0x5c, 0x7f, 0x75, 0x3f, 0x58, 0xa5, 0x13, 0xa1, 0x47, 0x97, + 0xc9, 0xc4, 0x77, 0x24, 0x37, 0xb6, 0xb6, 0x21, 0x87, 0x1b, 0x3b, 0x44, + 0x6c, 0xab, 0x67, 0x8e, 0xee, 0x3f, 0x3a, 0x69, 0xaf, 0xdc, 0x42, 0x34, + 0xa4, 0xe4, 0xad, 0x69, 0x4a, 0x96, 0x12, 0x12, 0x09, 0xc1, 0x51, 0xda, + 0x1b, 0x87, 0x6d, 0x44, 0x20, 0x9e, 0x4c, 0xf0, 0x06, 0x7f, 0xc8, 0x33, + 0xfc, 0xb4, 0x01, 0xee, 0x80, 0xb4, 0xc2, 0x62, 0xcd, 0x12, 0xe0, 0xd3, + 0x3b, 0x32, 0xde, 0x68, 0xa5, 0x6e, 0x6d, 0xa8, 0xe4, 0x6d, 0x76, 0x67, + 0x1d, 0x43, 0xaa, 0x91, 0xe9, 0x2b, 0x51, 0xba, 0x5c, 0x2f, 0x69, 0x7a, + 0x5c, 0x86, 0xa2, 0xa6, 0x49, 0xdb, 0x65, 0x95, 0x6c, 0xf3, 0x87, 0x2a, + 0xc6, 0x4f, 0x1c, 0x7b, 0x2a, 0xd3, 0xa1, 0x54, 0x15, 0xa4, 0xed, 0xdb, + 0x24, 0x1c, 0x20, 0x8f, 0x9e, 0xd1, 0xa5, 0x3c, 0x9d, 0xfe, 0xd9, 0xa8, + 0x3e, 0x2b, 0xf3, 0x55, 0x00, 0x41, 0xbc, 0x41, 0x4e, 0x95, 0xbf, 0xda, + 0xa4, 0x5a, 0x54, 0xeb, 0x71, 0xa4, 0xb9, 0xcd, 0xbc, 0xd1, 0x59, 0x50, + 0x3b, 0xc7, 0x6f, 0xb0, 0xfd, 0xe2, 0x98, 0xea, 0x69, 0x4f, 0xdc, 0xf5, + 0x24, 0x4d, 0x3f, 0x1d, 0xe5, 0xb1, 0x1d, 0x69, 0xe7, 0x24, 0xad, 0x07, + 0x0a, 0x52, 0x70, 0x4e, 0xce, 0x7d, 0xc3, 0xf8, 0xd6, 0x1c, 0xa3, 0x7e, + 0xbe, 0xc5, 0xf1, 0x5f, 0x9a, 0x6b, 0x5d, 0xdc, 0x8b, 0x4f, 0x28, 0x90, + 0xa7, 0xc8, 0x21, 0x11, 0x65, 0xb7, 0xcd, 0x97, 0x0f, 0x00, 0xac, 0x63, + 0x7f, 0xf2, 0xfd, 0xf4, 0x01, 0x22, 0xf5, 0xa4, 0xe1, 0x42, 0xb6, 0xb9, + 0x2e, 0xca, 0x1c, 0x87, 0x3a, 0x32, 0x0b, 0x88, 0x71, 0x0e, 0x28, 0xed, + 0x63, 0x79, 0x07, 0x27, 0xae, 0xb4, 0x4d, 0xd4, 0xb2, 0xde, 0xd0, 0x29, + 0xb9, 0x47, 0x58, 0x6a, 0x66, 0xd0, 0x69, 0x6a, 0x00, 0x1c, 0x1c, 0xe0, + 0x91, 0xef, 0xe3, 0xf3, 0xab, 0x26, 0xa6, 0x98, 0xd4, 0x1b, 0x0c, 0xd7, + 0x9d, 0x50, 0x4f, 0xf5, 0x4a, 0x4a, 0x72, 0x78, 0xa8, 0x8c, 0x00, 0x3e, + 0x75, 0x44, 0x95, 0x05, 0xc8, 0x3c, 0x97, 0xa4, 0x3c, 0x9d, 0x95, 0xbc, + 0xf2, 0x5e, 0xc1, 0xea, 0x04, 0xee, 0xfe, 0x00, 0x50, 0x05, 0x9e, 0x1e, + 0x94, 0xb4, 0xcc, 0xb7, 0x34, 0xec, 0xc6, 0x5c, 0x91, 0x21, 0xe6, 0xd2, + 0xb5, 0xbe, 0xe3, 0xaa, 0x2b, 0x24, 0x8c, 0xe7, 0x39, 0xa8, 0x5c, 0x9f, + 0x3b, 0x21, 0x89, 0xb7, 0x9b, 0x5b, 0xaf, 0xad, 0xe6, 0x61, 0xba, 0x12, + 0xd1, 0x59, 0xc9, 0x03, 0x2a, 0x1f, 0x90, 0xdd, 0x56, 0xab, 0x4f, 0xf9, + 0xaa, 0x1f, 0xf7, 0x28, 0xfa, 0x45, 0x55, 0x74, 0x5f, 0xad, 0x5a, 0x9f, + 0xfb, 0xff, 0x00, 0xf1, 0x2e, 0x80, 0x2c, 0x7a, 0x8f, 0xd5, 0xeb, 0xa7, + 0xc2, 0xbb, 0xf4, 0x1a, 0x28, 0xd4, 0x7e, 0xaf, 0x5d, 0x3e, 0x15, 0xdf, + 0xa0, 0xd1, 0x40, 0x06, 0x9c, 0xf5, 0x7a, 0xd7, 0xf0, 0xad, 0x7d, 0x02, + 0xb9, 0xb7, 0x28, 0xef, 0x97, 0x75, 0x3b, 0xa8, 0x3c, 0x19, 0x6d, 0x08, + 0x1f, 0x76, 0xd7, 0xf8, 0xab, 0xa4, 0xe9, 0xcf, 0x57, 0xad, 0x7f, 0x0a, + 0xd7, 0xd0, 0x2b, 0x9a, 0x72, 0x8a, 0xc9, 0x6f, 0x54, 0x3e, 0xa2, 0x37, + 0x3a, 0x84, 0x2c, 0x7f, 0xdb, 0xb3, 0xf9, 0x55, 0xfe, 0x9d, 0x8e, 0xef, + 0xd8, 0xa1, 0xd4, 0x73, 0xda, 0xfb, 0x95, 0x73, 0x58, 0x1a, 0xcc, 0xd7, + 0x49, 0xe4, 0xdf, 0x4c, 0xc7, 0x5c, 0x34, 0x5d, 0x26, 0xb6, 0x97, 0x5c, + 0x59, 0x25, 0x94, 0xa8, 0x64, 0x20, 0x03, 0x8c, 0xe3, 0xb7, 0x22, 0xb4, + 0x35, 0xda, 0xda, 0xf4, 0x54, 0xbb, 0xed, 0xf0, 0x8c, 0xdd, 0x2e, 0x9a, + 0x7a, 0x9b, 0x15, 0x70, 0xf2, 0x73, 0x84, 0x43, 0x92, 0xea, 0x76, 0x9a, + 0x8e, 0xf2, 0xd2, 0x7a, 0xd2, 0x82, 0x45, 0x06, 0xdf, 0x37, 0xfd, 0x92, + 0x47, 0xfe, 0x25, 0x79, 0x57, 0xd1, 0x3b, 0x09, 0x1d, 0x55, 0x89, 0x48, + 0xec, 0xae, 0x42, 0x5e, 0xb6, 0xa7, 0x3c, 0x54, 0xff, 0x00, 0xd1, 0xbc, + 0xbd, 0x3d, 0x2f, 0xf3, 0x3e, 0x7b, 0x8f, 0x67, 0xb9, 0x49, 0x70, 0x36, + 0xc4, 0x09, 0x4b, 0x51, 0x38, 0xdc, 0xd1, 0xc0, 0xf7, 0x9e, 0xaa, 0xeb, + 0xda, 0x26, 0xc4, 0xab, 0x15, 0x9b, 0x9a, 0x7c, 0x83, 0x25, 0xd5, 0x73, + 0x8e, 0xe0, 0xe4, 0x03, 0xc0, 0x01, 0xee, 0x1f, 0x9d, 0x59, 0x08, 0x15, + 0x8a, 0xab, 0x17, 0xac, 0x7a, 0xa5, 0xeb, 0xa8, 0x74, 0x55, 0x0d, 0xa9, + 0xf9, 0xc9, 0xa3, 0xa0, 0xe8, 0xf1, 0xd3, 0x59, 0xdc, 0x94, 0xb2, 0xd1, + 0xac, 0xd7, 0xa8, 0xe2, 0x6b, 0xc3, 0x42, 0x3e, 0xd5, 0x60, 0x74, 0x76, + 0xd6, 0xba, 0xbd, 0xbf, 0x3f, 0xd1, 0xab, 0xab, 0xc7, 0x62, 0x59, 0x20, + 0x6a, 0x5f, 0x57, 0x2e, 0x9f, 0x0a, 0xef, 0xd0, 0x6a, 0xad, 0xa4, 0x74, + 0xdd, 0xb6, 0xe7, 0xa5, 0xe0, 0xc8, 0x7d, 0xb5, 0xa2, 0x42, 0x83, 0x81, + 0x4e, 0xb4, 0xb2, 0x85, 0x28, 0x6d, 0xa8, 0x6f, 0xc7, 0x1a, 0xba, 0xcd, + 0x8c, 0x89, 0x90, 0xdf, 0x8c, 0xe9, 0x50, 0x6d, 0xe6, 0xd4, 0xda, 0x8a, + 0x4e, 0xfc, 0x11, 0x83, 0x8a, 0x8f, 0x63, 0xb6, 0xb7, 0x68, 0xb5, 0xb3, + 0x05, 0x95, 0xa9, 0xc6, 0xda, 0xda, 0xc2, 0x95, 0xc4, 0xe5, 0x44, 0xfe, + 0x75, 0xf4, 0x93, 0x9c, 0x31, 0xb3, 0xd9, 0xa0, 0x59, 0xda, 0x52, 0x20, + 0x30, 0x1b, 0xdb, 0xde, 0xa5, 0x12, 0x4a, 0x95, 0xef, 0x26, 0xb0, 0xbd, + 0x58, 0xad, 0xf7, 0x94, 0xa3, 0xd3, 0xd8, 0xda, 0x5a, 0x3e, 0xca, 0xd2, + 0x4a, 0x54, 0x3e, 0x62, 0x99, 0xd6, 0xb9, 0x0d, 0xa9, 0xd8, 0xee, 0x36, + 0x87, 0x14, 0xd2, 0x94, 0x92, 0x90, 0xb4, 0xf1, 0x49, 0x23, 0x88, 0xf7, + 0x50, 0x05, 0x1f, 0x58, 0x58, 0x6d, 0x56, 0x9d, 0x39, 0x25, 0xf4, 0x36, + 0xa5, 0xca, 0x3b, 0x2d, 0xb6, 0xb7, 0x9c, 0x2b, 0x20, 0x95, 0x0c, 0xe3, + 0x27, 0xb3, 0x35, 0x6c, 0xb0, 0xc6, 0x2c, 0xd8, 0x20, 0x47, 0x79, 0x03, + 0x29, 0x8e, 0x84, 0xad, 0x24, 0x75, 0xec, 0x8c, 0x8a, 0x81, 0x1b, 0x4b, + 0xc5, 0x4c, 0xb6, 0xe5, 0x4f, 0x91, 0x2e, 0xe0, 0xfb, 0x7b, 0xd0, 0x64, + 0xb9, 0xb4, 0x94, 0x9e, 0xd0, 0x9e, 0x14, 0xfe, 0x80, 0x15, 0x5a, 0xec, + 0x30, 0x6d, 0x52, 0x9c, 0x7a, 0x0a, 0x5c, 0x6b, 0x6c, 0x10, 0x5b, 0xe7, + 0x09, 0x46, 0xf3, 0x9c, 0xec, 0x9e, 0xbd, 0xd5, 0xbe, 0xd9, 0x69, 0x89, + 0x6c, 0x72, 0x4a, 0xe1, 0xb6, 0x50, 0xa9, 0x0b, 0xdb, 0x73, 0x2a, 0x27, + 0x27, 0x7f, 0x6f, 0xbe, 0xa7, 0x51, 0x40, 0x10, 0x6e, 0x96, 0x98, 0x97, + 0x35, 0x47, 0x54, 0xc6, 0xca, 0xcb, 0x0b, 0xdb, 0x6f, 0x0a, 0x23, 0x07, + 0xe5, 0xee, 0xad, 0xb7, 0x08, 0x31, 0xae, 0x31, 0x55, 0x1e, 0x6b, 0x29, + 0x79, 0x95, 0x71, 0x4a, 0xbf, 0x10, 0x78, 0x83, 0x52, 0x68, 0xa0, 0x04, + 0x31, 0xf4, 0x9d, 0xa5, 0x97, 0x90, 0xe1, 0x65, 0xc7, 0xb9, 0xb3, 0x96, + 0xd2, 0xf3, 0xaa, 0x5a, 0x51, 0xee, 0x04, 0xe3, 0xef, 0xa6, 0x57, 0x5b, + 0x6c, 0x6b, 0xac, 0x33, 0x16, 0x6a, 0x0a, 0xd9, 0x24, 0x28, 0x80, 0xa2, + 0x37, 0x8f, 0x75, 0x4c, 0xa2, 0x80, 0x30, 0x65, 0xb4, 0xb2, 0xca, 0x1a, + 0x6c, 0x61, 0x08, 0x48, 0x4a, 0x47, 0xb0, 0x54, 0x48, 0x16, 0x98, 0x90, + 0x66, 0x4b, 0x95, 0x19, 0xb2, 0x97, 0xa5, 0x2b, 0x69, 0xd2, 0x54, 0x4e, + 0x4e, 0x49, 0xe1, 0xd5, 0xc4, 0xd4, 0xea, 0x85, 0x02, 0xe9, 0x16, 0x7c, + 0xa9, 0xb1, 0xe3, 0x29, 0x4a, 0x72, 0x22, 0xc3, 0x6e, 0xe5, 0x24, 0x00, + 0xae, 0xc1, 0xdb, 0x4f, 0x02, 0xc9, 0xaf, 0x51, 0xfa, 0xbd, 0x74, 0xf8, + 0x57, 0x7e, 0x83, 0x45, 0x1a, 0x8f, 0xd5, 0xeb, 0xa7, 0xc2, 0xbb, 0xf4, + 0x1a, 0x29, 0x0c, 0x34, 0xe7, 0xab, 0xd6, 0xbf, 0x85, 0x6b, 0xe8, 0x15, + 0x5c, 0xe5, 0x22, 0xc8, 0xec, 0xe8, 0xec, 0xce, 0x88, 0xd9, 0x71, 0xd6, + 0x01, 0x4b, 0x89, 0x48, 0xc9, 0x28, 0xe3, 0x91, 0xee, 0x39, 0xfb, 0xea, + 0xc7, 0xa7, 0x3d, 0x5e, 0xb5, 0xfc, 0x2b, 0x5f, 0x40, 0xa6, 0x35, 0x25, + 0x36, 0xba, 0xa6, 0xa6, 0x88, 0xed, 0xa9, 0x5b, 0x07, 0x06, 0x7c, 0xf6, + 0x6b, 0xba, 0x68, 0xc0, 0x06, 0x98, 0xb7, 0x60, 0x7f, 0xa9, 0x4f, 0xe1, + 0x5b, 0xdd, 0xb7, 0xc2, 0x79, 0x65, 0x6e, 0xc3, 0x8e, 0xb5, 0x1e, 0xb5, + 0x34, 0x92, 0x7f, 0x0a, 0x9b, 0x11, 0x08, 0x69, 0x90, 0x86, 0xd2, 0x94, + 0x21, 0x3b, 0x82, 0x52, 0x30, 0x00, 0xaa, 0x1e, 0xa7, 0xd5, 0xab, 0xf4, + 0x2d, 0x25, 0x8e, 0x50, 0xfa, 0x4e, 0x91, 0xd1, 0x7e, 0x5b, 0xcf, 0x06, + 0xd3, 0x58, 0x1a, 0xcc, 0xd6, 0xb5, 0x0c, 0xd7, 0xce, 0x21, 0x18, 0xca, + 0x58, 0x9b, 0xc2, 0xfe, 0x4e, 0x9d, 0xb6, 0x97, 0x08, 0xc4, 0xd6, 0x2a, + 0xad, 0x81, 0x95, 0x96, 0xcb, 0x81, 0x0b, 0x28, 0x1c, 0x55, 0x8d, 0xc3, + 0xe7, 0x5a, 0xcd, 0x5d, 0xd3, 0xf4, 0xfa, 0xf5, 0x36, 0x2a, 0xa1, 0x67, + 0x2f, 0xe6, 0x3f, 0x92, 0x29, 0xdf, 0x2a, 0xe2, 0xe4, 0xe3, 0xc2, 0xfa, + 0x9a, 0xcd, 0x64, 0x91, 0x81, 0x5e, 0xd1, 0x5d, 0x37, 0x4b, 0xe8, 0x6b, + 0x45, 0x6f, 0x7a, 0x72, 0xdc, 0xfd, 0xb8, 0xf0, 0x67, 0xea, 0x75, 0xae, + 0xe8, 0xec, 0x4b, 0x08, 0xf1, 0x4a, 0x09, 0x49, 0x52, 0x88, 0x09, 0x03, + 0x24, 0x9e, 0x02, 0xab, 0xce, 0xeb, 0x0b, 0x52, 0x02, 0xd6, 0xd7, 0xa5, + 0xc9, 0x61, 0xbc, 0xed, 0xbe, 0xc4, 0x65, 0xad, 0xb4, 0xe3, 0x89, 0x2a, + 0x03, 0x18, 0xf7, 0x54, 0xfd, 0x50, 0x76, 0x74, 0xcd, 0xdc, 0xf6, 0x43, + 0x78, 0xff, 0x00, 0x21, 0xa8, 0xba, 0x1d, 0x23, 0xfa, 0x1f, 0x69, 0x04, + 0x02, 0x0c, 0x74, 0xee, 0xae, 0x8d, 0x25, 0xb7, 0x73, 0x33, 0x5b, 0x7b, + 0xb6, 0xa1, 0xa4, 0x49, 0xd1, 0x65, 0xc1, 0x44, 0xc8, 0xef, 0xa1, 0x71, + 0x56, 0x9d, 0xb0, 0xe0, 0x38, 0x18, 0xeb, 0xcf, 0x66, 0x3f, 0x85, 0x25, + 0x3a, 0xce, 0xce, 0x0e, 0xd7, 0x39, 0x23, 0xd1, 0x76, 0xb6, 0x3d, 0x2b, + 0xd1, 0xd7, 0xcc, 0xe7, 0x38, 0xfb, 0x78, 0xc7, 0xcf, 0x85, 0x73, 0xe4, + 0xc8, 0x7a, 0x3f, 0x27, 0x9a, 0x85, 0x98, 0xc5, 0x42, 0x32, 0x6e, 0x25, + 0xa4, 0x90, 0x78, 0x20, 0x91, 0x90, 0x3d, 0x9c, 0x3e, 0xfa, 0xea, 0x4a, + 0x8d, 0x11, 0x5a, 0x74, 0xc7, 0x08, 0x4f, 0xa1, 0x18, 0xdb, 0x21, 0x38, + 0xdd, 0xb1, 0xb3, 0xe5, 0x5e, 0xe5, 0x5c, 0x61, 0xe7, 0xe4, 0x8e, 0x36, + 0x4a, 0x7e, 0x3e, 0x0c, 0xa6, 0xde, 0x21, 0x43, 0x76, 0x02, 0x1e, 0x7b, + 0x7c, 0xe5, 0x86, 0xd8, 0x29, 0x05, 0x49, 0x59, 0x3c, 0x37, 0x8d, 0xdd, + 0x62, 0x8b, 0xc5, 0xe6, 0x0d, 0x9d, 0x31, 0xcc, 0xf7, 0xb9, 0xbe, 0x7d, + 0xc0, 0xd3, 0x7b, 0x89, 0xc9, 0x3f, 0x80, 0xf6, 0xd7, 0x3b, 0x8f, 0x1e, + 0x44, 0xde, 0x48, 0xe3, 0xc8, 0x05, 0x5e, 0x91, 0x01, 0xc3, 0x21, 0x85, + 0x75, 0x80, 0x85, 0x9f, 0xc0, 0x6d, 0x7d, 0xc2, 0xa6, 0x5f, 0x9b, 0x4e, + 0xb3, 0x6e, 0x53, 0xac, 0x0d, 0xa6, 0xe1, 0xdb, 0x52, 0xeb, 0x40, 0x75, + 0x3e, 0xe6, 0x17, 0x8f, 0x7e, 0xca, 0x00, 0xff, 0x00, 0xaa, 0x9f, 0x66, + 0x39, 0xe5, 0xf1, 0xee, 0x2e, 0xec, 0xb1, 0xc2, 0xe7, 0xd8, 0xbf, 0x4f, + 0x9c, 0xc4, 0x04, 0xb0, 0xa9, 0x0a, 0x23, 0x9e, 0x79, 0x2c, 0x20, 0x01, + 0x92, 0x56, 0xa3, 0x80, 0x2a, 0x25, 0xca, 0xff, 0x00, 0x06, 0x04, 0xaf, + 0x45, 0x59, 0x79, 0xe9, 0x5b, 0x3b, 0x65, 0x98, 0xec, 0xa9, 0xd5, 0x25, + 0x3d, 0xa4, 0x24, 0x1c, 0x0f, 0x7d, 0x57, 0x2c, 0x77, 0x1f, 0xe9, 0x24, + 0xdd, 0x3a, 0xa2, 0x76, 0x84, 0x58, 0xaa, 0x96, 0xff, 0x00, 0x67, 0x3b, + 0xfa, 0xa4, 0xfc, 0xf2, 0x16, 0x6a, 0x26, 0x91, 0xe9, 0x77, 0xae, 0xba, + 0x8e, 0x54, 0x23, 0x03, 0x9d, 0x5c, 0xe5, 0x36, 0xe7, 0xa4, 0x85, 0x95, + 0x00, 0x9c, 0xec, 0x81, 0xb3, 0xd4, 0x01, 0xc7, 0xca, 0x92, 0xa9, 0x2c, + 0xee, 0xf6, 0x3d, 0x77, 0x5b, 0xc6, 0xdf, 0x72, 0xf1, 0x6c, 0xb8, 0xc4, + 0xba, 0x43, 0x4c, 0xa8, 0x2f, 0x25, 0xd6, 0x15, 0x91, 0xb4, 0x37, 0x60, + 0x8e, 0x20, 0x83, 0xbc, 0x1f, 0x61, 0xa5, 0x87, 0x55, 0xdb, 0x57, 0x30, + 0x46, 0x86, 0x64, 0x4d, 0x73, 0x9c, 0x0d, 0xa9, 0x51, 0x99, 0x52, 0xd0, + 0x92, 0x4e, 0x37, 0xab, 0xec, 0xff, 0x00, 0x1a, 0x5f, 0x6f, 0xb3, 0xcc, + 0xb4, 0x5b, 0x35, 0x23, 0xd3, 0x1d, 0x61, 0x5e, 0x98, 0x1c, 0x90, 0x10, + 0xc6, 0xd0, 0x4a, 0x14, 0x52, 0x76, 0xb8, 0xf6, 0xee, 0xa9, 0x5c, 0x9c, + 0x24, 0x27, 0x45, 0x5a, 0xf0, 0x31, 0x94, 0x28, 0xff, 0x00, 0x3a, 0xab, + 0xcb, 0x8c, 0x52, 0x6f, 0xc8, 0xd4, 0xa4, 0xda, 0x5e, 0x06, 0x37, 0x4b, + 0xe4, 0x2b, 0x6c, 0x84, 0x47, 0x79, 0x4e, 0xbb, 0x29, 0x69, 0xda, 0x4b, + 0x0c, 0x34, 0xa7, 0x5c, 0x29, 0xed, 0xc2, 0x41, 0xc0, 0xf7, 0xd6, 0xeb, + 0x4d, 0xd2, 0x25, 0xda, 0x31, 0x7e, 0x0b, 0xa1, 0xc4, 0x05, 0x14, 0x28, + 0x10, 0x52, 0xa4, 0x28, 0x71, 0x0a, 0x07, 0x78, 0x3e, 0xfa, 0xa5, 0x58, + 0xba, 0x55, 0xfd, 0x63, 0xa9, 0xe4, 0x42, 0x30, 0xb9, 0xe4, 0x3c, 0x96, + 0x4f, 0xa5, 0x05, 0x12, 0x10, 0x33, 0xb3, 0xb3, 0xb3, 0xd4, 0x40, 0xfe, + 0x14, 0xc1, 0x9b, 0x65, 0xce, 0xcc, 0xc6, 0xa7, 0xba, 0x3c, 0xf4, 0x72, + 0xec, 0xa8, 0xe5, 0xe4, 0x37, 0x1f, 0x68, 0x04, 0x38, 0x94, 0x2b, 0x78, + 0xcf, 0x6d, 0x37, 0x5c, 0x57, 0x19, 0xe7, 0x81, 0x2b, 0x24, 0xf9, 0xc7, + 0x03, 0x49, 0x1a, 0xae, 0xda, 0xd4, 0xb7, 0xa3, 0xb5, 0xe9, 0x52, 0x54, + 0xc1, 0xc3, 0xca, 0x8d, 0x1d, 0x6e, 0xa5, 0xb3, 0xed, 0x20, 0x52, 0x9e, + 0x4f, 0x24, 0x35, 0x32, 0xe1, 0xa9, 0x24, 0xc7, 0x5e, 0xdb, 0x2e, 0xcd, + 0xda, 0x42, 0xb1, 0x8c, 0x8c, 0x1c, 0x71, 0xa9, 0x9c, 0x99, 0xb2, 0xd3, + 0x3a, 0x32, 0x02, 0x9a, 0xc1, 0x2e, 0xed, 0xb8, 0xb5, 0x75, 0x95, 0x6d, + 0x10, 0x73, 0xee, 0xc6, 0x3e, 0x55, 0xa3, 0x40, 0xef, 0xb8, 0x6a, 0x83, + 0xff, 0x00, 0x33, 0x70, 0x7d, 0xc6, 0x9b, 0x4a, 0x2a, 0x49, 0x7b, 0x7f, + 0xd1, 0x26, 0xdb, 0x8b, 0x7e, 0xe3, 0xed, 0x47, 0xea, 0xf5, 0xd3, 0xe1, + 0x5d, 0xfa, 0x0d, 0x14, 0x6a, 0x3f, 0x57, 0xae, 0x9f, 0x0a, 0xef, 0xd0, + 0x68, 0xa8, 0x09, 0xc3, 0x4e, 0x7a, 0xbd, 0x6b, 0xf8, 0x56, 0xbe, 0x81, + 0x4c, 0x6a, 0xbd, 0xa7, 0xaf, 0x76, 0xa4, 0xd8, 0x2d, 0x89, 0x55, 0xce, + 0x08, 0x50, 0x8a, 0xd0, 0x20, 0xc8, 0x46, 0x41, 0xd8, 0x1e, 0xda, 0x61, + 0xd3, 0x96, 0x9e, 0xf4, 0x81, 0xe2, 0x11, 0xe7, 0x40, 0x0c, 0x6b, 0x63, + 0x4a, 0x03, 0x20, 0xf5, 0xd2, 0xae, 0x9c, 0xb4, 0xf7, 0xa4, 0x0f, 0x10, + 0x8f, 0x3a, 0x3a, 0x72, 0xd3, 0xde, 0x90, 0x3c, 0x42, 0x3c, 0xea, 0xbe, + 0xaf, 0x4d, 0x1d, 0x55, 0x4e, 0xa9, 0xf8, 0x64, 0x95, 0x58, 0xeb, 0x92, + 0x92, 0x1c, 0x9a, 0xc4, 0xd2, 0x91, 0x7e, 0xb5, 0x0e, 0x17, 0x58, 0x3e, + 0x21, 0x1e, 0x75, 0xef, 0x4f, 0xda, 0xbb, 0xd6, 0x07, 0x88, 0x47, 0x9d, + 0x72, 0x96, 0x7a, 0x6f, 0x50, 0x9f, 0xe8, 0x92, 0x6b, 0xee, 0xbf, 0xa6, + 0x69, 0x47, 0xa8, 0x43, 0xdd, 0x32, 0xd5, 0x1a, 0xee, 0x86, 0x6d, 0x6a, + 0x8c, 0x59, 0x25, 0x7b, 0x25, 0x20, 0xf5, 0x1c, 0xe7, 0x8f, 0xdf, 0x49, + 0x4d, 0x2e, 0xe9, 0xdb, 0x4f, 0x7a, 0x41, 0xf1, 0x08, 0xf3, 0xaf, 0x3a, + 0x72, 0xd3, 0xde, 0x90, 0x3c, 0x42, 0x3c, 0xeb, 0x73, 0x45, 0xd3, 0xac, + 0xaa, 0x51, 0xb2, 0xf9, 0xee, 0x71, 0x58, 0x58, 0x5c, 0x25, 0xfd, 0x94, + 0xee, 0xd4, 0x46, 0x49, 0xc6, 0x0b, 0x09, 0xf2, 0xc6, 0x34, 0x52, 0xee, + 0x9c, 0xb4, 0xf7, 0xa4, 0x0f, 0x10, 0x8f, 0x3a, 0x3a, 0x72, 0xd3, 0xde, + 0x90, 0x3c, 0x42, 0x3c, 0xeb, 0x5c, 0xaa, 0x6a, 0xd5, 0xc7, 0x1a, 0x56, + 0xf0, 0x7f, 0xe1, 0x1d, 0x1f, 0xc8, 0x6a, 0xb7, 0xa7, 0x2d, 0x57, 0xb7, + 0xb4, 0xb5, 0xbd, 0x10, 0xef, 0x81, 0x88, 0xcf, 0x47, 0x49, 0xc2, 0xa3, + 0x05, 0x2d, 0xa0, 0x47, 0x04, 0xa8, 0x11, 0xf2, 0xce, 0xf1, 0xdb, 0x56, + 0x1b, 0x95, 0xc2, 0xcb, 0x70, 0xb7, 0xc8, 0x86, 0xfd, 0xd6, 0x18, 0x69, + 0xf6, 0xcb, 0x6a, 0x29, 0x92, 0x80, 0x70, 0x46, 0x37, 0x6f, 0xaf, 0x2d, + 0x77, 0x0b, 0x35, 0xba, 0xdb, 0x1a, 0x1b, 0x57, 0x68, 0x4a, 0x6d, 0x86, + 0xd2, 0xda, 0x54, 0xa9, 0x08, 0xc9, 0x00, 0x63, 0x7e, 0xfa, 0x92, 0x33, + 0xdb, 0x1c, 0x22, 0x37, 0x0c, 0xcb, 0x2c, 0x20, 0xe9, 0xab, 0x7c, 0x5d, + 0x38, 0xab, 0x28, 0x6c, 0xb9, 0x11, 0x69, 0x21, 0xc2, 0xa3, 0xfa, 0x4b, + 0x27, 0x8a, 0x89, 0xed, 0xcf, 0xdd, 0x81, 0x4b, 0xc6, 0x9c, 0xba, 0x1b, + 0x77, 0x45, 0xae, 0xf8, 0x4d, 0xb7, 0x67, 0x9b, 0x38, 0x8e, 0x03, 0xc5, + 0xbe, 0x1b, 0x1b, 0x79, 0xc7, 0x0d, 0xd9, 0xc5, 0x39, 0xe9, 0xcb, 0x4f, + 0x7a, 0x40, 0xf1, 0x08, 0xf3, 0xa3, 0xa7, 0x2d, 0x3d, 0xe9, 0x03, 0xc4, + 0x23, 0xce, 0x92, 0xb2, 0x43, 0xed, 0xc4, 0xde, 0xc5, 0xbe, 0x33, 0x16, + 0xc4, 0xdb, 0xda, 0x68, 0x26, 0x22, 0x5b, 0xe6, 0x82, 0x3f, 0xdd, 0xc6, + 0x29, 0x66, 0x8f, 0xd3, 0xcd, 0xe9, 0xbb, 0x5a, 0xa2, 0x21, 0xe2, 0xfa, + 0xd6, 0xe1, 0x71, 0x6e, 0x14, 0xec, 0xe7, 0x70, 0x00, 0x63, 0xd8, 0x05, + 0x4c, 0xe9, 0xcb, 0x4f, 0x7a, 0x40, 0xf1, 0x08, 0xf3, 0xa3, 0xa7, 0x2d, + 0x3d, 0xe9, 0x03, 0xc4, 0x23, 0xce, 0x96, 0xe7, 0x86, 0xbe, 0x47, 0xb5, + 0x65, 0x32, 0x06, 0x93, 0xd3, 0x2c, 0x69, 0xd5, 0x4f, 0x2c, 0x39, 0xce, + 0x19, 0x4e, 0xed, 0x8f, 0xd1, 0xc6, 0xc2, 0x01, 0x3b, 0x29, 0xf6, 0xe3, + 0x27, 0x7d, 0x6b, 0x93, 0xa7, 0x64, 0xb1, 0x78, 0x91, 0x71, 0xb1, 0x4f, + 0x4c, 0x27, 0x24, 0xe3, 0xd2, 0x19, 0x71, 0x9e, 0x71, 0xb7, 0x08, 0xff, + 0x00, 0x4b, 0x19, 0x18, 0x34, 0xcf, 0xa7, 0x2d, 0x3d, 0xe9, 0x03, 0xc4, + 0x23, 0xce, 0x8e, 0x9c, 0xb4, 0xf7, 0xa4, 0x0f, 0x10, 0x8f, 0x3a, 0x7d, + 0xc9, 0x67, 0x22, 0xed, 0xc7, 0x18, 0x34, 0xb3, 0x6c, 0x98, 0xab, 0x64, + 0xd8, 0xd3, 0xee, 0x6b, 0x94, 0xf4, 0x96, 0xd4, 0x8e, 0x70, 0xb2, 0x94, + 0x25, 0xbc, 0x82, 0x3f, 0x45, 0x23, 0xdf, 0xd6, 0x6b, 0x76, 0x9e, 0xb6, + 0xf4, 0x3d, 0x96, 0x2c, 0x0e, 0x77, 0x9e, 0xe6, 0x13, 0xb3, 0xb7, 0xb3, + 0xb3, 0xb5, 0xbc, 0x9e, 0x19, 0x3d, 0xb4, 0x74, 0xe5, 0xa7, 0xbd, 0x20, + 0x78, 0x84, 0x79, 0xd1, 0xd3, 0x96, 0x9e, 0xf4, 0x81, 0xe2, 0x11, 0xe7, + 0x49, 0xc9, 0xb5, 0x81, 0xa8, 0xa4, 0xf2, 0x2e, 0xb8, 0xe9, 0xe7, 0xfa, + 0x65, 0x57, 0x5b, 0x2c, 0xe1, 0x06, 0x5b, 0xa8, 0x08, 0x7d, 0x2b, 0x6b, + 0x9c, 0x6d, 0xe0, 0x38, 0x12, 0x32, 0x30, 0x7d, 0xb4, 0xc2, 0xd7, 0x06, + 0x6b, 0x29, 0x78, 0xdd, 0x2e, 0x06, 0x72, 0x9d, 0x18, 0xd9, 0xe6, 0x52, + 0xdb, 0x68, 0x1b, 0xf7, 0x00, 0x32, 0x7a, 0xfa, 0xc9, 0xaf, 0x7a, 0x72, + 0xd3, 0xde, 0x90, 0x3c, 0x42, 0x3c, 0xe8, 0xe9, 0xcb, 0x4f, 0x7a, 0x40, + 0xf1, 0x08, 0xf3, 0xa1, 0xcd, 0xb5, 0x86, 0x0a, 0x09, 0x3c, 0xa1, 0x45, + 0xbb, 0x4e, 0x5c, 0x2d, 0x09, 0x76, 0x35, 0xa2, 0xee, 0x96, 0x6d, 0xeb, + 0x51, 0x5a, 0x1a, 0x76, 0x3f, 0x3a, 0xa6, 0x73, 0xc4, 0x25, 0x5b, 0x43, + 0x77, 0xbc, 0x1a, 0x6d, 0x62, 0xb3, 0xc7, 0xb3, 0xb2, 0xf2, 0x58, 0x53, + 0x8e, 0x3a, 0xfb, 0x85, 0xe7, 0x9d, 0x70, 0xe5, 0x4e, 0x2c, 0xf1, 0x3b, + 0xb7, 0x0f, 0x70, 0xaf, 0x7a, 0x72, 0xd3, 0xde, 0x90, 0x3c, 0x42, 0x3c, + 0xe8, 0xe9, 0xcb, 0x4f, 0x7a, 0x40, 0xf1, 0x08, 0xf3, 0xa6, 0xe7, 0x29, + 0x79, 0x05, 0x04, 0xbc, 0x06, 0xa3, 0xf5, 0x7a, 0xe9, 0xf0, 0xae, 0xfd, + 0x06, 0x8a, 0x5f, 0xa8, 0x6f, 0x76, 0xa5, 0x58, 0x2e, 0x69, 0x4d, 0xce, + 0x09, 0x51, 0x8a, 0xe8, 0x00, 0x48, 0x46, 0x49, 0xd8, 0x3e, 0xda, 0x2b, + 0xc1, 0xe8, 0xff, 0xd9 +}; + +unsigned int image_size = 4096; + + +int main() { + + lxw_workbook *workbook = workbook_new("test_background52.xlsx"); + lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL); + + worksheet_set_background_buffer(worksheet, image_buffer, image_size); + + return workbook_close(workbook); +} diff --git a/test/functional/test_background.py b/test/functional/test_background.py new file mode 100644 index 00000000..7dca7daa --- /dev/null +++ b/test/functional/test_background.py @@ -0,0 +1,44 @@ +############################################################################### +# +# Tests for libxlsxwriter. +# +# Copyright 2014-2021, John McNamara, jmcnamara@cpan.org +# +import os +import pytest +import base_test_class + +class TestCompareXLSXFiles(base_test_class.XLSXBaseTest): + """ + Test file created with libxlsxwriter against a file created by Excel. + + """ + + def test_background01(self): + self.run_exe_test('test_background01') + + def test_background02(self): + self.run_exe_test('test_background02') + + def test_background03(self): + self.run_exe_test('test_background03') + + @pytest.mark.skipif(os.environ.get('USE_NO_MD5'), reason="compiled without MD5 support") + def test_background04(self): + self.run_exe_test('test_background04') + + def test_background05(self): + self.run_exe_test('test_background05') + + def test_background06(self): + self.ignore_elements = {'xl/worksheets/sheet1.xml': ['