write.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. /*
  2. +----------------------------------------------------------------------+
  3. | Vtiful Extension |
  4. +----------------------------------------------------------------------+
  5. | Copyright (c) 2017-2017 The Viest |
  6. +----------------------------------------------------------------------+
  7. | http://www.vtiful.com |
  8. +----------------------------------------------------------------------+
  9. | Author: viest <[email protected]> |
  10. +----------------------------------------------------------------------+
  11. */
  12. #include "php.h"
  13. #include "excel.h"
  14. #include "write.h"
  15. #include "xlsxwriter.h"
  16. #include "xlsxwriter/packager.h"
  17. STATIC void _prepare_defined_names(lxw_workbook *self);
  18. STATIC void _prepare_drawings(lxw_workbook *self);
  19. STATIC void _add_chart_cache_data(lxw_workbook *self);
  20. /*
  21. * According to the zval type written to the file
  22. */
  23. void type_writer(zval *value, zend_long row, zend_long columns, excel_resource_t *res)
  24. {
  25. switch (Z_TYPE_P(value)) {
  26. case IS_STRING:
  27. worksheet_write_string(res->worksheet, row, columns, ZSTR_VAL(zval_get_string(value)), NULL);
  28. break;
  29. case IS_LONG:
  30. worksheet_write_number(res->worksheet, row, columns, zval_get_long(value), NULL);
  31. break;
  32. case IS_DOUBLE:
  33. worksheet_write_string(res->worksheet, row, columns, ZSTR_VAL(zval_get_string(value)), NULL);
  34. break;
  35. }
  36. }
  37. /*
  38. * Call finalization code and close file.
  39. */
  40. lxw_error
  41. workbook_file(excel_resource_t *self, zval *handle)
  42. {
  43. lxw_worksheet *worksheet = NULL;
  44. lxw_packager *packager = NULL;
  45. lxw_error error = LXW_NO_ERROR;
  46. /* Add a default worksheet if non have been added. */
  47. if (!self->workbook->num_sheets)
  48. workbook_add_worksheet(self, NULL);
  49. /* Ensure that at least one worksheet has been selected. */
  50. if (self->workbook->active_sheet == 0) {
  51. worksheet = STAILQ_FIRST(self->workbook->worksheets);
  52. worksheet->selected = 1;
  53. worksheet->hidden = 0;
  54. }
  55. /* Set the active sheet. */
  56. STAILQ_FOREACH(worksheet, self->workbook->worksheets, list_pointers) {
  57. if (worksheet->index == self->workbook->active_sheet)
  58. worksheet->active = 1;
  59. }
  60. /* Set the defined names for the worksheets such as Print Titles. */
  61. _prepare_defined_names(self->workbook);
  62. /* Prepare the drawings, charts and images. */
  63. _prepare_drawings(self->workbook);
  64. /* Add cached data to charts. */
  65. _add_chart_cache_data(self->workbook);
  66. /* Create a packager object to assemble sub-elements into a zip file. */
  67. packager = lxw_packager_new(self->workbook->filename, self->workbook->options.tmpdir);
  68. /* If the packager fails it is generally due to a zip permission error. */
  69. if (packager == NULL) {
  70. fprintf(stderr, "[ERROR] workbook_close(): "
  71. "Error creating '%s'. "
  72. "Error = %s\n", self->workbook->filename, strerror(errno));
  73. error = LXW_ERROR_CREATING_XLSX_FILE;
  74. goto mem_error;
  75. }
  76. /* Set the workbook object in the packager. */
  77. packager->workbook = self->workbook;
  78. /* Assemble all the sub-files in the xlsx package. */
  79. error = lxw_create_package(packager);
  80. /* Error and non-error conditions fall through to the cleanup code. */
  81. if (error == LXW_ERROR_CREATING_TMPFILE) {
  82. fprintf(stderr, "[ERROR] workbook_close(): "
  83. "Error creating tmpfile(s) to assemble '%s'. "
  84. "Error = %s\n", self->workbook->filename, strerror(errno));
  85. }
  86. /* If LXW_ERROR_ZIP_FILE_OPERATION then errno is set by zlib. */
  87. if (error == LXW_ERROR_ZIP_FILE_OPERATION) {
  88. fprintf(stderr, "[ERROR] workbook_close(): "
  89. "Zlib error while creating xlsx file '%s'. "
  90. "Error = %s\n", self->workbook->filename, strerror(errno));
  91. }
  92. /* The next 2 error conditions don't set errno. */
  93. if (error == LXW_ERROR_ZIP_FILE_ADD) {
  94. fprintf(stderr, "[ERROR] workbook_close(): "
  95. "Zlib error adding file to xlsx file '%s'.\n",
  96. self->workbook->filename);
  97. }
  98. if (error == LXW_ERROR_ZIP_CLOSE) {
  99. fprintf(stderr, "[ERROR] workbook_close(): "
  100. "Zlib error closing xlsx file '%s'.\n", self->workbook->filename);
  101. }
  102. mem_error:
  103. if (handle) {
  104. zend_list_close(Z_RES_P(handle));
  105. lxw_packager_free(packager);
  106. lxw_workbook_free(self->workbook);
  107. }
  108. return error;
  109. }
  110. void _php_vtiful_excel_close(zend_resource *rsrc TSRMLS_DC)
  111. {
  112. //Here to PHP for gc
  113. }
  114. /*
  115. * Iterate through the worksheets and store any defined names used for print
  116. * ranges or repeat rows/columns.
  117. */
  118. STATIC void
  119. _prepare_defined_names(lxw_workbook *self)
  120. {
  121. lxw_worksheet *worksheet;
  122. char app_name[LXW_DEFINED_NAME_LENGTH];
  123. char range[LXW_DEFINED_NAME_LENGTH];
  124. char area[LXW_MAX_CELL_RANGE_LENGTH];
  125. char first_col[8];
  126. char last_col[8];
  127. STAILQ_FOREACH(worksheet, self->worksheets, list_pointers) {
  128. /*
  129. * Check for autofilter settings and store them.
  130. */
  131. if (worksheet->autofilter.in_use) {
  132. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  133. "%s!_FilterDatabase", worksheet->quoted_name);
  134. lxw_rowcol_to_range_abs(area,
  135. worksheet->autofilter.first_row,
  136. worksheet->autofilter.first_col,
  137. worksheet->autofilter.last_row,
  138. worksheet->autofilter.last_col);
  139. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH, "%s!%s",
  140. worksheet->quoted_name, area);
  141. /* Autofilters are the only defined name to set the hidden flag. */
  142. _store_defined_name(self, "_xlnm._FilterDatabase", app_name,
  143. range, worksheet->index, LXW_TRUE);
  144. }
  145. /*
  146. * Check for Print Area settings and store them.
  147. */
  148. if (worksheet->print_area.in_use) {
  149. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  150. "%s!Print_Area", worksheet->quoted_name);
  151. /* Check for print area that is the max row range. */
  152. if (worksheet->print_area.first_row == 0
  153. && worksheet->print_area.last_row == LXW_ROW_MAX - 1) {
  154. lxw_col_to_name(first_col,
  155. worksheet->print_area.first_col, LXW_FALSE);
  156. lxw_col_to_name(last_col,
  157. worksheet->print_area.last_col, LXW_FALSE);
  158. lxw_snprintf(area, LXW_MAX_CELL_RANGE_LENGTH - 1, "$%s:$%s",
  159. first_col, last_col);
  160. }
  161. /* Check for print area that is the max column range. */
  162. else if (worksheet->print_area.first_col == 0
  163. && worksheet->print_area.last_col == LXW_COL_MAX - 1) {
  164. lxw_snprintf(area, LXW_MAX_CELL_RANGE_LENGTH - 1, "$%d:$%d",
  165. worksheet->print_area.first_row + 1,
  166. worksheet->print_area.last_row + 1);
  167. }
  168. else {
  169. lxw_rowcol_to_range_abs(area,
  170. worksheet->print_area.first_row,
  171. worksheet->print_area.first_col,
  172. worksheet->print_area.last_row,
  173. worksheet->print_area.last_col);
  174. }
  175. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH, "%s!%s",
  176. worksheet->quoted_name, area);
  177. _store_defined_name(self, "_xlnm.Print_Area", app_name,
  178. range, worksheet->index, LXW_FALSE);
  179. }
  180. /*
  181. * Check for repeat rows/cols. aka, Print Titles and store them.
  182. */
  183. if (worksheet->repeat_rows.in_use || worksheet->repeat_cols.in_use) {
  184. if (worksheet->repeat_rows.in_use
  185. && worksheet->repeat_cols.in_use) {
  186. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  187. "%s!Print_Titles", worksheet->quoted_name);
  188. lxw_col_to_name(first_col,
  189. worksheet->repeat_cols.first_col, LXW_FALSE);
  190. lxw_col_to_name(last_col,
  191. worksheet->repeat_cols.last_col, LXW_FALSE);
  192. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  193. "%s!$%s:$%s,%s!$%d:$%d",
  194. worksheet->quoted_name, first_col,
  195. last_col, worksheet->quoted_name,
  196. worksheet->repeat_rows.first_row + 1,
  197. worksheet->repeat_rows.last_row + 1);
  198. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  199. range, worksheet->index, LXW_FALSE);
  200. }
  201. else if (worksheet->repeat_rows.in_use) {
  202. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  203. "%s!Print_Titles", worksheet->quoted_name);
  204. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  205. "%s!$%d:$%d", worksheet->quoted_name,
  206. worksheet->repeat_rows.first_row + 1,
  207. worksheet->repeat_rows.last_row + 1);
  208. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  209. range, worksheet->index, LXW_FALSE);
  210. }
  211. else if (worksheet->repeat_cols.in_use) {
  212. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  213. "%s!Print_Titles", worksheet->quoted_name);
  214. lxw_col_to_name(first_col,
  215. worksheet->repeat_cols.first_col, LXW_FALSE);
  216. lxw_col_to_name(last_col,
  217. worksheet->repeat_cols.last_col, LXW_FALSE);
  218. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  219. "%s!$%s:$%s", worksheet->quoted_name,
  220. first_col, last_col);
  221. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  222. range, worksheet->index, LXW_FALSE);
  223. }
  224. }
  225. }
  226. }
  227. /*
  228. * Iterate through the worksheets and set up any chart or image drawings.
  229. */
  230. STATIC void
  231. _prepare_drawings(lxw_workbook *self)
  232. {
  233. lxw_worksheet *worksheet;
  234. lxw_image_options *image_options;
  235. uint16_t chart_ref_id = 0;
  236. uint16_t image_ref_id = 0;
  237. uint16_t drawing_id = 0;
  238. STAILQ_FOREACH(worksheet, self->worksheets, list_pointers) {
  239. if (STAILQ_EMPTY(worksheet->image_data)
  240. && STAILQ_EMPTY(worksheet->chart_data))
  241. continue;
  242. drawing_id++;
  243. STAILQ_FOREACH(image_options, worksheet->chart_data, list_pointers) {
  244. chart_ref_id++;
  245. lxw_worksheet_prepare_chart(worksheet, chart_ref_id, drawing_id,
  246. image_options);
  247. if (image_options->chart)
  248. STAILQ_INSERT_TAIL(self->ordered_charts, image_options->chart,
  249. ordered_list_pointers);
  250. }
  251. STAILQ_FOREACH(image_options, worksheet->image_data, list_pointers) {
  252. if (image_options->image_type == LXW_IMAGE_PNG)
  253. self->has_png = LXW_TRUE;
  254. if (image_options->image_type == LXW_IMAGE_JPEG)
  255. self->has_jpeg = LXW_TRUE;
  256. if (image_options->image_type == LXW_IMAGE_BMP)
  257. self->has_bmp = LXW_TRUE;
  258. image_ref_id++;
  259. lxw_worksheet_prepare_image(worksheet, image_ref_id, drawing_id,
  260. image_options);
  261. }
  262. }
  263. self->drawing_count = drawing_id;
  264. }
  265. /*
  266. * Add "cached" data to charts to provide the numCache and strCache data for
  267. * series and title/axis ranges.
  268. */
  269. STATIC void
  270. _add_chart_cache_data(lxw_workbook *self)
  271. {
  272. lxw_chart *chart;
  273. lxw_chart_series *series;
  274. STAILQ_FOREACH(chart, self->ordered_charts, ordered_list_pointers) {
  275. _populate_range(self, chart->title.range);
  276. _populate_range(self, chart->x_axis->title.range);
  277. _populate_range(self, chart->y_axis->title.range);
  278. if (STAILQ_EMPTY(chart->series_list))
  279. continue;
  280. STAILQ_FOREACH(series, chart->series_list, list_pointers) {
  281. _populate_range(self, series->categories);
  282. _populate_range(self, series->values);
  283. _populate_range(self, series->title.range);
  284. }
  285. }
  286. }