# encoding: UTF-8
require 'cgi'
module Axlsx
# A cell in a worksheet.
# Cell stores inforamation requried to serialize a single worksheet cell to xml. You must provde the Row that the cell belongs to and the cells value. The data type will automatically be determed if you do not specify the :type option. The default style will be applied if you do not supply the :style option. Changing the cell's type will recast the value to the type specified. Altering the cell's value via the property accessor will also automatically cast the provided value to the cell's type.
# @note The recommended way to generate cells is via Worksheet#add_row
#
# @see Worksheet#add_row
class Cell
# An array of available inline styes.
# TODO change this to a hash where each key defines attr name and validator (and any info the validator requires)
# then move it out to a module so we can re-use in in other classes.
# needs to define bla=(v) and bla methods on the class that hook into a
# set_attr method that kicks the suplied validator and updates the instance_variable
# for the key
INLINE_STYLES = ['value', 'type', 'font_name', 'charset',
'family', 'b', 'i', 'strike','outline',
'shadow', 'condense', 'extend', 'u',
'vertAlign', 'sz', 'color', 'scheme']
# The index of the cellXfs item to be applied to this cell.
# @return [Integer]
# @see Axlsx::Styles
attr_reader :style
# The row this cell belongs to.
# @return [Row]
attr_reader :row
# The cell's data type. Currently only six types are supported, :date, :time, :float, :integer, :string and :boolean.
# Changing the type for a cell will recast the value into that type. If no type option is specified in the constructor, the type is
# automatically determed.
# @see Cell#cell_type_from_value
# @return [Symbol] The type of data this cell's value is cast to.
# @raise [ArgumentExeption] Cell.type must be one of [:date, time, :float, :integer, :string, :boolean]
# @note
# If the value provided cannot be cast into the type specified, type is changed to :string and the following logic is applied.
# :string to :integer or :float, type conversions always return 0 or 0.0
# :string, :integer, or :float to :time conversions always return the original value as a string and set the cells type to :string.
# No support is currently implemented for parsing time strings.
attr_reader :type
# @see type
def type=(v)
RestrictionValidator.validate "Cell.type", [:date, :time, :float, :integer, :string, :boolean], v
@type=v
self.value = @value unless @value.nil?
end
# The value of this cell.
# @return [String, Integer, Float, Time] casted value based on cell's type attribute.
attr_reader :value
# @see value
def value=(v)
#TODO: consider doing value based type determination first?
@value = cast_value(v)
end
# Indicates that the cell has one or more of the custom cell styles applied.
# @return [Boolean]
def is_text_run?
@is_text_run ||= false
end
# Indicates if the cell is good for shared string table
def plain_string?
@type == :string && # String typed
!is_text_run? && # No inline styles
!@value.nil? && # Not nil
!@value.empty? && # Not empty
!@value.start_with?('=') # Not a formula
end
# The inline font_name property for the cell
# @return [String]
attr_reader :font_name
# @see font_name
def font_name=(v) set_run_style :validate_string, :font_name, v; end
# The inline charset property for the cell
# @return [String]
attr_reader :charset
# @see charset
def charset=(v) set_run_style :validate_unsigned_int, :charset, v; end
# The inline family property for the cell
# @return [String]
attr_reader :family
# @see family
def family=(v) set_run_style :validate_string, :family, v; end
# The inline bold property for the cell
# @return [Boolean]
attr_reader :b
# @see b
def b=(v) set_run_style :validate_boolean, :b, v; end
# The inline italic property for the cell
# @return [Boolean]
attr_reader :i
# @see i
def i=(v) set_run_style :validate_boolean, :i, v; end
# The inline strike property for the cell
# @return [Boolean]
attr_reader :strike
# @see strike
def strike=(v) set_run_style :validate_boolean, :strike, v; end
# The inline outline property for the cell
# @return [Boolean]
attr_reader :outline
# @see outline
def outline=(v) set_run_style :validate_boolean, :outline, v; end
# The inline shadow property for the cell
# @return [Boolean]
attr_reader :shadow
# @see shadow
def shadow=(v) set_run_style :validate_boolean, :shadow, v; end
# The inline condense property for the cell
# @return [Boolean]
attr_reader :condense
# @see condense
def condense=(v) set_run_style :validate_boolean, :condense, v; end
# The inline extend property for the cell
# @return [Boolean]
attr_reader :extend
# @see extend
def extend=(v) set_run_style :validate_boolean, :extend, v; end
# The inline underline property for the cell
# @return [Boolean]
attr_reader :u
# @see u
def u=(v) set_run_style :validate_boolean, :u, v; end
# The inline color property for the cell
# @return [Color]
attr_reader :color
# @param [String] v The 8 character representation for an rgb color #FFFFFFFF"
def color=(v)
@color = v.is_a?(Color) ? v : Color.new(:rgb=>v)
@is_text_run = true
end
# The inline sz property for the cell
# @return [Boolean]
attr_reader :sz
# @see sz
def sz=(v) set_run_style :validate_unsigned_int, :sz, v; end
# The inline vertical alignment property for the cell
# this must be one of [:baseline, :subscript, :superscript]
# @return [Symbol]
attr_reader :vertAlign
# @see vertAlign
def vertAlign=(v)
RestrictionValidator.validate "Cell.vertAlign", [:baseline, :subscript, :superscript], v
set_run_style nil, :vertAlign, v
end
# The inline scheme property for the cell
# this must be one of [:none, major, minor]
# @return [Symbol]
attr_reader :scheme
# @see scheme
def scheme=(v)
RestrictionValidator.validate "Cell.schema", [:none, :major, :minor], v
set_run_style nil, :scheme, v
end
# @param [Row] row The row this cell belongs to.
# @param [Any] value The value associated with this cell.
# @option options [Symbol] type The intended data type for this cell. If not specified the data type will be determined internally based on the vlue provided.
# @option options [Integer] style The index of the cellXfs item to be applied to this cell. If not specified, the default style (0) will be applied.
# @option options [String] font_name
# @option options [Integer] charset
# @option options [String] family
# @option options [Boolean] b
# @option options [Boolean] i
# @option options [Boolean] strike
# @option options [Boolean] outline
# @option options [Boolean] shadow
# @option options [Boolean] condense
# @option options [Boolean] extend
# @option options [Boolean] u
# @option options [Symbol] vertAlign must be one of :baseline, :subscript, :superscript
# @option options [Integer] sz
# @option options [String] color an 8 letter rgb specification
# @option options [Symbol] scheme must be one of :none, major, :minor
def initialize(row, value="", options={})
self.row=row
@value = @font_name = @charset = @family = @b = @i = @strike = @outline = @shadow = nil
@condense = @u = @vertAlign = @sz = @color = @scheme = @extend = @ssti = nil
@styles = row.worksheet.workbook.styles
@row.cells << self
options.each do |o|
self.send("#{o[0]}=", o[1]) if self.respond_to? "#{o[0]}="
end
@style ||= 0
@type ||= cell_type_from_value(value)
@value = cast_value(value)
end
# The Shared Strings Table index for this cell
# @return [Integer]
attr_reader :ssti
# @return [Integer] The index of the cell in the containing row.
def index
@row.cells.index(self)
end
# @return [String] The alpha(column)numeric(row) reference for this sell.
# @example Relative Cell Reference
# ws.rows.first.cells.first.r #=> "A1"
def r
Axlsx::cell_r index, @row.index
end
# @return [String] The absolute alpha(column)numeric(row) reference for this sell.
# @example Absolute Cell Reference
# ws.rows.first.cells.first.r #=> "$A$1"
def r_abs
"$#{r.match(%r{([A-Z]+)([0-9]+)})[1,2].join('$')}"
end
# @return [Integer] The cellXfs item index applied to this cell.
# @raise [ArgumentError] Invalid cellXfs id if the value provided is not within cellXfs items range.
def style=(v)
Axlsx::validate_unsigned_int(v)
count = @styles.cellXfs.size
raise ArgumentError, "Invalid cellXfs id" unless v < count
@style = v
end
# @return [Array] of x/y coordinates in the cheet for this cell.
def pos
[index, row.index]
end
# Merges all the cells in a range created between this cell and the cell or string name for a cell provided
# @see worksheet.merge_cells
# @param [Cell, String] target The last cell, or str ref for the cell in the merge range
def merge(target)
range_end = if target.is_a?(String)
target
elsif(target.is_a?(Cell))
target.r
end
self.row.worksheet.merge_cells "#{self.r}:#{range_end}" unless range_end.nil?
end
# builds an xml text run based on this cells attributes.
# @param [String] str The string instance this run will be concated to.
# @return [String]
def run_xml_string(str = '')
if is_text_run?
data = instance_values.reject{|key, value| value == nil || key == 'value' || key == 'type' }
keys = data.keys & INLINE_STYLES
str << ""
keys.each do |key|
case key
when 'font_name'
str << ""
when 'color'
str << data[key].to_xml_string
else
str << "<" << key.to_s << " val='" << data[key].to_s << "'/>"
end
end
str << "" << "" << value.to_s << ""
else
str << "" << value.to_s << ""
end
str
end
# Serializes the cell
# @param [Integer] r_index The row index for the cell
# @param [Integer] c_index The cell index in the row.
# @param [String] str The string index the cell content will be appended to. Defaults to empty string.
# @return [String] xml text for the cell
def to_xml_string(r_index, c_index, str = '')
str << '' if @value.nil?
case @type
when :string
#parse formula
if @value.start_with?('=')
str << 't="str">' << @value.to_s.sub('=', '') << ''
else
#parse shared
if @ssti
str << 't="s">' << @ssti.to_s << ''
else
str << 't="inlineStr">' << run_xml_string << ''
end
end
when :date
# TODO: See if this is subject to the same restriction as Time below
str << '>' << DateTimeConverter::date_to_serial(@value).to_s << ''
when :time
str << '>' << DateTimeConverter::time_to_serial(@value).to_s << ''
when :boolean
str << 't="b">' << @value.to_s << ''
else
str << '>' << @value.to_s << ''
end
str << ''
end
def is_formula?
@type == :string && @value.start_with?('=')
end
private
# Utility method for setting inline style attributes
def set_run_style( validator, attr, value)
return unless INLINE_STYLES.include?(attr.to_s)
Axlsx.send(validator, value) unless validator == nil
self.instance_variable_set :"@#{attr.to_s}", value
@is_text_run = true
end
# @see ssti
def ssti=(v)
Axlsx::validate_unsigned_int(v)
@ssti = v
end
# assigns the owning row for this cell.
def row=(v) DataTypeValidator.validate "Cell.row", Row, v; @row=v end
# Determines the cell type based on the cell value.
# @note This is only used when a cell is created but no :type option is specified, the following rules apply:
# 1. If the value is an instance of Date, the type is set to :date
# 2. If the value is an instance of Time, the type is set to :time
# 3. If the value is an instance of TrueClass or FalseClass, the type is set to :boolean
# 4. :float and :integer types are determined by regular expression matching.
# 5. Anything that does not meet either of the above is determined to be :string.
# @return [Symbol] The determined type
def cell_type_from_value(v)
if v.is_a?(Date)
:date
elsif v.is_a?(Time)
:time
elsif v.is_a?(TrueClass) || v.is_a?(FalseClass)
:boolean
elsif v.to_s.match(/\A[+-]?\d+?\Z/) #numeric
:integer
elsif v.to_s.match(/\A[+-]?\d+\.\d+?\Z/) #float
:float
else
:string
end
end
# Cast the value into this cells data type.
# @note
# About Time - Time in OOXML is *different* from what you might expect. The history as to why is interesting, but you can safely assume that if you are generating docs on a mac, you will want to specify Workbook.1904 as true when using time typed values.
# @see Axlsx#date1904
def cast_value(v)
return nil if v.nil?
if @type == :date
self.style = STYLE_DATE if self.style == 0
v
elsif (@type == :time && v.is_a?(Time)) || (@type == :time && v.respond_to?(:to_time))
self.style = STYLE_DATE if self.style == 0
v.respond_to?(:to_time) ? v.to_time : v
elsif @type == :float
v.to_f
elsif @type == :integer
v.to_i
elsif @type == :boolean
v ? 1 : 0
else
@type = :string
::CGI.escapeHTML(v.to_s)
end
end
end
end