Initial working chart line formatting.

This commit is contained in:
John McNamara 2016-12-31 02:35:38 +00:00
parent 038cece4ad
commit 9e645b2a5c
8 changed files with 223 additions and 26 deletions

1
.indent.pro vendored
View File

@ -53,6 +53,7 @@
-T lxw_chart_font
-T lxw_chart_legend
-T lxw_chart_legend_position
-T lxw_chart_line
-T lxw_chart_series
-T lxw_chart_title
-T lxw_chart_type

View File

@ -225,6 +225,16 @@ typedef struct lxw_series_data_point {
} lxw_series_data_point;
typedef struct lxw_chart_line {
lxw_color_t color;
uint8_t none;
uint8_t dash_type;
uint8_t has_color;
float width;
} lxw_chart_line;
typedef struct lxw_chart_fill {
lxw_color_t color;
@ -305,6 +315,7 @@ typedef struct lxw_chart_series {
lxw_series_range *categories;
lxw_series_range *values;
lxw_chart_title title;
lxw_chart_line *line;
STAILQ_ENTRY (lxw_chart_series) list_pointers;
@ -605,6 +616,8 @@ void chart_series_set_name_range(lxw_chart_series *series,
const char *sheetname, lxw_row_t row,
lxw_col_t col);
void chart_series_set_line(lxw_chart_series *series, lxw_chart_line *line);
/**
* @brief Set the name caption of the an axis.
*

View File

@ -56,6 +56,7 @@ _chart_series_free(lxw_chart_series *series)
return;
free(series->title.name);
free(series->line);
_chart_free_range(series->categories);
_chart_free_range(series->values);
@ -242,6 +243,28 @@ _chart_convert_font_args(lxw_chart_font *user_font)
return font;
}
/*
* Create a copy of a user supplied line.
*/
STATIC lxw_chart_line *
_chart_convert_line_args(lxw_chart_line *user_line)
{
lxw_chart_line *line;
if (!user_line)
return NULL;
line = calloc(1, sizeof(struct lxw_chart_line));
RETURN_ON_MEM_ERROR(line, NULL);
memcpy(line, user_line, sizeof(lxw_chart_line));
if (line->color)
line->has_color = LXW_TRUE;
return line;
}
/*
* Add unique ids for primary or secondary axes.
*/
@ -519,15 +542,14 @@ _chart_write_a_srgb_clr(lxw_chart *self, lxw_color_t color,
* Write the <a:solidFill> element.
*/
STATIC void
_chart_write_a_solid_fill(lxw_chart *self, lxw_chart_fill *fill)
_chart_write_a_solid_fill(lxw_chart *self, lxw_color_t color,
uint8_t transparency)
{
lxw_xml_start_tag(self->file, "a:solidFill", NULL);
if (fill->has_color) {
/* Write the a:srgbClr element. */
_chart_write_a_srgb_clr(self, fill->color, fill->transparency);
}
/* Write the a:srgbClr element. */
_chart_write_a_srgb_clr(self, color, transparency);
lxw_xml_end_tag(self->file, "a:solidFill");
}
@ -600,11 +622,7 @@ _chart_write_a_def_rpr(lxw_chart *self, lxw_chart_font *font)
lxw_xml_start_tag(self->file, "a:defRPr", &attributes);
if (has_color) {
lxw_chart_fill fill = { 0, 0, 0 };
fill.color = font->color;
fill.has_color = LXW_TRUE;
_chart_write_a_solid_fill(self, &fill);
_chart_write_a_solid_fill(self, font->color, LXW_FALSE);
}
if (has_latin) {
@ -676,11 +694,7 @@ _chart_write_a_r_pr(lxw_chart *self, lxw_chart_font *font)
lxw_xml_start_tag(self->file, "a:rPr", &attributes);
if (has_color) {
lxw_chart_fill fill = { 0, 0, 0 };
fill.color = font->color;
fill.has_color = LXW_TRUE;
_chart_write_a_solid_fill(self, &fill);
_chart_write_a_solid_fill(self, font->color, LXW_FALSE);
}
if (has_latin) {
@ -1330,19 +1344,35 @@ _chart_write_a_no_fill(lxw_chart *self)
* Write the <a:ln> element.
*/
STATIC void
_chart_write_a_ln(lxw_chart *self)
_chart_write_a_ln(lxw_chart *self, lxw_chart_line *line)
{
struct xml_attribute_list attributes;
struct xml_attribute *attribute;
char w[] = "28575";
float width_flt = line->width;
uint32_t width_int;
LXW_INIT_ATTRIBUTES();
LXW_PUSH_ATTRIBUTES_STR("w", w);
/* Round width to nearest 0.25, like Excel. */
width_flt = (float) (uint32_t) ((line->width + 0.125) * 4.0) / 4.0;
/* Convert to internal units. */
width_int = (uint32_t) (0.5 + (12700.0 * width_flt));
if (width_int)
LXW_PUSH_ATTRIBUTES_INT("w", width_int);
lxw_xml_start_tag(self->file, "a:ln", &attributes);
/* Write the a:noFill element. */
_chart_write_a_no_fill(self);
/* Write the line fill. */
if (line->none) {
/* Write the a:noFill element. */
_chart_write_a_no_fill(self);
}
else if (line->has_color) {
/* Write the a:solidFill element. */
_chart_write_a_solid_fill(self, line->color, LXW_FALSE);
}
lxw_xml_end_tag(self->file, "a:ln");
@ -1353,12 +1383,16 @@ _chart_write_a_ln(lxw_chart *self)
* Write the <c:spPr> element.
*/
STATIC void
_chart_write_sp_pr(lxw_chart *self)
_chart_write_sp_pr(lxw_chart *self, lxw_chart_series *series)
{
if (!series->line)
return;
lxw_xml_start_tag(self->file, "c:spPr", NULL);
/* Write the a:ln element. */
_chart_write_a_ln(self);
_chart_write_a_ln(self, series->line);
lxw_xml_end_tag(self->file, "c:spPr");
}
@ -1627,6 +1661,9 @@ _chart_write_ser(lxw_chart *self, lxw_chart_series *series)
/* Write the series name. */
_chart_write_series_name(self, series);
/* Write the c:spPr element. */
_chart_write_sp_pr(self, series);
/* Write the c:marker element. */
_chart_write_marker(self);
@ -1656,10 +1693,11 @@ _chart_write_xval_ser(lxw_chart *self, lxw_chart_series *series)
/* Write the c:order element. */
_chart_write_order(self, index);
if (self->type == LXW_CHART_SCATTER) {
/* Write the c:spPr element. */
_chart_write_sp_pr(self);
}
/* Write the series name. */
_chart_write_series_name(self, series);
/* Write the c:spPr element. */
_chart_write_sp_pr(self, series);
if (self->type == LXW_CHART_SCATTER_STRAIGHT
|| self->type == LXW_CHART_SCATTER_SMOOTH) {
@ -2501,6 +2539,17 @@ _chart_write_scatter_chart(lxw_chart *self)
_chart_write_scatter_style(self);
STAILQ_FOREACH(series, self->series_list, list_pointers) {
/* Add default scatter chart formatting to the series data unless
* it has already been specified by the user.*/
if (self->type == LXW_CHART_SCATTER) {
if (!series->line) {
lxw_chart_line line =
{ 0x000000, LXW_TRUE, 0x0, LXW_FALSE, 2.25 };
series->line = _chart_convert_line_args(&line);
}
}
/* Write the c:ser element. */
_chart_write_xval_ser(self, series);
}
@ -3043,6 +3092,18 @@ chart_series_set_values(lxw_chart_series *series, const char *sheetname,
first_row, first_col, last_row, last_col);
}
/*
* Set a line type for a series.
*/
void
chart_series_set_line(lxw_chart_series *series, lxw_chart_line *line)
{
if (!line)
return;
series->line = _chart_convert_line_args(line);
}
/*
* Set an axis caption.
*/

View File

@ -0,0 +1,48 @@
/*****************************************************************************
* Test cases for libxlsxwriter.
*
* Test to compare output against Excel files.
*
* Copyright 2014-2017, John McNamara, jmcnamara@cpan.org
*
*/
#include "xlsxwriter.h"
int main() {
lxw_workbook *workbook = new_workbook("test_chart_format01.xlsx");
lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL);
lxw_chart *chart = workbook_add_chart(workbook, LXW_CHART_LINE);
/* For testing, copy the randomly generated axis ids in the target file. */
chart->axis_id_1 = 46335872;
chart->axis_id_2 = 46365696;
uint8_t data[5][3] = {
{1, 2, 3},
{2, 4, 6},
{3, 6, 9},
{4, 8, 12},
{5, 10, 15}
};
int row, col;
for (row = 0; row < 5; row++)
for (col = 0; col < 3; col++)
worksheet_write_number(worksheet, row, col, data[row][col], NULL);
chart_add_series(chart,
"=Sheet1!$A$1:$A$5",
"=Sheet1!$B$1:$B$5"
);
chart_add_series(chart,
"=Sheet1!$A$1:$A$5",
"=Sheet1!$C$1:$C$5"
);
worksheet_insert_chart(worksheet, CELL("E9"), chart);
return workbook_close(workbook);
}

View File

@ -0,0 +1,54 @@
/*****************************************************************************
* Test cases for libxlsxwriter.
*
* Test to compare output against Excel files.
*
* Copyright 2014-2017, John McNamara, jmcnamara@cpan.org
*
*/
#include "xlsxwriter.h"
int main() {
lxw_workbook *workbook = new_workbook("test_chart_format02.xlsx");
lxw_worksheet *worksheet = workbook_add_worksheet(workbook, NULL);
lxw_chart *chart = workbook_add_chart(workbook, LXW_CHART_LINE);
/* For testing, copy the randomly generated axis ids in the target file. */
chart->axis_id_1 = 46335872;
chart->axis_id_2 = 46365696;
uint8_t data[5][3] = {
{1, 2, 3},
{2, 4, 6},
{3, 6, 9},
{4, 8, 12},
{5, 10, 15}
};
int row, col;
for (row = 0; row < 5; row++)
for (col = 0; col < 3; col++)
worksheet_write_number(worksheet, row, col, data[row][col], NULL);
lxw_chart_series *series1 = chart_add_series(chart,
"=Sheet1!$A$1:$A$5",
"=Sheet1!$B$1:$B$5"
);
chart_add_series(chart,
"=Sheet1!$A$1:$A$5",
"=Sheet1!$C$1:$C$5"
);
/* 'line': { 'color': "red" }, */
lxw_chart_line line = {.color = LXW_COLOR_RED};
chart_series_set_line(series1, &line);
worksheet_insert_chart(worksheet, CELL("E9"), chart);
return workbook_close(workbook);
}

View File

@ -0,0 +1,20 @@
###############################################################################
#
# Tests for libxlsxwriter.
#
# Copyright 2014-2017, John McNamara, jmcnamara@cpan.org
#
import base_test_class
class TestCompareXLSXFiles(base_test_class.XLSXBaseTest):
"""
Test file created with libxlsxwriter against a file created by Excel.
"""
def test_chart_format01(self):
self.run_exe_test('test_chart_format01')
def test_chart_format02(self):
self.run_exe_test('test_chart_format02')

Binary file not shown.

Binary file not shown.