fat_table 0.9.5 → 0.9.8
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +4 -4
- data/.envrc +1 -0
- data/.gitignore +1 -0
- data/.rubocop.yml +9 -0
- data/Gemfile +15 -0
- data/README.org +0 -80
- data/Rakefile +5 -1
- data/fat_table.gemspec +6 -18
- data/lib/ext/array.rb +2 -2
- data/lib/fat_table/column.rb +11 -8
- data/lib/fat_table/convert.rb +22 -11
- data/lib/fat_table/evaluator.rb +4 -1
- data/lib/fat_table/footer.rb +9 -6
- data/lib/fat_table/formatters/aoa_formatter.rb +1 -1
- data/lib/fat_table/formatters/aoh_formatter.rb +1 -1
- data/lib/fat_table/formatters/formatter.rb +76 -69
- data/lib/fat_table/formatters/latex_formatter.rb +13 -63
- data/lib/fat_table/formatters/term_formatter.rb +5 -5
- data/lib/fat_table/formatters/xcolors.txt +337 -0
- data/lib/fat_table/patches.rb +6 -9
- data/lib/fat_table/table.rb +79 -91
- data/lib/fat_table/version.rb +1 -1
- data/lib/fat_table.rb +2 -0
- metadata +10 -165
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: baa77411122a8531eae63a312545846cb1a496a808e57aa8e2660dc720950144
|
4
|
+
data.tar.gz: d4000cda7c6b01dc18bf0baf8a9b6150882ae2189f2e3a1ccfcc750cc8f952d2
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 56752060d0e589fb35b71d31f046a01fd307f6958f6383d6ae564dfcd12981f7ed41223b0266f693df281ef04fb412bc748bb448d7b65fceb4331135a6d87816
|
7
|
+
data.tar.gz: f54634a20445d7715cb8608bd4eef5851229e2a2085dd1924bb8279c6f2779db879aec281de57d810715fee388e116abba70ff7f0956d70a8d2f8de9fa76cf01
|
data/.envrc
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
PATH_add .bundle/bin
|
data/.gitignore
CHANGED
data/.rubocop.yml
CHANGED
@@ -2,6 +2,9 @@ inherit_from:
|
|
2
2
|
- ~/.rubocop.yml
|
3
3
|
|
4
4
|
AllCops:
|
5
|
+
Include:
|
6
|
+
- 'lib/**/*'
|
7
|
+
- 'spec/**/*'
|
5
8
|
Exclude:
|
6
9
|
- 'test/tmp/**/*'
|
7
10
|
- 'vendor/bundle/**/*'
|
@@ -14,3 +17,9 @@ Style/MethodCallWithArgsParentheses:
|
|
14
17
|
Style/ClassAndModuleChildren:
|
15
18
|
Exclude:
|
16
19
|
- 'test/**/*'
|
20
|
+
|
21
|
+
Naming/VariableName:
|
22
|
+
Enabled: false
|
23
|
+
|
24
|
+
RSpec/VariableName:
|
25
|
+
Enabled: false
|
data/Gemfile
CHANGED
@@ -4,3 +4,18 @@ gem 'rubocop-shopify', require: false
|
|
4
4
|
|
5
5
|
# Specify your gem's dependencies in fat_table.gemspec
|
6
6
|
gemspec
|
7
|
+
|
8
|
+
gem 'bundler'
|
9
|
+
gem 'debug', '>= 1.0.0'
|
10
|
+
gem 'pry'
|
11
|
+
gem 'pry-doc'
|
12
|
+
gem 'rake', '~> 13.0'
|
13
|
+
gem 'redcarpet'
|
14
|
+
gem 'pg'
|
15
|
+
gem 'sqlite3'
|
16
|
+
gem 'rspec', '~> 3.0'
|
17
|
+
gem 'rubocop'
|
18
|
+
gem 'rubocop-rspec'
|
19
|
+
gem 'rubocop-rake'
|
20
|
+
gem 'rubocop-performance'
|
21
|
+
gem 'simplecov'
|
data/README.org
CHANGED
@@ -79,90 +79,10 @@ org-mode buffer as an org-table, ready for processing by other code blocks.
|
|
79
79
|
- [[#quick-start][Quick Start]]
|
80
80
|
- [[#a-word-about-the-examples][A Word About the Examples]]
|
81
81
|
- [[#anatomy-of-a-table][Anatomy of a Table]]
|
82
|
-
- [[#columns][Columns]]
|
83
|
-
- [[#headers][Headers]]
|
84
|
-
- [[#groups][Groups]]
|
85
82
|
- [[#constructing-tables][Constructing Tables]]
|
86
|
-
- [[#empty-tables][Empty Tables]]
|
87
|
-
- [[#without-headers][Without Headers]]
|
88
|
-
- [[#with-headers][With Headers]]
|
89
|
-
- [[#forcing-string-type][Forcing String Type]]
|
90
|
-
- [[#designating-tolerant-columns][Designating "Tolerant" Columns]]
|
91
|
-
- [[#from-csv-or-org-mode-files-or-strings][From CSV or Org Mode files or strings]]
|
92
|
-
- [[#from-arrays-of-arrays][From Arrays of Arrays]]
|
93
|
-
- [[#in-ruby-code][In Ruby Code]]
|
94
|
-
- [[#in-emacs-org-files][In Emacs Org Files]]
|
95
|
-
- [[#from-arrays-of-hashes][From Arrays of Hashes]]
|
96
|
-
- [[#from-sql-queries][From SQL queries]]
|
97
|
-
- [[#marking-groups-in-input][Marking Groups in Input]]
|
98
|
-
- [[#manually][Manually]]
|
99
|
-
- [[#when-reading-in-tables][When Reading in Tables]]
|
100
83
|
- [[#accessing-parts-of-tables][Accessing Parts of Tables]]
|
101
|
-
- [[#rows][Rows]]
|
102
|
-
- [[#columns-1][Columns]]
|
103
|
-
- [[#cells][Cells]]
|
104
|
-
- [[#other-table-attributes][Other table attributes]]
|
105
84
|
- [[#operations-on-tables][Operations on Tables]]
|
106
|
-
- [[#example-input-tables][Example Input Tables]]
|
107
|
-
- [[#select][Select]]
|
108
|
-
- [[#selecting-existing-columns-also-of-omni][Selecting Existing Columns (Also of :omni)]]
|
109
|
-
- [[#copying-and-renaming-existing-columns][Copying and Renaming Existing Columns.]]
|
110
|
-
- [[#adding-new-columns][Adding New Columns]]
|
111
|
-
- [[#adding-constant-strings-and-other-types-in-select][Adding Constant Strings and Other Types in select]]
|
112
|
-
- [[#custom-instance-variables-and-hooks][Custom Instance Variables and Hooks]]
|
113
|
-
- [[#argument-order-and-boundaries][Argument Order and Boundaries]]
|
114
|
-
- [[#where][Where]]
|
115
|
-
- [[#order_by][Order_by]]
|
116
|
-
- [[#order_with][Order_with]]
|
117
|
-
- [[#group_by][Group_by]]
|
118
|
-
- [[#join][Join]]
|
119
|
-
- [[#join-types][Join Types]]
|
120
|
-
- [[#join-expressions][Join Expressions]]
|
121
|
-
- [[#join-examples][Join Examples]]
|
122
|
-
- [[#inner-joins][Inner Joins]]
|
123
|
-
- [[#left-and-right-joins][Left and Right Joins]]
|
124
|
-
- [[#full-join][Full Join]]
|
125
|
-
- [[#cross-join][Cross Join]]
|
126
|
-
- [[#set-operations][Set Operations]]
|
127
|
-
- [[#unions][Unions]]
|
128
|
-
- [[#intersections][Intersections]]
|
129
|
-
- [[#set-differences-with-except][Set Differences with Except]]
|
130
|
-
- [[#uniq-aka-distinct][Uniq (aka Distinct)]]
|
131
|
-
- [[#remove-groups-with-degroup][Remove groups with degroup!]]
|
132
85
|
- [[#formatting-tables][Formatting Tables]]
|
133
|
-
- [[#available-formatter-output-targets][Available Formatter Output Targets]]
|
134
|
-
- [[#output-media][Output Media]]
|
135
|
-
- [[#examples][Examples]]
|
136
|
-
- [[#to-text][To Text]]
|
137
|
-
- [[#to-org][To Org]]
|
138
|
-
- [[#to-term][To Term]]
|
139
|
-
- [[#to-latex][To LaTeX]]
|
140
|
-
- [[#to-aoa-array-of-arrays][To AoA (Array of Arrays)]]
|
141
|
-
- [[#to-aoh-array-of-hashes][To AoH (Array of Hashes)]]
|
142
|
-
- [[#formatting-directives][Formatting Directives]]
|
143
|
-
- [[#all-types-as-strings][All Types as Strings]]
|
144
|
-
- [[#numeric][Numeric]]
|
145
|
-
- [[#datetime][DateTime]]
|
146
|
-
- [[#boolean][Boolean]]
|
147
|
-
- [[#nilclass][NilClass]]
|
148
|
-
- [[#the-format-and-format_for-methods][The ~format~ and ~format_for~ methods]]
|
149
|
-
- [[#table-locations][Table Locations]]
|
150
|
-
- [[#location-priority][Location priority]]
|
151
|
-
- [[#type-and-column-priority][Type and Column priority]]
|
152
|
-
- [[#footers][Footers]]
|
153
|
-
- [[#adding-footers][Adding Footers]]
|
154
|
-
- [[#dynamic-labels][Dynamic Labels]]
|
155
|
-
- [[#aggregators][Aggregators]]
|
156
|
-
- [[#footer-objects][Footer objects]]
|
157
|
-
- [[#footer-examples][Footer Examples]]
|
158
|
-
- [[#built-in-aggregators][Built-in Aggregators]]
|
159
|
-
- [[#string-aggregators][String Aggregators]]
|
160
|
-
- [[#ruby-objects][Ruby Objects]]
|
161
|
-
- [[#lambdas][Lambdas]]
|
162
|
-
- [[#invoking-formatters][Invoking Formatters]]
|
163
|
-
- [[#by-instantiating-a-formatter][By Instantiating a Formatter]]
|
164
|
-
- [[#by-using-fattable-module-level-method-calls][By Using ~FatTable~ module-level method calls]]
|
165
|
-
- [[#by-calling-methods-on-table-objects][By Calling Methods on Table Objects]]
|
166
86
|
- [[#development][Development]]
|
167
87
|
- [[#contributing][Contributing]]
|
168
88
|
|
data/Rakefile
CHANGED
@@ -2,6 +2,10 @@ require 'bundler/gem_tasks'
|
|
2
2
|
require 'rspec/core/rake_task'
|
3
3
|
require 'rdoc/task'
|
4
4
|
|
5
|
+
require 'rubocop/rake_task'
|
6
|
+
|
7
|
+
RuboCop::RakeTask.new
|
8
|
+
|
5
9
|
RDoc::Task.new do |rdoc|
|
6
10
|
rdoc.main = 'README.rdoc'
|
7
11
|
rdoc.rdoc_files.include('README.rdoc', 'lib')
|
@@ -9,4 +13,4 @@ end
|
|
9
13
|
|
10
14
|
RSpec::Core::RakeTask.new(:spec)
|
11
15
|
|
12
|
-
task default: :
|
16
|
+
task :default => [:spec, :rubocop]
|
data/fat_table.gemspec
CHANGED
@@ -55,7 +55,7 @@ Gem::Specification.new do |spec|
|
|
55
55
|
'public gem pushes.'
|
56
56
|
end
|
57
57
|
|
58
|
-
spec.files =
|
58
|
+
spec.files = %x(git ls-files -z).split("\x0").reject do |f|
|
59
59
|
f.match(%r{^(test|spec|features)/})
|
60
60
|
end
|
61
61
|
spec.bindir = 'bin'
|
@@ -63,21 +63,9 @@ Gem::Specification.new do |spec|
|
|
63
63
|
spec.require_paths = ['lib']
|
64
64
|
spec.metadata['yard.run'] = 'yri' # use "yard" to build full HTML docs.
|
65
65
|
|
66
|
-
spec.
|
67
|
-
spec.
|
68
|
-
spec.
|
69
|
-
spec.
|
70
|
-
spec.
|
71
|
-
spec.add_development_dependency 'redcarpet'
|
72
|
-
spec.add_development_dependency 'pg'
|
73
|
-
spec.add_development_dependency 'sqlite3'
|
74
|
-
spec.add_development_dependency 'rspec', '~> 3.0'
|
75
|
-
spec.add_development_dependency 'rubocop-rspec'
|
76
|
-
spec.add_development_dependency 'rubocop-performance'
|
77
|
-
spec.add_development_dependency 'simplecov'
|
78
|
-
|
79
|
-
spec.add_runtime_dependency 'fat_core', '>= 4.9.0'
|
80
|
-
spec.add_runtime_dependency 'rainbow'
|
81
|
-
spec.add_runtime_dependency 'sequel'
|
82
|
-
spec.add_runtime_dependency 'gem-path'
|
66
|
+
spec.add_dependency 'csv'
|
67
|
+
spec.add_dependency 'fat_core', '>= 4.9.0'
|
68
|
+
spec.add_dependency 'gem-path'
|
69
|
+
spec.add_dependency 'rainbow'
|
70
|
+
spec.add_dependency 'sequel'
|
83
71
|
end
|
data/lib/ext/array.rb
CHANGED
@@ -18,11 +18,11 @@ class Array
|
|
18
18
|
compact.select { |i| i.is_a?(TrueClass) || i.is_a?(FalseClass) }
|
19
19
|
elsif typ == 'DateTime'
|
20
20
|
compact.select { |i| i.is_a?(Date) || i.is_a?(DateTime) || i.is_a?(Time) }
|
21
|
-
.map
|
21
|
+
.map(&:to_datetime)
|
22
22
|
elsif typ == 'Numeric'
|
23
23
|
compact.select { |i| i.is_a?(Numeric) }
|
24
24
|
elsif typ == 'String'
|
25
|
-
map
|
25
|
+
map(&:to_s)
|
26
26
|
elsif typ == 'NilClass'
|
27
27
|
self
|
28
28
|
else
|
data/lib/fat_table/column.rb
CHANGED
@@ -174,9 +174,9 @@ module FatTable
|
|
174
174
|
# Yield each item in the Column in the order in which they appear in the
|
175
175
|
# Column. This makes Columns Enumerable, so all the Enumerable methods are
|
176
176
|
# available on a Column.
|
177
|
-
def each
|
178
|
-
if
|
179
|
-
items.each
|
177
|
+
def each(&block)
|
178
|
+
if block
|
179
|
+
items.each(&block)
|
180
180
|
self
|
181
181
|
else
|
182
182
|
to_enum(:each)
|
@@ -201,7 +201,7 @@ module FatTable
|
|
201
201
|
# Return the first non-nil item in the Column, or nil if all items are
|
202
202
|
# nil. Works with any Column type.
|
203
203
|
def first
|
204
|
-
return
|
204
|
+
return if items.all?(&:nil?)
|
205
205
|
|
206
206
|
if type == 'String'
|
207
207
|
items.reject(&:blank?).first
|
@@ -214,7 +214,7 @@ module FatTable
|
|
214
214
|
|
215
215
|
# Return the last non-nil item in the Column. Works with any Column type.
|
216
216
|
def last
|
217
|
-
return
|
217
|
+
return if items.all?(&:nil?)
|
218
218
|
|
219
219
|
if type == 'String'
|
220
220
|
items.reject(&:blank?).last
|
@@ -231,7 +231,7 @@ module FatTable
|
|
231
231
|
return items.size if items.all?(&:nil?)
|
232
232
|
|
233
233
|
if type == 'String'
|
234
|
-
items.
|
234
|
+
items.count { |i| !i.blank? }
|
235
235
|
else
|
236
236
|
items.filter_to_type(type).count.to_d
|
237
237
|
end
|
@@ -270,7 +270,7 @@ module FatTable
|
|
270
270
|
# Columns.
|
271
271
|
def range
|
272
272
|
only_with('range', 'NilClass', 'Numeric', 'String', 'DateTime')
|
273
|
-
return
|
273
|
+
return if items.all?(&:nil?)
|
274
274
|
|
275
275
|
Range.new(min, max)
|
276
276
|
end
|
@@ -304,7 +304,7 @@ module FatTable
|
|
304
304
|
itms = items.filter_to_type(type)
|
305
305
|
size = itms.size.to_d
|
306
306
|
if type == 'DateTime'
|
307
|
-
avg_jd = itms.
|
307
|
+
avg_jd = itms.sum(&:jd) / size
|
308
308
|
DateTime.jd(avg_jd)
|
309
309
|
else
|
310
310
|
itms.sum / size
|
@@ -331,6 +331,7 @@ module FatTable
|
|
331
331
|
end
|
332
332
|
n = count
|
333
333
|
return BigDecimal('0.0') if n <= 1
|
334
|
+
|
334
335
|
mu = Column.new(header: :mu, items: all_items).avg
|
335
336
|
sq_dev = BigDecimal('0.0')
|
336
337
|
all_items.each do |itm|
|
@@ -353,6 +354,7 @@ module FatTable
|
|
353
354
|
only_with('var', 'DateTime', 'Numeric')
|
354
355
|
n = items.filter_to_type(type).size.to_d
|
355
356
|
return BigDecimal('0.0') if n <= 1
|
357
|
+
|
356
358
|
var * ((n - 1) / n)
|
357
359
|
end
|
358
360
|
|
@@ -471,6 +473,7 @@ module FatTable
|
|
471
473
|
def +(other)
|
472
474
|
msg = 'cannot combine columns with different types'
|
473
475
|
raise UserError, msg unless type == other.type
|
476
|
+
|
474
477
|
Column.new(header: header, items: items + other.items)
|
475
478
|
end
|
476
479
|
|
data/lib/fat_table/convert.rb
CHANGED
@@ -38,6 +38,7 @@ module FatTable
|
|
38
38
|
if new_val.nil?
|
39
39
|
raise IncompatibleTypeError
|
40
40
|
end
|
41
|
+
|
41
42
|
new_val
|
42
43
|
end
|
43
44
|
when 'DateTime'
|
@@ -48,6 +49,7 @@ module FatTable
|
|
48
49
|
if new_val.nil?
|
49
50
|
raise IncompatibleTypeError
|
50
51
|
end
|
52
|
+
|
51
53
|
new_val
|
52
54
|
end
|
53
55
|
when 'Numeric'
|
@@ -58,6 +60,7 @@ module FatTable
|
|
58
60
|
if new_val.nil?
|
59
61
|
raise IncompatibleTypeError
|
60
62
|
end
|
63
|
+
|
61
64
|
new_val
|
62
65
|
end
|
63
66
|
when 'String'
|
@@ -68,6 +71,7 @@ module FatTable
|
|
68
71
|
if new_val.nil?
|
69
72
|
raise IncompatibleTypeError
|
70
73
|
end
|
74
|
+
|
71
75
|
new_val
|
72
76
|
end
|
73
77
|
else
|
@@ -80,8 +84,10 @@ module FatTable
|
|
80
84
|
# of case is assumed to be a boolean.
|
81
85
|
def self.convert_to_boolean(val)
|
82
86
|
return val if val.is_a?(TrueClass) || val.is_a?(FalseClass)
|
87
|
+
|
83
88
|
val = val.to_s.clean
|
84
|
-
return
|
89
|
+
return if val.blank?
|
90
|
+
|
85
91
|
if val.match?(/\A(false|f|n|no)\z/i)
|
86
92
|
false
|
87
93
|
elsif val.match?(/\A(true|t|y|yes)\z/i)
|
@@ -94,7 +100,7 @@ module FatTable
|
|
94
100
|
(?<dy>\d\d?)\s*
|
95
101
|
(T?\s*\d\d:\d\d(:\d\d)?\s*
|
96
102
|
([-+](\d\d?)(:\d\d?))?)?\s*
|
97
|
-
((mon|tue|wed|thu|fri|sat|sun)[a-zA-
|
103
|
+
((mon|tue|wed|thu|fri|sat|sun)[a-zA-Z]*)?\s*
|
98
104
|
\z}xi
|
99
105
|
|
100
106
|
AMR_DATE_RE = %r{\A(?<dy>\d\d?)[-/](?<mo>\d\d?)[-/](?<yr>\d\d\d\d)\s*
|
@@ -103,7 +109,7 @@ module FatTable
|
|
103
109
|
# A Date like 'Tue, 01 Nov 2016' or 'Tue 01 Nov 2016' or '01 Nov 2016'.
|
104
110
|
# These are emitted by Postgresql, so it makes from_sql constructor
|
105
111
|
# possible without special formatting of the dates.
|
106
|
-
INV_DATE_RE = %r{\A((mon|tue|wed|thu|fri|sat|sun)[a-zA-z]*,?)?\s+ # looks like dow
|
112
|
+
INV_DATE_RE = %r{\A((mon|tue|wed|thu|fri|sat|sun)[a-zA-Za-z]*,?)?\s+ # looks like dow
|
107
113
|
(?<dy>\d\d?)\s+ # one or two-digit day
|
108
114
|
(?<mo_name>[jfmasondJFMASOND][A-Za-z]{2,})\s+ # looks like a month name
|
109
115
|
(?<yr>\d\d\d\d) # and a 4-digit year
|
@@ -122,20 +128,25 @@ module FatTable
|
|
122
128
|
|
123
129
|
begin
|
124
130
|
str = val.to_s.clean.sub(/\A[\[\(\{\<]\s*/, '').sub(/[\]\)\}\>]\s*\z/, '')
|
125
|
-
return
|
131
|
+
return if str.blank?
|
126
132
|
|
127
133
|
if str.match(ISO_DATE_RE)
|
128
134
|
date = DateTime.parse(val)
|
129
135
|
elsif str =~ AMR_DATE_RE
|
130
|
-
date = DateTime.new(
|
131
|
-
|
132
|
-
|
136
|
+
date = DateTime.new(
|
137
|
+
Regexp.last_match[:yr].to_i,
|
138
|
+
Regexp.last_match[:mo].to_i,
|
139
|
+
Regexp.last_match[:dy].to_i,
|
140
|
+
)
|
133
141
|
elsif str =~ INV_DATE_RE
|
134
142
|
mo = Date.mo_name_to_num(last_match[:mo_name])
|
135
|
-
date = DateTime.new(
|
136
|
-
|
143
|
+
date = DateTime.new(
|
144
|
+
Regexp.last_match[:yr].to_i,
|
145
|
+
mo,
|
146
|
+
Regexp.last_match[:dy].to_i,
|
147
|
+
)
|
137
148
|
else
|
138
|
-
return
|
149
|
+
return
|
139
150
|
end
|
140
151
|
# val = val.to_date if
|
141
152
|
date.seconds_since_midnight.zero? ? date.to_date : date
|
@@ -155,7 +166,7 @@ module FatTable
|
|
155
166
|
cursym = Regexp.quote(FatTable.currency_symbol)
|
156
167
|
clean_re = /[,_#{cursym}]/
|
157
168
|
val = val.to_s.clean.gsub(clean_re, '')
|
158
|
-
return
|
169
|
+
return if val.blank?
|
159
170
|
|
160
171
|
case val
|
161
172
|
when /\A[-+]?\d+\.\z/
|
data/lib/fat_table/evaluator.rb
CHANGED
@@ -50,11 +50,14 @@ module FatTable
|
|
50
50
|
# errors simply return nil as the result. This can happen, for example,
|
51
51
|
# when a string gets into an otherwise numeric column because the column
|
52
52
|
# is set to tolerant.
|
53
|
+
#
|
54
|
+
# rubocop:disable Security/Eval
|
53
55
|
def evaluate(expr = '', locals: {})
|
54
56
|
eval(expr, local_vars(binding, locals))
|
55
|
-
rescue NoMethodError, TypeError
|
57
|
+
rescue NoMethodError, TypeError
|
56
58
|
nil
|
57
59
|
end
|
60
|
+
# rubocop:enable Security/Eval
|
58
61
|
|
59
62
|
private
|
60
63
|
|
data/lib/fat_table/footer.rb
CHANGED
@@ -13,18 +13,20 @@ module FatTable
|
|
13
13
|
# Initialize a labeled footer, optionally specifying a column for the
|
14
14
|
# label and whether the footer is to be a group footer. One or more values
|
15
15
|
# for the footer are added later with the #add_value method.
|
16
|
-
def initialize(label
|
16
|
+
def initialize(table, label: 'Total', label_col: nil, group: false)
|
17
17
|
@label = label
|
18
18
|
|
19
19
|
unless table.is_a?(Table)
|
20
20
|
raise ArgumentError, 'Footer.new needs a table argument'
|
21
21
|
end
|
22
|
+
|
22
23
|
if label_col.nil?
|
23
24
|
@label_col = table.headers.first
|
24
25
|
else
|
25
26
|
unless table.headers.include?(label_col.as_sym)
|
26
27
|
raise ArgumentError, "Footer.new label column '#{label_col}' not a header of table."
|
27
28
|
end
|
29
|
+
|
28
30
|
@label_col = label_col.as_sym
|
29
31
|
end
|
30
32
|
@table = table
|
@@ -77,7 +79,7 @@ module FatTable
|
|
77
79
|
# header.
|
78
80
|
def [](key)
|
79
81
|
key = key.as_sym
|
80
|
-
if values.
|
82
|
+
if values.key?(key)
|
81
83
|
if group
|
82
84
|
values[key]
|
83
85
|
else
|
@@ -140,8 +142,6 @@ module FatTable
|
|
140
142
|
hsh[h] =
|
141
143
|
if values[h]
|
142
144
|
values[h].first
|
143
|
-
else
|
144
|
-
nil
|
145
145
|
end
|
146
146
|
end
|
147
147
|
end
|
@@ -243,9 +243,12 @@ module FatTable
|
|
243
243
|
end
|
244
244
|
else
|
245
245
|
if k
|
246
|
-
|
246
|
+
m = "group footer label proc may only have 0, 1, or 2 arguments for group number k and containing footer f"
|
247
|
+
raise ArgumentError, m
|
248
|
+
|
247
249
|
else
|
248
|
-
raise ArgumentError,
|
250
|
+
raise ArgumentError,
|
251
|
+
"a non-group footer label proc may only have 0 or 1 arguments for the containing footer f"
|
249
252
|
end
|
250
253
|
end
|
251
254
|
else
|
@@ -40,7 +40,7 @@ module FatTable
|
|
40
40
|
# eval'ed, we need to escape any single-quotes (') that appear in the
|
41
41
|
# string.
|
42
42
|
def quote_cell(val)
|
43
|
-
if val.
|
43
|
+
if val.include?("'")
|
44
44
|
# Use a negative look-behind to only quote single-quotes that are not
|
45
45
|
# already preceded by a backslash
|
46
46
|
val.gsub(/(?<!\\)'/, "'" => "\\'")
|
@@ -40,7 +40,7 @@ module FatTable
|
|
40
40
|
# eval'ed, we need to escape any single-quotes (') that appear in the
|
41
41
|
# string.
|
42
42
|
def quote_cell(val)
|
43
|
-
if val.
|
43
|
+
if val.include?("'")
|
44
44
|
# Use a negative look-behind to only quote single-quotes that are not
|
45
45
|
# already preceded by a backslash
|
46
46
|
val.gsub(/(?<!\\)'/, "'" => "\\'")
|