diff --git a/.gitignore b/.gitignore index 0cb6eeb..d4e690f 100644 --- a/.gitignore +++ b/.gitignore @@ -7,3 +7,4 @@ /pkg/ /spec/reports/ /tmp/ +.byebug_history diff --git a/README.md b/README.md index 5475554..93827d4 100644 --- a/README.md +++ b/README.md @@ -8,8 +8,8 @@ deduplication via a shared string table (SST). Its purpose is to replace CSV for CSV in Excel is very buggy and error prone. It's very efficient and can quickly write millions of rows with low memory usage. -Xlsxtream does not support formatting, charts, comments and a myriad of -other [OOXML](https://en.wikipedia.org/wiki/Office_Open_XML) features. If you are looking for a +Xlsxtream supports some basic cell styling, fonts and bordering. However if you fancy charts, comments and a myriad of +other [OOXML](https://en.wikipedia.org/wiki/Office_Open_XML) features it might be worth looking for a fully featured solution take a look at [caxslx](https://github.com/caxlsx/caxlsx). Xlsxtream supports writing to files or IO-like objects, data is flushed as the ZIP compressor sees fit. @@ -57,6 +57,13 @@ xlsx.write_worksheet 'AppendixSheet' do |sheet| sheet.add_row [Time.now, 'Time-machine'] end +# Individual cells can be styled +# It does not impact performance or file size +xlsx.write_worksheet 'Styled sheet' do |sheet| + sheet << [Cell.new('in red fill', fill: { color: 'FF0000' }), 'no style here'] + sheet << [Cell.new('font style', font: { bold: true, italic: true })] +end + # If you have highly repetitive data, you can enable Shared String Tables (SST) # for the workbook or a single worksheet. The SST has to be kept in memory, # so do not use it if you have a huge amount of rows or a little duplication diff --git a/lib/xlsxtream.rb b/lib/xlsxtream.rb index 7cdb9d0..164452b 100644 --- a/lib/xlsxtream.rb +++ b/lib/xlsxtream.rb @@ -8,3 +8,4 @@ module Xlsxtream require "xlsxtream/worksheet" require "xlsxtream/columns" require "xlsxtream/row" +require "xlsxtream/cell" diff --git a/lib/xlsxtream/cell.rb b/lib/xlsxtream/cell.rb new file mode 100644 index 0000000..afcb574 --- /dev/null +++ b/lib/xlsxtream/cell.rb @@ -0,0 +1,16 @@ +# frozen_string_literal: true + +module Xlsxtream + class Cell + attr_reader :content, :style + + def initialize(content = nil, style = {}) + @content = content + @style = style + end + + def styled? + !@style.empty? + end + end +end diff --git a/lib/xlsxtream/row.rb b/lib/xlsxtream/row.rb index afba661..0994624 100644 --- a/lib/xlsxtream/row.rb +++ b/lib/xlsxtream/row.rb @@ -1,4 +1,5 @@ # frozen_string_literal: true + require "date" require "xlsxtream/xml" @@ -13,15 +14,20 @@ class Row # ISO 8601 yyyy-mm-ddThh:mm:ss(.s)(Z|+hh:mm|-hh:mm) TIME_PATTERN = /\A[0-9]{4}-[0-9]{2}-[0-9]{2}T[0-9]{2}:[0-9]{2}(?::[0-9]{2}(?:\.[0-9]{1,9})?)?(?:Z|[+-][0-9]{2}:[0-9]{2})?\z/.freeze - TRUE_STRING = 'true'.freeze - FALSE_STRING = 'false'.freeze - - DATE_STYLE = 1 - TIME_STYLE = 2 - - def initialize(row, rownum, options = {}) + TRUE_STRING = 'true' + FALSE_STRING = 'false' + GENERIC_STRING_TYPE = 'str' # can be used for cells containing formulas + SHARED_STRING_TYPE = 's' # used only for shared strings + INLINE_STRING_TYPE = 'inlineStr' # without formulas, can be of rich text format + NUMERIC_TYPE = 'n' + BOOLEAN_TYPE = 'b' + V_WRAPPER = ->(text) { "#{text}" } + IST_WRAPPER = ->(text) { "#{text}" } + + def initialize(row, rownum, workbook, options = {}) @row = row @rownum = rownum + @stylesheet = workbook.stylesheet @sst = options[:sst] @auto_format = options[:auto_format] end @@ -34,40 +40,68 @@ def to_xml cid = "#{column}#{@rownum}" column.next! - if @auto_format && value.is_a?(String) - value = auto_format(value) - end + value = auto_format(value) if @auto_format && value.is_a?(String) + content, type = prepare_cell_content_and_resolve_type(value) + + # no xml output for empty non-styled strings + next if content.nil? && !(value.respond_to?(:styled?) && value.styled?) + + style = resolve_cell_style(value) + + # The code below renders a single XML cell. + # + # As Xlsxtream library is optimized for performance and memory, it was decided to keep + # the cell rendering logic here and not in the `Cell` class despite OOP standards encourage + # otherwise. This is to avoid unnecessary memory allocations in case of low share of + # non-styled cell content (with no need to use `Cell` wrapper). + xml << %{' + xml << (type == INLINE_STRING_TYPE ? IST_WRAPPER : V_WRAPPER)[content] + xml << '' + end - case value - when Numeric - xml << %Q{#{value}} - when TrueClass, FalseClass - xml << %Q{#{value ? 1 : 0}} - when Time - xml << %Q{#{time_to_oa_date(value)}} - when DateTime - xml << %Q{#{datetime_to_oa_date(value)}} - when Date - xml << %Q{#{date_to_oa_date(value)}} - else - value = value.to_s + xml << '' + end + + private - unless value.empty? # no xml output for for empty strings - value = value.encode(ENCODING) if value.encoding != ENCODING + def prepare_cell_content_and_resolve_type(value) + case value + when Numeric + [value, NUMERIC_TYPE] + when TrueClass, FalseClass + [(value ? 1 : 0), BOOLEAN_TYPE] + when Time + [time_to_oa_date(value), nil] + when DateTime + [datetime_to_oa_date(value), nil] + when Date + [date_to_oa_date(value), nil] + when Cell + prepare_cell_content_and_resolve_type(value.content) + else + value = value.to_s + return [nil, nil] if value.empty? + + value = value.encode(ENCODING) if value.encoding != ENCODING - if @sst - xml << %Q{#{@sst[value]}} - else - xml << %Q{#{XML.escape_value(value)}} - end - end + if @sst + [@sst[value], SHARED_STRING_TYPE] + else + [XML.escape_value(value), INLINE_STRING_TYPE] end end - - xml << '' end - private + def resolve_cell_style(value) + case value + when Time, DateTime then @stylesheet.datetime_style_id + when Date then @stylesheet.date_style_id + when Cell then @stylesheet.style_id(value) + end + end # Detects and casts numbers, date, time in text def auto_format(value) diff --git a/lib/xlsxtream/styles/border.rb b/lib/xlsxtream/styles/border.rb new file mode 100644 index 0000000..4fe1594 --- /dev/null +++ b/lib/xlsxtream/styles/border.rb @@ -0,0 +1,24 @@ +# frozen_string_literal: true + +module Xlsxtream + module Styles + class Border + def to_xml + '' + end + + def ==(other) + self.class == other.class && state == other.state + end + alias eql? == + + def hash + state.hash + end + + def state + [] + end + end + end +end diff --git a/lib/xlsxtream/styles/fill.rb b/lib/xlsxtream/styles/fill.rb new file mode 100644 index 0000000..33fac50 --- /dev/null +++ b/lib/xlsxtream/styles/fill.rb @@ -0,0 +1,67 @@ +# frozen_string_literal: true + +module Xlsxtream + module Styles + class Fill + NONE = 'none' + SOLID = 'solid' + + # https://docs.microsoft.com/en-us/dotnet/api/documentformat.openxml.spreadsheet.patternvalues?view=openxml-2.8.1 + MS_SUPPORTED = Set[ + NONE, + SOLID, + 'darkDown', + 'darkGray', + 'darkGrid', + 'darkHorizontal', + 'darkTrellis', + 'darkUp', + 'darkVertical', + 'gray0625', + 'gray125', + 'lightDown', + 'lightGray', + 'lightGrid', + 'lightHorizontal', + 'lightTrellis', + 'lightUp', + 'lightVertical', + 'mediumGray' + ] + + def initialize(pattern: nil, color: nil) + @pattern = pattern || (color ? SOLID : NONE) + @color = color + end + + def to_xml + "#{pattern_tag}" + end + + def ==(other) + self.class == other.class && state == other.state + end + alias eql? == + + def hash + state.hash + end + + def state + [@pattern, @color] + end + + private + + def color_tag + return unless @color + %{} + end + + def pattern_tag + return unless color_tag + %{#{color_tag}} + end + end + end +end diff --git a/lib/xlsxtream/styles/font.rb b/lib/xlsxtream/styles/font.rb new file mode 100644 index 0000000..59fe887 --- /dev/null +++ b/lib/xlsxtream/styles/font.rb @@ -0,0 +1,86 @@ +# frozen_string_literal: true + +module Xlsxtream + module Styles + class Font + FAMILY_IDS = { + '' => 0, + 'roman' => 1, + 'swiss' => 2, + 'modern' => 3, + 'script' => 4, + 'decorative' => 5 + }.freeze + + DEFAULT_UNDERLINE = 'single' + SUPPORTED_UNDERLINES = [ + DEFAULT_UNDERLINE, + 'singleAccounting', + 'double', + 'doubleAccounting' + ] + + def initialize(bold: nil, + italic: nil, + strike: nil, + underline: nil, + color: nil, + size: 12, + name: 'Calibri', + family: 'Swiss') + @bold = bold + @italic = italic + @strike = strike + @underline = resolve_underline(underline) + @color = color + @size = size + @name = name + @family_id = resolve_family_id(family) + end + + def to_xml + "#{tags.join}" + end + + def ==(other) + self.class == other.class && state == other.state + end + alias eql? == + + def hash + state.hash + end + + def state + [@bold, @italic, @strike, @underline, @color, @size, @name, @family_id] + end + + private + + def tags + [ + %{}, + %{}, + %{} + ].tap do |arr| + arr << %{} if @bold + arr << %{} if @italic + arr << %{} if @underline + arr << %{} if @strike + arr << %{} if @color + end + end + + def resolve_family_id(value) + FAMILY_IDS[value.to_s.downcase] or fail Error, + "Invalid font family #{value}, must be one of "\ + + FAMILY_IDS.keys.map(&:inspect).join(', ') + end + + def resolve_underline(value) + return value if SUPPORTED_UNDERLINES.include?(value) + return DEFAULT_UNDERLINE if value == true + end + end + end +end diff --git a/lib/xlsxtream/styles/stylesheet.rb b/lib/xlsxtream/styles/stylesheet.rb new file mode 100644 index 0000000..3a688eb --- /dev/null +++ b/lib/xlsxtream/styles/stylesheet.rb @@ -0,0 +1,139 @@ +# frozen_string_literal: true + +require 'set' + +require "xlsxtream/styles/border" +require "xlsxtream/styles/fill" +require "xlsxtream/styles/font" +require "xlsxtream/styles/xf" + +module Xlsxtream + module Styles + class Stylesheet + # TODO: create a class for NumFormat to replace the constants below + DEFAULT_NUM_FORMAT_ID = 0 + DATE_NUM_FORMAT_ID = 164 + TIME_NUM_FORMAT_ID = 165 + SUPPORTED_GLOBAL_OPTIONS = Set[:font, :fill] + + # Parameter `global_options` - represents a hash of default options to be + # applied to the whole book, for example: + # `global_options = { font: { size: 10 }, fill: { color: 'FFFF00' } }` + def initialize(global_options = {}) + @global_options = extract_styles_from global_options + + @borders = Hash.new { |h, k| h[k] = h.size } + @fills = Hash.new { |h, k| h[k] = h.size } + @fonts = Hash.new { |h, k| h[k] = h.size } + @xfs = Hash.new { |h, k| h[k] = h.size } + + populate_with_basic_xfs! + end + + def style_id(cell) + xf_args = {} + + xf_args[:numFmtId] = case cell.content + when Date then DATE_NUM_FORMAT_ID + when Time, DateTime then TIME_NUM_FORMAT_ID + else DEFAULT_NUM_FORMAT_ID + end + + xf_args[:borderId] = @borders[Border.new] + xf_args[:fontId] = @fonts[Font.new(cell.style[:font] || {})] + xf_args[:fillId] = @fills[Fill.new(cell.style[:fill] || {})] + xf_args[:applyFill] = 1 unless xf_args[:fillId] == @fills[default_fill] + + @xfs[Xf.new(xf_args)] + end + + def to_xml + XML.header + XML.strip(<<~XML) + + + + + + #{@fonts.keys.map(&:to_xml).join} + #{@fills.keys.map(&:to_xml).join} + #{@borders.keys.map(&:to_xml).join} + #{@xfs.keys.first.to_xml} + #{@xfs.keys.map(&:to_xml).join} + + + + + + + XML + end + + def default_style_id + @default_style_id ||= @xfs[Xf.new( + numFmtId: DEFAULT_NUM_FORMAT_ID, + fontId: @fonts[default_font], + fillId: @fills[default_fill], + borderId: @borders[default_border] + )] + end + + def datetime_style_id + @datetime_style_id ||= @xfs[Xf.new( + numFmtId: TIME_NUM_FORMAT_ID, + fontId: @fonts[default_font], + fillId: @fills[default_fill], + borderId: @borders[default_border], + applyNumberFormat: 1 + )] + end + + def date_style_id + @date_style_id ||= @xfs[Xf.new( + numFmtId: DATE_NUM_FORMAT_ID, + fontId: @fonts[default_font], + fillId: @fills[default_fill], + borderId: @borders[default_border], + applyNumberFormat: 1 + )] + end + + private + + def default_border + # no options for `Border` are supported currently + @default_border ||= Border.new + end + + def default_fill + @default_fill ||= begin + @global_options.key?(:fill) ? Fill.new(@global_options[:fill]) : Fill.new + end + end + + def default_font + @default_font ||= begin + @global_options.key?(:font) ? Font.new(@global_options[:font]) : Font.new + end + end + + def populate_with_basic_xfs! + # add default entities to the collections + @borders[default_border] + @fills[default_fill] + @fonts[default_font] + + # populate xfs + default_style_id + date_style_id + datetime_style_id + end + + def extract_styles_from(options) + SUPPORTED_GLOBAL_OPTIONS.reduce({}) do |result, option_key| + result[option_key] = options[option_key] if options[option_key].is_a?(Hash) + result + end + end + end + end +end diff --git a/lib/xlsxtream/styles/xf.rb b/lib/xlsxtream/styles/xf.rb new file mode 100644 index 0000000..4c7ed91 --- /dev/null +++ b/lib/xlsxtream/styles/xf.rb @@ -0,0 +1,51 @@ +# frozen_string_literal: true + +module Xlsxtream + module Styles + class Xf + REQUIRED_FIELDS = %i[ + numFmtId + fontId + fillId + borderId + xfId + ].freeze + + OPTIONAL_FIELDS = %i[ + applyFill + applyNumberFormat + ].freeze + + SUPPORTED_FIELDS = REQUIRED_FIELDS + OPTIONAL_FIELDS + + def initialize(attrs) + @attrs = filter attrs + end + + def to_xml + tag_attrs = @attrs.map { |k, v| %{#{k}="#{v}"} }.join(' ') + + %{} + end + + def ==(other) + self.class == other.class && state == other.state + end + alias eql? == + + def hash + state.hash + end + + def state + @attrs + end + + private + + def filter(attrs) + attrs.select { |key| SUPPORTED_FIELDS.include?(key.to_sym) } + end + end + end +end diff --git a/lib/xlsxtream/workbook.rb b/lib/xlsxtream/workbook.rb index 1fdb04b..aa7c4c5 100644 --- a/lib/xlsxtream/workbook.rb +++ b/lib/xlsxtream/workbook.rb @@ -4,21 +4,13 @@ require "xlsxtream/shared_string_table" require "xlsxtream/worksheet" require "xlsxtream/io/zip_tricks" +require "xlsxtream/styles/stylesheet" module Xlsxtream class Workbook - - FONT_FAMILY_IDS = { - '' => 0, - 'roman' => 1, - 'swiss' => 2, - 'modern' => 3, - 'script' => 4, - 'decorative' => 5 - }.freeze + attr_reader :stylesheet, :io class << self - def open(output, options = {}) workbook = new(output, options) if block_given? @@ -31,7 +23,6 @@ def open(output, options = {}) workbook end end - end def initialize(output, options = {}) @@ -44,6 +35,7 @@ def initialize(output, options = {}) "The Xlsxtream::Workbook.new :io_wrapper option is deprecated. "\ "Please pass an IO wrapper instance as the first argument instead." end + if output.is_a?(String) || !output.respond_to?(:<<) @file = File.open(output, 'wb') @io = IO::ZipTricks.new(@file) @@ -54,8 +46,11 @@ def initialize(output, options = {}) @file = nil @io = IO::ZipTricks.new(output) end + @sst = SharedStringTable.new @worksheets = [] + + @stylesheet = Styles::Stylesheet.new(options) end def add_worksheet(*args, &block) @@ -111,7 +106,7 @@ def build_worksheet(name = nil, options = {}) @io.add_file "xl/worksheets/sheet#{sheet_id}.xml" - worksheet = Worksheet.new(@io, :id => sheet_id, :name => name, :sst => sst, :auto_format => auto_format, :columns => columns) + worksheet = Worksheet.new(self, id: sheet_id, name: name, sst: sst, auto_format: auto_format, columns: columns) @worksheets << worksheet worksheet @@ -146,55 +141,8 @@ def write_workbook end def write_styles - font_options = @options.fetch(:font, {}) - font_size = font_options.fetch(:size, 12).to_s - font_name = font_options.fetch(:name, 'Calibri').to_s - font_family = font_options.fetch(:family, 'Swiss').to_s.downcase - font_family_id = FONT_FAMILY_IDS[font_family] or fail Error, - "Invalid font family #{font_family}, must be one of "\ - + FONT_FAMILY_IDS.keys.map(&:inspect).join(', ') - @io.add_file "xl/styles.xml" - @io << XML.header - @io << XML.strip(<<-XML) - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - XML + @io << @stylesheet.to_xml end def write_sst diff --git a/lib/xlsxtream/worksheet.rb b/lib/xlsxtream/worksheet.rb index c4e8b41..3665761 100644 --- a/lib/xlsxtream/worksheet.rb +++ b/lib/xlsxtream/worksheet.rb @@ -4,8 +4,9 @@ module Xlsxtream class Worksheet - def initialize(io, options = {}) - @io = io + def initialize(workbook, options = {}) + @workbook = workbook + @io = workbook.io @rownum = 1 @closed = false @options = options @@ -14,7 +15,7 @@ def initialize(io, options = {}) end def <<(row) - @io << Row.new(row, @rownum, @options).to_xml + @io << Row.new(row, @rownum, @workbook, @options).to_xml @rownum += 1 end alias_method :add_row, :<< diff --git a/test/xlsxtream/row_test.rb b/test/xlsxtream/row_test.rb index db051a2..2565b26 100644 --- a/test/xlsxtream/row_test.rb +++ b/test/xlsxtream/row_test.rb @@ -4,100 +4,103 @@ module Xlsxtream class RowTest < Minitest::Test + DUMMY_IO = StringIO.new + DUMMY_WORKBOOK = Workbook.new(DUMMY_IO) + def test_empty_column - row = Row.new([nil], 1) + row = Row.new([nil], 1, DUMMY_WORKBOOK) expected = '' actual = row.to_xml assert_equal expected, actual end def test_string_column - row = Row.new(['hello'], 1) + row = Row.new(['hello'], 1, DUMMY_WORKBOOK) expected = 'hello' actual = row.to_xml assert_equal expected, actual end def test_symbol_column - row = Row.new([:hello], 1) + row = Row.new([:hello], 1, DUMMY_WORKBOOK) expected = 'hello' actual = row.to_xml assert_equal expected, actual end def test_boolean_column - row = Row.new([true], 1) + row = Row.new([true], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '1' assert_equal expected, actual - row = Row.new([false], 1) + row = Row.new([false], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '0' assert_equal expected, actual end def test_text_boolean_column - row = Row.new(['true'], 1, :auto_format => true) + row = Row.new(['true'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '1' assert_equal expected, actual - row = Row.new(['false'], 1, :auto_format => true) + row = Row.new(['false'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '0' assert_equal expected, actual end def test_integer_column - row = Row.new([1], 1) + row = Row.new([1], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '1' assert_equal expected, actual end def test_text_integer_column - row = Row.new(['1'], 1, :auto_format => true) + row = Row.new(['1'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '1' assert_equal expected, actual end def test_float_column - row = Row.new([1.5], 1) + row = Row.new([1.5], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '1.5' assert_equal expected, actual end def test_text_float_column - row = Row.new(['1.5'], 1, :auto_format => true) + row = Row.new(['1.5'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '1.5' assert_equal expected, actual end def test_date_column - row = Row.new([Date.new(1900, 1, 1)], 1) + row = Row.new([Date.new(1900, 1, 1)], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '2.0' assert_equal expected, actual end def test_text_date_column - row = Row.new(['1900-01-01'], 1, :auto_format => true) + row = Row.new(['1900-01-01'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '2.0' assert_equal expected, actual end def test_invalid_text_date_column - row = Row.new(['1900-02-29'], 1, :auto_format => true) + row = Row.new(['1900-02-29'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '1900-02-29' assert_equal expected, actual end def test_date_time_column - row = Row.new([DateTime.new(1900, 1, 1, 12, 0, 0, '+00:00')], 1) + row = Row.new([DateTime.new(1900, 1, 1, 12, 0, 0, '+00:00')], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '2.5' assert_equal expected, actual @@ -113,26 +116,26 @@ def test_text_date_time_column '1900-01-01T12:00:00.000000000Z' ] candidates.each do |timestamp| - row = Row.new([timestamp], 1, :auto_format => true) + row = Row.new([timestamp], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '2.5' assert_equal expected, actual end - row = Row.new(['1900-01-01T12'], 1, :auto_format => true) + row = Row.new(['1900-01-01T12'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '2.5' refute_equal expected, actual end def test_invalid_text_date_time_column - row = Row.new(['1900-02-29T12:00'], 1, :auto_format => true) + row = Row.new(['1900-02-29T12:00'], 1, DUMMY_WORKBOOK, :auto_format => true) actual = row.to_xml expected = '1900-02-29T12:00' assert_equal expected, actual end def test_time_column - row = Row.new([Time.new(1900, 1, 1, 12, 0, 0, '+00:00')], 1) + row = Row.new([Time.new(1900, 1, 1, 12, 0, 0, '+00:00')], 1, DUMMY_WORKBOOK) actual = row.to_xml expected = '2.5' assert_equal expected, actual @@ -140,17 +143,52 @@ def test_time_column def test_string_column_with_shared_string_table mock_sst = { 'hello' => 0 } - row = Row.new(['hello'], 1, :sst => mock_sst) + row = Row.new(['hello'], 1, DUMMY_WORKBOOK, :sst => mock_sst) expected = '0' actual = row.to_xml assert_equal expected, actual end def test_multiple_columns - row = Row.new(['foo', nil, 23], 1) + row = Row.new(['foo', nil, 23], 1, DUMMY_WORKBOOK) expected = 'foo23' actual = row.to_xml assert_equal expected, actual end + + def test_styled_column_with_really_no_style + row = Row.new([Cell.new('foo')], 1, DUMMY_WORKBOOK) + expected_style_id = DUMMY_WORKBOOK.stylesheet.default_style_id + + expected = <<~HTML.gsub(/\n\s*/, '') + + + + foo + + + + HTML + + assert_equal expected, row.to_xml + end + + def test_styled_columns + cells = [ + Cell.new('Red fill', fill: { color: 'FF0000' }), + nil, + Cell.new(23, font: { bold: true, italic: true }) + ] + row = Row.new(cells, 1, DUMMY_WORKBOOK) + + expected = <<~HTML.gsub(/\n\s*/, '') + + Red fill + 23 + + HTML + + assert_equal expected, row.to_xml + end end end diff --git a/test/xlsxtream/styles/stylesheet_test.rb b/test/xlsxtream/styles/stylesheet_test.rb new file mode 100644 index 0000000..68fce12 --- /dev/null +++ b/test/xlsxtream/styles/stylesheet_test.rb @@ -0,0 +1,121 @@ +require 'test_helper' + +module Xlsxtream; module Styles + class StylesheetTest < Minitest::Test + DEFAULT_NUM_FORMAT_ID = 0 + DATE_NUM_FORMAT_ID = 164 + TIME_NUM_FORMAT_ID = 165 + + SUPPORTED_OPTIONS = { + fill: { color: 'FF0000' }, font: { bold: true, italic: true } + } + + UNSUPPORTED_OPTIONS = { + foo: { color: 'FF0000' }, bar: { bold: true, italic: true } + } + + def test_constants + assert_equal DEFAULT_NUM_FORMAT_ID, Stylesheet::DEFAULT_NUM_FORMAT_ID + assert_equal DATE_NUM_FORMAT_ID, Stylesheet::DATE_NUM_FORMAT_ID + assert_equal TIME_NUM_FORMAT_ID, Stylesheet::TIME_NUM_FORMAT_ID + end + + def test_initialize_populates_basic_xfs + workbook = Workbook.new(StringIO.new) + assert_equal 3, peek_num_of_styles(workbook.stylesheet) + end + + def test_initialize_respects_global_options + workbook = Workbook.new(StringIO.new, SUPPORTED_OPTIONS) + + # does not create a new default + assert_equal 3, peek_num_of_styles(workbook.stylesheet) + + # the default style is set using global options + actual_style_id = workbook.stylesheet.style_id(Cell.new('foo', SUPPORTED_OPTIONS)) + assert_equal workbook.stylesheet.default_style_id, actual_style_id + end + + def test_style_id + workbook = Workbook.new(StringIO.new) + initial_num_styles = peek_num_of_styles(workbook.stylesheet) + + # creates a new style for a cell with the new style + cell = Cell.new('foo', SUPPORTED_OPTIONS) + assert_equal initial_num_styles, workbook.stylesheet.style_id(cell) + + # does not create a new style for the cell with the same style + cell = Cell.new('bar', SUPPORTED_OPTIONS.dup) + assert_equal initial_num_styles, workbook.stylesheet.style_id(cell) + + # does not respect unsupported options + cell = Cell.new('foo', UNSUPPORTED_OPTIONS) + assert_equal 0, workbook.stylesheet.style_id(cell) + end + + def test_to_xml + workbook = Workbook.new(StringIO.new) + + cell = Cell.new('foo', SUPPORTED_OPTIONS) + workbook.stylesheet.style_id(cell) + + expected = \ + ''"\r\n" \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' \ + '' + + actual = workbook.stylesheet.to_xml + assert_equal expected, actual + end + + private + + def peek_num_of_styles(stylesheet) + stylesheet.instance_variable_get("@xfs").size + end + end +end; end diff --git a/test/xlsxtream/workbook_test.rb b/test/xlsxtream/workbook_test.rb index a589cae..890f145 100644 --- a/test/xlsxtream/workbook_test.rb +++ b/test/xlsxtream/workbook_test.rb @@ -384,24 +384,20 @@ def test_styles_content '' \ '' \ '' \ - '' \ + '' \ '' \ - '' \ - '' \ - '' \ - '' \ '' \ '' \ '' \ '' \ '' \ '' \ - '' \ + '' \ '' \ '' \ - '' \ - '' \ - '' \ + '' \ + '' \ + '' \ '' \ '' \ '' \ @@ -415,8 +411,8 @@ def test_styles_content def test_custom_font_size iow_spy = io_wrapper_spy - font_options = { :size => 23 } - Workbook.open(iow_spy, :font => font_options) {} + font_options = { size: 23 } + Workbook.open(iow_spy, font: font_options) {} expected = '' actual = iow_spy['xl/styles.xml'][/]+>/] assert_equal expected, actual diff --git a/test/xlsxtream/worksheet_test.rb b/test/xlsxtream/worksheet_test.rb index 9f2aa3b..363f291 100644 --- a/test/xlsxtream/worksheet_test.rb +++ b/test/xlsxtream/worksheet_test.rb @@ -1,13 +1,14 @@ # frozen_string_literal: true require 'test_helper' require 'stringio' +require 'xlsxtream/workbook' require 'xlsxtream/worksheet' module Xlsxtream class WorksheetTest < Minitest::Test def test_empty_worksheet io = StringIO.new - ws = Worksheet.new(io) + ws = Worksheet.new(mock_workbook(io)) ws.close expected = \ ''"\r\n" \ @@ -17,7 +18,7 @@ def test_empty_worksheet def test_add_row io = StringIO.new - ws = Worksheet.new(io) + ws = Worksheet.new(mock_workbook(io)) ws << ['foo'] ws.add_row ['bar'] ws.close @@ -33,7 +34,7 @@ def test_add_row def test_add_row_with_sst_option io = StringIO.new mock_sst = { 'foo' => 0 } - ws = Worksheet.new(io, :sst => mock_sst) + ws = Worksheet.new(mock_workbook(io), :sst => mock_sst) ws << ['foo'] ws.close expected = \ @@ -46,7 +47,7 @@ def test_add_row_with_sst_option def test_add_row_with_auto_format_option io = StringIO.new - ws = Worksheet.new(io, :auto_format => true) + ws = Worksheet.new(mock_workbook(io), :auto_format => true) ws << ['1.5'] ws.close expected = \ @@ -59,7 +60,7 @@ def test_add_row_with_auto_format_option def test_add_columns_via_worksheet_options io = StringIO.new - ws = Worksheet.new(io, { :columns => [ {}, {}, { :width_pixels => 42 } ] } ) + ws = Worksheet.new(mock_workbook(io), { :columns => [ {}, {}, { :width_pixels => 42 } ] } ) ws.close expected = \ ''"\r\n" \ @@ -74,7 +75,7 @@ def test_add_columns_via_worksheet_options def test_add_columns_via_worksheet_options_and_add_rows io = StringIO.new - ws = Worksheet.new(io, { :columns => [ {}, {}, { :width_pixels => 42 } ] } ) + ws = Worksheet.new(mock_workbook(io), { :columns => [ {}, {}, { :width_pixels => 42 } ] } ) ws << ['foo'] ws.add_row ['bar'] ws.close @@ -93,13 +94,24 @@ def test_add_columns_via_worksheet_options_and_add_rows end def test_respond_to_id - ws = Worksheet.new(StringIO.new, id: 1) + ws = Worksheet.new(mock_workbook, id: 1) assert_equal 1, ws.id end def test_respond_to_name - ws = Worksheet.new(StringIO.new, name: 'test') + ws = Worksheet.new(mock_workbook, name: 'test') assert_equal 'test', ws.name end + + private + + def mock_workbook(io = StringIO.new) + ss = Styles::Stylesheet.new + + Minitest::Mock.new + .expect(:io, io) + .expect(:stylesheet, ss) + .expect(:stylesheet, ss) + end end end