write.c 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352
  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. }
  106. return error;
  107. }
  108. void _php_vtiful_excel_close(zend_resource *rsrc TSRMLS_DC)
  109. {
  110. //Here to PHP for gc
  111. }
  112. /*
  113. * Iterate through the worksheets and store any defined names used for print
  114. * ranges or repeat rows/columns.
  115. */
  116. STATIC void
  117. _prepare_defined_names(lxw_workbook *self)
  118. {
  119. lxw_worksheet *worksheet;
  120. char app_name[LXW_DEFINED_NAME_LENGTH];
  121. char range[LXW_DEFINED_NAME_LENGTH];
  122. char area[LXW_MAX_CELL_RANGE_LENGTH];
  123. char first_col[8];
  124. char last_col[8];
  125. STAILQ_FOREACH(worksheet, self->worksheets, list_pointers) {
  126. /*
  127. * Check for autofilter settings and store them.
  128. */
  129. if (worksheet->autofilter.in_use) {
  130. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  131. "%s!_FilterDatabase", worksheet->quoted_name);
  132. lxw_rowcol_to_range_abs(area,
  133. worksheet->autofilter.first_row,
  134. worksheet->autofilter.first_col,
  135. worksheet->autofilter.last_row,
  136. worksheet->autofilter.last_col);
  137. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH, "%s!%s",
  138. worksheet->quoted_name, area);
  139. /* Autofilters are the only defined name to set the hidden flag. */
  140. _store_defined_name(self, "_xlnm._FilterDatabase", app_name,
  141. range, worksheet->index, LXW_TRUE);
  142. }
  143. /*
  144. * Check for Print Area settings and store them.
  145. */
  146. if (worksheet->print_area.in_use) {
  147. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  148. "%s!Print_Area", worksheet->quoted_name);
  149. /* Check for print area that is the max row range. */
  150. if (worksheet->print_area.first_row == 0
  151. && worksheet->print_area.last_row == LXW_ROW_MAX - 1) {
  152. lxw_col_to_name(first_col,
  153. worksheet->print_area.first_col, LXW_FALSE);
  154. lxw_col_to_name(last_col,
  155. worksheet->print_area.last_col, LXW_FALSE);
  156. lxw_snprintf(area, LXW_MAX_CELL_RANGE_LENGTH - 1, "$%s:$%s",
  157. first_col, last_col);
  158. }
  159. /* Check for print area that is the max column range. */
  160. else if (worksheet->print_area.first_col == 0
  161. && worksheet->print_area.last_col == LXW_COL_MAX - 1) {
  162. lxw_snprintf(area, LXW_MAX_CELL_RANGE_LENGTH - 1, "$%d:$%d",
  163. worksheet->print_area.first_row + 1,
  164. worksheet->print_area.last_row + 1);
  165. }
  166. else {
  167. lxw_rowcol_to_range_abs(area,
  168. worksheet->print_area.first_row,
  169. worksheet->print_area.first_col,
  170. worksheet->print_area.last_row,
  171. worksheet->print_area.last_col);
  172. }
  173. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH, "%s!%s",
  174. worksheet->quoted_name, area);
  175. _store_defined_name(self, "_xlnm.Print_Area", app_name,
  176. range, worksheet->index, LXW_FALSE);
  177. }
  178. /*
  179. * Check for repeat rows/cols. aka, Print Titles and store them.
  180. */
  181. if (worksheet->repeat_rows.in_use || worksheet->repeat_cols.in_use) {
  182. if (worksheet->repeat_rows.in_use
  183. && worksheet->repeat_cols.in_use) {
  184. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  185. "%s!Print_Titles", worksheet->quoted_name);
  186. lxw_col_to_name(first_col,
  187. worksheet->repeat_cols.first_col, LXW_FALSE);
  188. lxw_col_to_name(last_col,
  189. worksheet->repeat_cols.last_col, LXW_FALSE);
  190. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  191. "%s!$%s:$%s,%s!$%d:$%d",
  192. worksheet->quoted_name, first_col,
  193. last_col, worksheet->quoted_name,
  194. worksheet->repeat_rows.first_row + 1,
  195. worksheet->repeat_rows.last_row + 1);
  196. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  197. range, worksheet->index, LXW_FALSE);
  198. }
  199. else if (worksheet->repeat_rows.in_use) {
  200. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  201. "%s!Print_Titles", worksheet->quoted_name);
  202. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  203. "%s!$%d:$%d", worksheet->quoted_name,
  204. worksheet->repeat_rows.first_row + 1,
  205. worksheet->repeat_rows.last_row + 1);
  206. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  207. range, worksheet->index, LXW_FALSE);
  208. }
  209. else if (worksheet->repeat_cols.in_use) {
  210. lxw_snprintf(app_name, LXW_DEFINED_NAME_LENGTH,
  211. "%s!Print_Titles", worksheet->quoted_name);
  212. lxw_col_to_name(first_col,
  213. worksheet->repeat_cols.first_col, LXW_FALSE);
  214. lxw_col_to_name(last_col,
  215. worksheet->repeat_cols.last_col, LXW_FALSE);
  216. lxw_snprintf(range, LXW_DEFINED_NAME_LENGTH,
  217. "%s!$%s:$%s", worksheet->quoted_name,
  218. first_col, last_col);
  219. _store_defined_name(self, "_xlnm.Print_Titles", app_name,
  220. range, worksheet->index, LXW_FALSE);
  221. }
  222. }
  223. }
  224. }
  225. /*
  226. * Iterate through the worksheets and set up any chart or image drawings.
  227. */
  228. STATIC void
  229. _prepare_drawings(lxw_workbook *self)
  230. {
  231. lxw_worksheet *worksheet;
  232. lxw_image_options *image_options;
  233. uint16_t chart_ref_id = 0;
  234. uint16_t image_ref_id = 0;
  235. uint16_t drawing_id = 0;
  236. STAILQ_FOREACH(worksheet, self->worksheets, list_pointers) {
  237. if (STAILQ_EMPTY(worksheet->image_data)
  238. && STAILQ_EMPTY(worksheet->chart_data))
  239. continue;
  240. drawing_id++;
  241. STAILQ_FOREACH(image_options, worksheet->chart_data, list_pointers) {
  242. chart_ref_id++;
  243. lxw_worksheet_prepare_chart(worksheet, chart_ref_id, drawing_id,
  244. image_options);
  245. if (image_options->chart)
  246. STAILQ_INSERT_TAIL(self->ordered_charts, image_options->chart,
  247. ordered_list_pointers);
  248. }
  249. STAILQ_FOREACH(image_options, worksheet->image_data, list_pointers) {
  250. if (image_options->image_type == LXW_IMAGE_PNG)
  251. self->has_png = LXW_TRUE;
  252. if (image_options->image_type == LXW_IMAGE_JPEG)
  253. self->has_jpeg = LXW_TRUE;
  254. if (image_options->image_type == LXW_IMAGE_BMP)
  255. self->has_bmp = LXW_TRUE;
  256. image_ref_id++;
  257. lxw_worksheet_prepare_image(worksheet, image_ref_id, drawing_id,
  258. image_options);
  259. }
  260. }
  261. self->drawing_count = drawing_id;
  262. }
  263. /*
  264. * Add "cached" data to charts to provide the numCache and strCache data for
  265. * series and title/axis ranges.
  266. */
  267. STATIC void
  268. _add_chart_cache_data(lxw_workbook *self)
  269. {
  270. lxw_chart *chart;
  271. lxw_chart_series *series;
  272. STAILQ_FOREACH(chart, self->ordered_charts, ordered_list_pointers) {
  273. _populate_range(self, chart->title.range);
  274. _populate_range(self, chart->x_axis->title.range);
  275. _populate_range(self, chart->y_axis->title.range);
  276. if (STAILQ_EMPTY(chart->series_list))
  277. continue;
  278. STAILQ_FOREACH(series, chart->series_list, list_pointers) {
  279. _populate_range(self, series->categories);
  280. _populate_range(self, series->values);
  281. _populate_range(self, series->title.range);
  282. }
  283. }
  284. }