line |
stmt |
bran |
cond |
sub |
pod |
time |
code |
1
|
|
|
|
|
|
|
|
2
|
|
|
|
|
|
|
# some private subroutines here are called by RangeGroup, so think of RangeGroup as |
3
|
|
|
|
|
|
|
# a friend of Range. the routines RangeGroup calls are commented thusly: |
4
|
|
|
|
|
|
|
# "private range routine called here!" |
5
|
|
|
|
|
|
|
|
6
|
|
|
|
|
|
|
# there are different ways to specify ranges. this came about by merging different |
7
|
|
|
|
|
|
|
# ideas from different spreadsheet implementations. |
8
|
|
|
|
|
|
|
|
9
|
|
|
|
|
|
|
# column: |
10
|
|
|
|
|
|
|
# A ====> A:A |
11
|
|
|
|
|
|
|
# A:A ====> A:A |
12
|
|
|
|
|
|
|
# [A] ====> A:A |
13
|
|
|
|
|
|
|
# [[A]] ====> A:A |
14
|
|
|
|
|
|
|
# [1] ====> A:A |
15
|
|
|
|
|
|
|
# [[1]] ====> A:A |
16
|
|
|
|
|
|
|
# {col => A} ====> A:A |
17
|
|
|
|
|
|
|
# [{col => A}] ====> A:A |
18
|
|
|
|
|
|
|
# {col => 1} ====> A:A |
19
|
|
|
|
|
|
|
# [{col => 1}] ====> A:A |
20
|
|
|
|
|
|
|
# a partial column is still considered a column |
21
|
|
|
|
|
|
|
# A5:A10 ====> A5:A10 |
22
|
|
|
|
|
|
|
|
23
|
|
|
|
|
|
|
# row: |
24
|
|
|
|
|
|
|
# 1 ====> 1:1 |
25
|
|
|
|
|
|
|
# 1:1 ====> 1:1 |
26
|
|
|
|
|
|
|
# [undef, 1] ====> 1:1 |
27
|
|
|
|
|
|
|
# [[undef, 1]] ====> 1:1 |
28
|
|
|
|
|
|
|
# 0's are 'valid' in that they are ignored. |
29
|
|
|
|
|
|
|
# [0, 1] ====> 1:1 |
30
|
|
|
|
|
|
|
# [[0, 1]] ====> 1:1 |
31
|
|
|
|
|
|
|
# {row => 1} ====> 1:1 |
32
|
|
|
|
|
|
|
# [{row => 1}] ====> 1:1 |
33
|
|
|
|
|
|
|
# a partial row is still considered a row |
34
|
|
|
|
|
|
|
# D1:1 ====> D1:1 |
35
|
|
|
|
|
|
|
|
36
|
|
|
|
|
|
|
# partial rows/columns: |
37
|
|
|
|
|
|
|
# [[A, 5], [A]] ====> A5:A |
38
|
|
|
|
|
|
|
# [{col => A, row => 5}, {col => A}] ====> A5:A |
39
|
|
|
|
|
|
|
# [[A], [A, 5]] ====> A:A5 |
40
|
|
|
|
|
|
|
# [{col => A}, {col => A, row => 5}] ====> A:A5 |
41
|
|
|
|
|
|
|
# [[5, 1], [undef, 1]] ====> E1:1 |
42
|
|
|
|
|
|
|
# [[5, 1], [0, 1]] ====> E1:1 |
43
|
|
|
|
|
|
|
# [{col => 5, row => 1}, {row => 1}] ====> E1:1 |
44
|
|
|
|
|
|
|
|
45
|
|
|
|
|
|
|
# single cell: |
46
|
|
|
|
|
|
|
# A1 ====> A1 |
47
|
|
|
|
|
|
|
# [A, 1] ====> A1 |
48
|
|
|
|
|
|
|
# [[A, 1]] ====> A1 |
49
|
|
|
|
|
|
|
# [1, 1] ====> A1 |
50
|
|
|
|
|
|
|
# [[1, 1]] ====> A1 |
51
|
|
|
|
|
|
|
# {col => A, row => 1} ====> A1 |
52
|
|
|
|
|
|
|
# [{col => A, row => 1}] ====> A1 |
53
|
|
|
|
|
|
|
# {col => 1, row => 1} ====> A1 |
54
|
|
|
|
|
|
|
# [{col => 1, row => 1}] ====> A1 |
55
|
|
|
|
|
|
|
|
56
|
|
|
|
|
|
|
# same cell twice gets reduced to single cell: |
57
|
|
|
|
|
|
|
# A1:A1 ====> A1 |
58
|
|
|
|
|
|
|
# these only get reduced using the factory method. |
59
|
|
|
|
|
|
|
# [[A, 1], [A, 1]] ====> A1 |
60
|
|
|
|
|
|
|
# [[1, 1], [1, 1]] ====> A1 |
61
|
|
|
|
|
|
|
# [{col => A, row => 1}, {col => A, row => 1}] ====> A1 |
62
|
|
|
|
|
|
|
# [{col => 1, row => 1}, {col => 1, row => 1}] ====> A1 |
63
|
|
|
|
|
|
|
|
64
|
|
|
|
|
|
|
# general ranges: |
65
|
|
|
|
|
|
|
# A1:B2 ====> A1:B2 |
66
|
|
|
|
|
|
|
# [[1,1],[2,2]] ====> A1:B2 |
67
|
|
|
|
|
|
|
# [[A,1],[B,2]] ====> A1:B2 |
68
|
|
|
|
|
|
|
# [{col => 1, row => 1}, {col => 2, row => 2}] ====> A1:B2 |
69
|
|
|
|
|
|
|
# [{col => A, row => 1}, {col => B, row => 2}] ====> A1:B2 |
70
|
|
|
|
|
|
|
|
71
|
|
|
|
|
|
|
# mixing is ok: |
72
|
|
|
|
|
|
|
# [A1, [2, 2]] ====> A1:B2 |
73
|
|
|
|
|
|
|
# [{col => 1, row => 1}, [2, 2]] ====> A1:B2 |
74
|
|
|
|
|
|
|
# [{col => 1. row => 1}, B2] ====> A1:B2 |
75
|
|
|
|
|
|
|
|
76
|
|
|
|
|
|
|
# bad ranges: |
77
|
|
|
|
|
|
|
# should be able to support this but makes range routine |
78
|
|
|
|
|
|
|
# too complex, and the simple workaround is to just A1:B2. |
79
|
|
|
|
|
|
|
# [A1, B2] |
80
|
|
|
|
|
|
|
|
81
|
|
|
|
|
|
|
our $VERSION = '1.0.2'; |
82
|
|
|
|
|
|
|
|
83
|
|
|
|
|
|
|
use Google::RestApi::Setup; |
84
|
1
|
|
|
1
|
|
596
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
7
|
|
85
|
|
|
|
|
|
|
use Carp qw( confess ); |
86
|
1
|
|
|
1
|
|
19193
|
use List::Util qw( max ); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
46
|
|
87
|
1
|
|
|
1
|
|
8
|
use Readonly; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
50
|
|
88
|
1
|
|
|
1
|
|
495
|
use Scalar::Util qw( looks_like_number ); |
|
1
|
|
|
|
|
3596
|
|
|
1
|
|
|
|
|
49
|
|
89
|
1
|
|
|
1
|
|
6
|
use Try::Tiny qw( try catch ); |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
48
|
|
90
|
1
|
|
|
1
|
|
5
|
|
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
41
|
|
91
|
|
|
|
|
|
|
use experimental qw( switch ); |
92
|
1
|
|
|
1
|
|
501
|
|
|
1
|
|
|
|
|
3136
|
|
|
1
|
|
|
|
|
5
|
|
93
|
|
|
|
|
|
|
use aliased 'Google::RestApi::SheetsApi4::Range::Col'; |
94
|
1
|
|
|
1
|
|
88
|
use aliased 'Google::RestApi::SheetsApi4::Range::Row'; |
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
7
|
|
95
|
1
|
|
|
1
|
|
85
|
use aliased 'Google::RestApi::SheetsApi4::Range::Cell'; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
3
|
|
96
|
1
|
|
|
1
|
|
83
|
use aliased 'Google::RestApi::SheetsApi4::Range::Iterator'; |
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
4
|
|
97
|
1
|
|
|
1
|
|
146
|
|
|
1
|
|
|
|
|
2
|
|
|
1
|
|
|
|
|
4
|
|
98
|
|
|
|
|
|
|
use parent 'Google::RestApi::SheetsApi4::Request::Spreadsheet::Worksheet::Range'; |
99
|
1
|
|
|
1
|
|
144
|
|
|
1
|
|
|
|
|
3
|
|
|
1
|
|
|
|
|
8
|
|
100
|
|
|
|
|
|
|
Readonly::Scalar our $RANGE_EXPANDED => 1; |
101
|
|
|
|
|
|
|
|
102
|
|
|
|
|
|
|
# this routine returns the best fitting object for the range specified. |
103
|
|
|
|
|
|
|
# A5:A10 will return a Col object. A5:J5 will return a Row object. Etc. |
104
|
|
|
|
|
|
|
# this grew organically into something more complicated that it was worth, |
105
|
|
|
|
|
|
|
# but i got it working reliably so, whatever... |
106
|
|
|
|
|
|
|
my %original_range_args = @_; |
107
|
|
|
|
|
|
|
my $original_range = $original_range_args{range}; |
108
|
90
|
|
|
90
|
0
|
431
|
|
109
|
90
|
|
|
|
|
330
|
state $check = multisig( |
110
|
|
|
|
|
|
|
compile_named( |
111
|
90
|
|
|
|
|
258
|
worksheet => HasApi, |
112
|
|
|
|
|
|
|
range => RangeCol, |
113
|
|
|
|
|
|
|
dim => DimColRow, { optional => 1 }, # is switched to 'col'. |
114
|
|
|
|
|
|
|
), |
115
|
|
|
|
|
|
|
compile_named( |
116
|
|
|
|
|
|
|
worksheet => HasApi, |
117
|
|
|
|
|
|
|
range => RangeRow, |
118
|
|
|
|
|
|
|
dim => DimColRow, { optional => 1 }, # is switched to 'row' |
119
|
|
|
|
|
|
|
), |
120
|
|
|
|
|
|
|
compile_named( |
121
|
|
|
|
|
|
|
worksheet => HasApi, |
122
|
|
|
|
|
|
|
range => RangeCell, |
123
|
|
|
|
|
|
|
dim => DimColRow, { optional => 1 }, # doesn't matter for cells. |
124
|
|
|
|
|
|
|
), |
125
|
|
|
|
|
|
|
compile_named( |
126
|
|
|
|
|
|
|
worksheet => HasApi, |
127
|
|
|
|
|
|
|
range => RangeAny, |
128
|
|
|
|
|
|
|
dim => DimColRow, { default => 'row' }, |
129
|
|
|
|
|
|
|
), |
130
|
|
|
|
|
|
|
compile_named( |
131
|
|
|
|
|
|
|
worksheet => HasApi, |
132
|
|
|
|
|
|
|
range => RangeNamed, |
133
|
|
|
|
|
|
|
dim => DimColRow, { default => 'row' }, |
134
|
|
|
|
|
|
|
), |
135
|
|
|
|
|
|
|
); |
136
|
|
|
|
|
|
|
my @range_args = $check->(@_); |
137
|
|
|
|
|
|
|
my $range_args = $range_args[0]; # no idea why it returns an arrayref pointer to a hashref. |
138
|
90
|
|
|
|
|
9340
|
my $range = $range_args->{range}; |
139
|
90
|
|
|
|
|
11021
|
|
140
|
90
|
|
|
|
|
374
|
# be careful here, recursive. |
141
|
|
|
|
|
|
|
given (${^TYPE_PARAMS_MULTISIG}) { |
142
|
|
|
|
|
|
|
when (0) { return Col->new(%$range_args); } |
143
|
90
|
|
|
|
|
195
|
when (1) { return Row->new(%$range_args); } |
144
|
90
|
|
|
|
|
367
|
when (2) { return Cell->new(%$range_args); } |
|
11
|
|
|
|
|
160
|
|
145
|
79
|
|
|
|
|
199
|
|
|
12
|
|
|
|
|
152
|
|
146
|
67
|
|
|
|
|
203
|
# if we've translated a range, it now may be a better fit for one of the above. |
|
43
|
|
|
|
|
638
|
|
147
|
|
|
|
|
|
|
when (3) { |
148
|
|
|
|
|
|
|
# convert the range to A1:A1 format and redrive the factory routine to see |
149
|
24
|
|
|
|
|
80
|
# if it ends up being a col, row, or cell range. |
150
|
|
|
|
|
|
|
if ($range ne $original_range) { |
151
|
|
|
|
|
|
|
DEBUG(sprintf("Range '%s' converted to '$range'", flatten_range($original_range))); |
152
|
23
|
100
|
|
|
|
90
|
# resolve cells by collapsing A1:A1 to just A1. also A:A and 1:1 will be |
153
|
9
|
|
|
|
|
93
|
# properly resolved to cols/rows. |
154
|
|
|
|
|
|
|
my ($start, $end) = split(':', $range); |
155
|
|
|
|
|
|
|
$range = $start if $end && $start eq $end; |
156
|
9
|
|
|
|
|
129
|
return factory(%$range_args); ##### recursion |
157
|
9
|
50
|
66
|
|
|
56
|
} |
158
|
9
|
|
|
|
|
65
|
# we're already in A1:A1 format so just create a new range object. |
159
|
|
|
|
|
|
|
return __PACKAGE__->new(%$range_args); |
160
|
|
|
|
|
|
|
} |
161
|
14
|
|
|
|
|
143
|
|
162
|
|
|
|
|
|
|
# range could be a named range or a column/row header. we have to resolve the |
163
|
|
|
|
|
|
|
# range first, then see what the best fit will be above. |
164
|
|
|
|
|
|
|
when (4) { |
165
|
|
|
|
|
|
|
my $worksheet = $range_args->{worksheet}; |
166
|
1
|
|
|
|
|
3
|
|
167
|
1
|
|
|
|
|
3
|
my $named = $range; |
168
|
|
|
|
|
|
|
$range = $worksheet->resolve_header_range($named); |
169
|
1
|
|
|
|
|
3
|
if ($range) { |
170
|
1
|
|
|
|
|
6
|
$range = factory(%$range_args, range => $range); ##### recursion |
171
|
1
|
50
|
|
|
|
4
|
$range->{header_name} = $named; |
172
|
0
|
|
|
|
|
0
|
return $range; |
173
|
0
|
|
|
|
|
0
|
} |
174
|
0
|
|
|
|
|
0
|
|
175
|
|
|
|
|
|
|
$range = $worksheet->normalize_named($named) |
176
|
|
|
|
|
|
|
or LOGDIE("Unable to resolve named range '$named'"); |
177
|
1
|
50
|
|
|
|
6
|
# we've resolved the name to A1 format, so redrive factory routine to |
178
|
|
|
|
|
|
|
# generate a range object with the resolved range. |
179
|
|
|
|
|
|
|
$range = factory(%$range_args, range => $range); ##### recursion |
180
|
|
|
|
|
|
|
$range->{named} = $named; |
181
|
1
|
|
|
|
|
10
|
return $range; |
182
|
1
|
|
|
|
|
4
|
} |
183
|
1
|
|
|
|
|
8
|
} |
184
|
|
|
|
|
|
|
|
185
|
|
|
|
|
|
|
LOGDIE("Unable to resolve '$range_args->{range}' to a range object"); |
186
|
|
|
|
|
|
|
} |
187
|
0
|
|
|
|
|
0
|
|
188
|
|
|
|
|
|
|
# this should not normally be called directly, everything should be created via |
189
|
|
|
|
|
|
|
# the factory method above or more commonly via the 'worksheet::range*' methods. |
190
|
|
|
|
|
|
|
my $class = shift; |
191
|
|
|
|
|
|
|
|
192
|
|
|
|
|
|
|
# TODO: sort start range and end range before storing. |
193
|
220
|
|
|
220
|
1
|
627
|
|
194
|
|
|
|
|
|
|
state $check = compile_named( |
195
|
|
|
|
|
|
|
worksheet => HasMethods[qw(api worksheet_name)], |
196
|
|
|
|
|
|
|
range => RangeAny, |
197
|
220
|
|
|
|
|
520
|
dim => DimColRow, { default => 'row' }, |
198
|
|
|
|
|
|
|
); |
199
|
|
|
|
|
|
|
my $self = $check->(@_); |
200
|
|
|
|
|
|
|
$self->{dim} = dims_any($self->{dim}); # convert internally to COLUMN | ROW |
201
|
|
|
|
|
|
|
DEBUG("New range " . flatten_range($self->{range}) . " has been created"); |
202
|
220
|
|
|
|
|
6313
|
|
203
|
220
|
|
|
|
|
15652
|
return bless $self, $class; |
204
|
220
|
|
|
|
|
905
|
} |
205
|
|
|
|
|
|
|
|
206
|
220
|
|
|
|
|
3586
|
my $self = shift; |
207
|
|
|
|
|
|
|
my $range = $self->range(); |
208
|
|
|
|
|
|
|
DEBUG("Clearing range '$range'"); |
209
|
|
|
|
|
|
|
my %p = ( |
210
|
0
|
|
|
0
|
1
|
0
|
uri => "/values/$range:clear", |
211
|
0
|
|
|
|
|
0
|
method => 'post', |
212
|
0
|
|
|
|
|
0
|
); |
213
|
0
|
|
|
|
|
0
|
my $response = $self->api(%p); |
214
|
|
|
|
|
|
|
$self->clear_cached_values(); |
215
|
|
|
|
|
|
|
return $response; |
216
|
|
|
|
|
|
|
} |
217
|
0
|
|
|
|
|
0
|
|
218
|
0
|
|
|
|
|
0
|
|
219
|
0
|
|
|
|
|
0
|
my $self = shift; |
220
|
|
|
|
|
|
|
$self->clear_cached_values(); |
221
|
|
|
|
|
|
|
return $self->values(); |
222
|
0
|
|
|
0
|
0
|
0
|
} |
223
|
|
|
|
|
|
|
|
224
|
|
|
|
|
|
|
# this gets or sets the values of a range immediately, no batch. |
225
|
0
|
|
|
0
|
1
|
0
|
# if no values passed, gets them. if values passed, sets them. |
226
|
0
|
|
|
|
|
0
|
my $self = shift; |
227
|
0
|
|
|
|
|
0
|
my %p = @_; |
228
|
|
|
|
|
|
|
$self->_send_values(%p) if defined $p{values}; |
229
|
|
|
|
|
|
|
return $self->_cache_range_values(%p)->{values}; |
230
|
|
|
|
|
|
|
} |
231
|
|
|
|
|
|
|
|
232
|
|
|
|
|
|
|
# immediately update the values for a range (no batch). |
233
|
143
|
|
|
143
|
1
|
257
|
my $self = shift; |
234
|
143
|
|
|
|
|
256
|
|
235
|
143
|
100
|
|
|
|
404
|
state $check = compile_named( |
236
|
143
|
|
|
|
|
405
|
values => ArrayRef[ArrayRef[Str]], { optional => 1 }, |
237
|
|
|
|
|
|
|
params => HashRef, { default => {} }, |
238
|
|
|
|
|
|
|
content => HashRef, { default => {} }, |
239
|
|
|
|
|
|
|
_extra_ => slurpy Any, |
240
|
|
|
|
|
|
|
); |
241
|
3
|
|
|
3
|
|
8
|
my $p = named_extra($check->(@_)); |
242
|
|
|
|
|
|
|
|
243
|
3
|
|
|
|
|
14
|
my $range = $self->range(); |
244
|
|
|
|
|
|
|
|
245
|
|
|
|
|
|
|
# since we're sending these values, fake a response from the |
246
|
|
|
|
|
|
|
# api to store them in our cache. |
247
|
|
|
|
|
|
|
# if includeValuesInResponse is sent in the params, then the |
248
|
|
|
|
|
|
|
# response will replace the cache again. |
249
|
3
|
|
|
|
|
2679
|
$self->_cache_range_values( |
250
|
|
|
|
|
|
|
range => $range, |
251
|
3
|
|
|
|
|
19
|
majorDimension => $self->dimension(), |
252
|
|
|
|
|
|
|
values => $p->{values}, |
253
|
|
|
|
|
|
|
); |
254
|
|
|
|
|
|
|
|
255
|
|
|
|
|
|
|
$p->{content}->{range} = $range; |
256
|
|
|
|
|
|
|
$p->{content}->{values} = delete $p->{values}; |
257
|
|
|
|
|
|
|
$p->{content}->{majorDimension} = $self->dimension(); |
258
|
|
|
|
|
|
|
$p->{params}->{valueInputOption} //= 'USER_ENTERED'; |
259
|
|
|
|
|
|
|
# $p->{params}->{includeValuesInResponse} = 1; |
260
|
|
|
|
|
|
|
$p->{uri} = "/values/$range"; |
261
|
3
|
|
|
|
|
19
|
$p->{method} = 'put'; |
262
|
|
|
|
|
|
|
|
263
|
3
|
|
|
|
|
10
|
my $update = $self->api(%$p); |
264
|
3
|
|
|
|
|
16
|
|
265
|
3
|
|
|
|
|
11
|
# this fakes out the batch response with this immediate response. |
266
|
3
|
|
50
|
|
|
23
|
# batch response would be in an arrayref, so wrap it thusly. |
267
|
|
|
|
|
|
|
return $self->values_response_from_api([ $update ]); |
268
|
3
|
|
|
|
|
12
|
} |
269
|
3
|
|
|
|
|
12
|
|
270
|
|
|
|
|
|
|
# this stores the response to an update request, either batch or immediate: |
271
|
3
|
|
|
|
|
17
|
# --- |
272
|
|
|
|
|
|
|
# spreadsheetId: 1ky2czjhPArP71a6woeo_dxRr8gBOZZxGAPjOCXJvCwA |
273
|
|
|
|
|
|
|
# updatedCells: 6 |
274
|
|
|
|
|
|
|
# updatedColumns: 3 |
275
|
3
|
|
|
|
|
28
|
# updatedRange: Sheet1!B3:D4 |
276
|
|
|
|
|
|
|
# updatedRows: 2 |
277
|
|
|
|
|
|
|
# if includeValuesInResponse was requested, the values are stashed |
278
|
|
|
|
|
|
|
# from the updatedData response key. |
279
|
|
|
|
|
|
|
# spreadsheet object will call this on a batch update response. |
280
|
|
|
|
|
|
|
my $self = shift; |
281
|
|
|
|
|
|
|
|
282
|
|
|
|
|
|
|
state $check = compile(ArrayRef, { optional => 1 }); |
283
|
|
|
|
|
|
|
my ($updates) = $check->(@_); |
284
|
|
|
|
|
|
|
return if !$updates; |
285
|
|
|
|
|
|
|
|
286
|
|
|
|
|
|
|
# shift off the next update from the batch api response. if this is |
287
|
|
|
|
|
|
|
# getting called from the spreadsheet object, it means we have an |
288
|
|
|
|
|
|
|
# update response to process. |
289
|
34
|
|
|
34
|
0
|
66
|
my $update = shift @$updates; |
290
|
|
|
|
|
|
|
|
291
|
34
|
|
|
|
|
56
|
# updatedData is included if includeValuesInResponse query param was sent. |
292
|
34
|
|
|
|
|
680
|
# updatedData: |
293
|
34
|
50
|
|
|
|
455
|
# majorDimension: ROWS |
294
|
|
|
|
|
|
|
# range: Sheet1!A1 |
295
|
|
|
|
|
|
|
# values: |
296
|
|
|
|
|
|
|
# - - Fred |
297
|
|
|
|
|
|
|
# if the data is included, then replace any cached values with this latest |
298
|
34
|
|
|
|
|
73
|
# updated set of values. |
299
|
|
|
|
|
|
|
$self->_cache_range_values(%{ delete $update->{updatedData} }) |
300
|
|
|
|
|
|
|
if $update->{updatedData}; |
301
|
|
|
|
|
|
|
$self->{values_response_from_api} = $update; |
302
|
|
|
|
|
|
|
|
303
|
|
|
|
|
|
|
return $self->values(); |
304
|
|
|
|
|
|
|
} |
305
|
|
|
|
|
|
|
|
306
|
|
|
|
|
|
|
# this returns the values and major dimension for a given range. it will store |
307
|
|
|
|
|
|
|
# the values returned from an api fetch. it will store the values from a staged |
308
|
0
|
|
|
|
|
0
|
# 'batch_values' call for later update from the api. it will store the values |
309
|
34
|
50
|
|
|
|
101
|
# from an update if includeValuesInResponse was included. |
310
|
34
|
|
|
|
|
67
|
# cache is in the format: |
311
|
|
|
|
|
|
|
# majorDimension: ROWS |
312
|
34
|
|
|
|
|
118
|
# range: Sheet1!A1 # only on an api return value. |
313
|
|
|
|
|
|
|
# values: |
314
|
|
|
|
|
|
|
# - - Fred |
315
|
|
|
|
|
|
|
# if a range is included it's a flag that the dim and values are present from |
316
|
|
|
|
|
|
|
# the returned api call. |
317
|
|
|
|
|
|
|
my $self = shift; |
318
|
|
|
|
|
|
|
|
319
|
|
|
|
|
|
|
my %p = @_; |
320
|
|
|
|
|
|
|
# if a range is included, assume this cache is coming from the api as a reply. |
321
|
|
|
|
|
|
|
# this is to store the values for this range when includeValuesInResponse is |
322
|
|
|
|
|
|
|
# added to the url or content on the original values call. you can replace the |
323
|
|
|
|
|
|
|
# original values using the valueRenderOption to replace, say, formulas with their |
324
|
|
|
|
|
|
|
# calculated value. |
325
|
|
|
|
|
|
|
if ($p{range}) { |
326
|
|
|
|
|
|
|
state $check = compile_named( |
327
|
200
|
|
|
200
|
|
362
|
majorDimension => DimColRow, |
328
|
|
|
|
|
|
|
range => StrMatch[qr/.+!/], |
329
|
200
|
|
|
|
|
391
|
values => ArrayRef, { optional => 1 } # will not exist if values aren't set in the ss. |
330
|
|
|
|
|
|
|
); |
331
|
|
|
|
|
|
|
my $p = $check->(@_); |
332
|
|
|
|
|
|
|
|
333
|
|
|
|
|
|
|
# remove all quotes for comparison. |
334
|
|
|
|
|
|
|
my $self_range = $self->range(); |
335
|
200
|
100
|
|
|
|
437
|
$self_range =~ s/'//g; |
336
|
57
|
|
|
|
|
101
|
my $range = $p->{range}; |
337
|
|
|
|
|
|
|
$range =~ s/'//g; |
338
|
|
|
|
|
|
|
my ($worksheet_name) = $range =~ /^(.+)!/; |
339
|
|
|
|
|
|
|
|
340
|
|
|
|
|
|
|
LOGDIE "Setting range data to worksheet name '$worksheet_name' that doesn't belong to this range: " . $self->worksheet_name() |
341
|
57
|
|
|
|
|
2330
|
if $worksheet_name ne $self->worksheet_name(); |
342
|
|
|
|
|
|
|
# TODO: sometimes the api returns a range that is different from the one we sent to the api. |
343
|
|
|
|
|
|
|
# a column A:A can be returned as A1:A1000 by the api, so have to come up with a way |
344
|
57
|
|
|
|
|
2355
|
# of identifying when a returned range is a close enough match to the one we have. |
345
|
57
|
|
|
|
|
245
|
#LOGDIE "Setting range data to '$range' which is not this range: " . $self_range |
346
|
57
|
|
|
|
|
134
|
# if $range ne $self_range; |
347
|
57
|
|
|
|
|
129
|
LOGDIE "Setting major dimention to '$p->{majorDimension}' that doesn't belong to this range: " . $self->dimention() |
348
|
57
|
|
|
|
|
235
|
if $p->{majorDimension} ne $self->dimension(); |
349
|
|
|
|
|
|
|
|
350
|
57
|
50
|
|
|
|
135
|
delete $p->{range}; # we only cache the values and dimensions. |
351
|
|
|
|
|
|
|
$self->{cache_range_values} = $p; |
352
|
|
|
|
|
|
|
DEBUG("Saved cached values for range " . $self->range()); |
353
|
|
|
|
|
|
|
} |
354
|
|
|
|
|
|
|
|
355
|
|
|
|
|
|
|
# if the value range was just stored (above) or was previously stored, return it. |
356
|
|
|
|
|
|
|
return $self->{cache_range_values} if $self->{cache_range_values}; |
357
|
|
|
|
|
|
|
DEBUG("No local cache values exist for range " . $self->range() . ", checking for shared values"); |
358
|
57
|
50
|
|
|
|
175
|
|
359
|
|
|
|
|
|
|
# used by iterators to use a 'parent' range to query the values for this 'child' |
360
|
57
|
|
|
|
|
156
|
# range. this is to reduce network calls when iterating through a range. |
361
|
57
|
|
|
|
|
137
|
my $shared = $self->{shared}; |
362
|
57
|
|
|
|
|
112
|
if ($shared && $shared->has_values()) { |
363
|
|
|
|
|
|
|
my $dim = $shared->{cache_range_values}->{majorDimension}; |
364
|
|
|
|
|
|
|
my $values = $shared->{cache_range_values}->{values}; |
365
|
|
|
|
|
|
|
|
366
|
200
|
100
|
|
|
|
999
|
my ($top, $left, $bottom, $right) = $self->offsets($shared); |
367
|
52
|
|
|
|
|
157
|
my $data; |
368
|
|
|
|
|
|
|
if ($dim =~ /^col/i) { |
369
|
|
|
|
|
|
|
my @cols = @$values[$left..$right]; |
370
|
|
|
|
|
|
|
$_ = [ @$_[$top..$bottom] ] foreach (@cols); |
371
|
52
|
|
|
|
|
307
|
$data = \@cols; |
372
|
52
|
50
|
33
|
|
|
156
|
} else { |
373
|
0
|
|
|
|
|
0
|
my @rows = @$values[$top..$bottom]; |
374
|
0
|
|
|
|
|
0
|
$_ = [ @$_[$left..$right] ] foreach (@rows); |
375
|
|
|
|
|
|
|
$data = \@rows; |
376
|
0
|
|
|
|
|
0
|
} |
377
|
0
|
|
|
|
|
0
|
|
378
|
0
|
0
|
|
|
|
0
|
# return a subsection of the cached value for the iterator. |
379
|
0
|
|
|
|
|
0
|
DEBUG("Returning shared values for range " . $self->range()); |
380
|
0
|
|
|
|
|
0
|
return { |
381
|
0
|
|
|
|
|
0
|
majorDimension => $dim, |
382
|
|
|
|
|
|
|
values => $data, |
383
|
0
|
|
|
|
|
0
|
}; |
384
|
0
|
|
|
|
|
0
|
} |
385
|
0
|
|
|
|
|
0
|
DEBUG("No shared values exist for range " . $self->range()); |
386
|
|
|
|
|
|
|
|
387
|
|
|
|
|
|
|
# no values are found for this range, go get them from the api immediately. |
388
|
|
|
|
|
|
|
$p{uri} = sprintf("/values/%s", $self->range()); |
389
|
0
|
|
|
|
|
0
|
$p{params}->{majorDimension} = $self->dimension(); |
390
|
|
|
|
|
|
|
$self->{cache_range_values} = $self->api(%p); |
391
|
0
|
|
|
|
|
0
|
DEBUG("Cached values loaded from api for range " . $self->range()); |
392
|
|
|
|
|
|
|
|
393
|
|
|
|
|
|
|
$self->{cache_range_values}->{values} //= []; # in case there's nothing set in the ss. |
394
|
|
|
|
|
|
|
|
395
|
52
|
|
|
|
|
113
|
return $self->{cache_range_values}; |
396
|
|
|
|
|
|
|
} |
397
|
|
|
|
|
|
|
|
398
|
52
|
|
|
|
|
301
|
my $self = shift; |
399
|
52
|
|
|
|
|
175
|
return $self->{cache_range_values} || ($self->{shared} && $self->{shared}->has_values()); |
400
|
52
|
|
|
|
|
268
|
} |
401
|
52
|
|
|
|
|
203
|
|
402
|
|
|
|
|
|
|
# for a given range, calculate the offsets from this range. |
403
|
52
|
|
100
|
|
|
442
|
my $self = shift; |
404
|
|
|
|
|
|
|
|
405
|
52
|
|
|
|
|
266
|
state $check = compile(HasRange); |
406
|
|
|
|
|
|
|
my ($other_range) = $check->(@_); |
407
|
|
|
|
|
|
|
|
408
|
|
|
|
|
|
|
my $range = $self->range_to_hash($RANGE_EXPANDED); |
409
|
134
|
|
|
134
|
1
|
212
|
$other_range = $other_range->range_to_hash($RANGE_EXPANDED); |
410
|
134
|
|
66
|
|
|
656
|
|
411
|
|
|
|
|
|
|
my $top = $range->[0]->{row} - $other_range->[0]->{row}; |
412
|
|
|
|
|
|
|
my $left = $range->[0]->{col} - $other_range->[0]->{col}; |
413
|
|
|
|
|
|
|
my $bottom = $range->[1]->{row} - $other_range->[0]->{row}; |
414
|
|
|
|
|
|
|
my $right = $range->[1]->{col} - $other_range->[0]->{col}; |
415
|
0
|
|
|
0
|
1
|
0
|
|
416
|
|
|
|
|
|
|
return ( $top, $left, $bottom, $right ); |
417
|
0
|
|
|
|
|
0
|
} |
418
|
0
|
|
|
|
|
0
|
|
419
|
|
|
|
|
|
|
my $self = shift; |
420
|
0
|
|
|
|
|
0
|
|
421
|
0
|
|
|
|
|
0
|
state $check = compile(HasRange); |
422
|
|
|
|
|
|
|
my ($shared_range) = $check->(@_); |
423
|
0
|
|
|
|
|
0
|
return if !$shared_range->is_other_inside($self); |
424
|
0
|
|
|
|
|
0
|
|
425
|
0
|
|
|
|
|
0
|
DEBUG(flatten_range($self) . " is sharing values with " . flatten_range($shared_range)); |
426
|
0
|
|
|
|
|
0
|
$self->{shared} = $shared_range; |
427
|
|
|
|
|
|
|
return $shared_range; |
428
|
0
|
|
|
|
|
0
|
} |
429
|
|
|
|
|
|
|
|
430
|
|
|
|
|
|
|
my $self = shift; |
431
|
|
|
|
|
|
|
|
432
|
22
|
|
|
22
|
1
|
41
|
state $check = compile(HasRange); |
433
|
|
|
|
|
|
|
my ($inside_range) = $check->(@_); |
434
|
22
|
|
|
|
|
47
|
|
435
|
22
|
|
|
|
|
804
|
my $range = $self->range_to_hash($RANGE_EXPANDED); |
436
|
22
|
50
|
|
|
|
283
|
$inside_range = $inside_range->range_to_hash($RANGE_EXPANDED); |
437
|
|
|
|
|
|
|
|
438
|
22
|
|
|
|
|
70
|
return 1 if |
439
|
22
|
|
|
|
|
159
|
$range->[0]->{col} <= $inside_range->[0]->{col} && |
440
|
22
|
|
|
|
|
54
|
$range->[0]->{row} <= $inside_range->[0]->{row} && |
441
|
|
|
|
|
|
|
$range->[1]->{col} >= $inside_range->[1]->{col} && |
442
|
|
|
|
|
|
|
$range->[1]->{row} >= $inside_range->[1]->{row} |
443
|
|
|
|
|
|
|
; |
444
|
10
|
|
|
10
|
1
|
58
|
return; |
445
|
|
|
|
|
|
|
} |
446
|
10
|
|
|
|
|
24
|
|
447
|
10
|
|
|
|
|
686
|
# stage batch values get/set for later use by submit_values |
448
|
|
|
|
|
|
|
my $self = shift; |
449
|
10
|
|
|
|
|
154
|
|
450
|
10
|
|
|
|
|
44
|
state $check = compile_named( |
451
|
|
|
|
|
|
|
values => ArrayRef, { optional => 1 }, |
452
|
|
|
|
|
|
|
); |
453
|
|
|
|
|
|
|
my $p = $check->(@_); |
454
|
|
|
|
|
|
|
|
455
|
|
|
|
|
|
|
if (defined $p->{values}) { |
456
|
|
|
|
|
|
|
# since we're sending these values, fake a response from the |
457
|
10
|
100
|
33
|
|
|
258
|
# api to store them in our cache. |
|
|
|
66
|
|
|
|
|
|
|
|
100
|
|
|
|
|
458
|
2
|
|
|
|
|
16
|
# if includeValuesInResponse is sent in the params, then the |
459
|
|
|
|
|
|
|
# response will replace the cache again. |
460
|
|
|
|
|
|
|
$self->_cache_range_values( |
461
|
|
|
|
|
|
|
range => $self->range(), |
462
|
|
|
|
|
|
|
majorDimension => $self->dimension(), |
463
|
62
|
|
|
62
|
1
|
100
|
values => $p->{values}, |
464
|
|
|
|
|
|
|
); |
465
|
62
|
|
|
|
|
103
|
} |
466
|
|
|
|
|
|
|
|
467
|
|
|
|
|
|
|
return if !$self->{cache_range_values}; |
468
|
62
|
|
|
|
|
840
|
|
469
|
|
|
|
|
|
|
return { |
470
|
62
|
100
|
|
|
|
905
|
range => $self->range(), |
471
|
|
|
|
|
|
|
%{ $self->{cache_range_values} }, |
472
|
|
|
|
|
|
|
}; |
473
|
|
|
|
|
|
|
} |
474
|
|
|
|
|
|
|
|
475
|
|
|
|
|
|
|
# tell the api to submit the batch values. the api will call back |
476
|
|
|
|
|
|
|
# values_response_from_api with the results of the update. |
477
|
|
|
|
|
|
|
my $self = shift; |
478
|
|
|
|
|
|
|
$self->spreadsheet()->submit_values(ranges => [ $self ], @_); |
479
|
31
|
|
|
|
|
82
|
return $self->values(); |
480
|
|
|
|
|
|
|
} |
481
|
|
|
|
|
|
|
|
482
|
62
|
50
|
|
|
|
149
|
my $self = shift; |
483
|
|
|
|
|
|
|
return $self->spreadsheet()->submit_requests(ranges => [ $self ], @_); |
484
|
|
|
|
|
|
|
} |
485
|
|
|
|
|
|
|
|
486
|
62
|
|
|
|
|
129
|
my $self = shift; |
|
62
|
|
|
|
|
372
|
|
487
|
|
|
|
|
|
|
|
488
|
|
|
|
|
|
|
state $check = compile_named( |
489
|
|
|
|
|
|
|
values => ArrayRef->plus_coercions(Str, sub { [ $_ ] } ), |
490
|
|
|
|
|
|
|
params => HashRef, { default => {} }, |
491
|
|
|
|
|
|
|
content => HashRef, { default => {} }, |
492
|
|
|
|
|
|
|
_extra_ => slurpy Any, |
493
|
0
|
|
|
0
|
1
|
0
|
); |
494
|
0
|
|
|
|
|
0
|
my $p = named_extra($check->(@_)); |
495
|
0
|
|
|
|
|
0
|
|
496
|
|
|
|
|
|
|
my $range = $self->range(); |
497
|
|
|
|
|
|
|
$p->{content}->{range} = $range; |
498
|
|
|
|
|
|
|
$p->{content}->{values} = delete $p->{values}; |
499
|
6
|
|
|
6
|
1
|
9049
|
$p->{content}->{majorDimension} = $self->dimension(); |
500
|
6
|
|
|
|
|
20
|
$p->{params}->{valueInputOption} //= 'USER_ENTERED'; |
501
|
|
|
|
|
|
|
$p->{uri} = "/values/$range:append"; |
502
|
|
|
|
|
|
|
$p->{method} = 'post'; |
503
|
|
|
|
|
|
|
|
504
|
0
|
|
|
0
|
0
|
0
|
return $self->api(%$p); |
505
|
|
|
|
|
|
|
} |
506
|
|
|
|
|
|
|
|
507
|
0
|
|
|
0
|
|
0
|
my $self = shift; |
|
0
|
|
|
|
|
0
|
|
508
|
|
|
|
|
|
|
TRACE("Range external caller: " . $self->_caller_external()); |
509
|
|
|
|
|
|
|
TRACE("Range internal caller: " . $self->_caller_internal()); |
510
|
|
|
|
|
|
|
my $name = $self->worksheet_name(); |
511
|
|
|
|
|
|
|
return "'$name'!$self->{range}"; |
512
|
0
|
|
|
|
|
0
|
} |
513
|
|
|
|
|
|
|
|
514
|
0
|
|
|
|
|
0
|
# some staic calls... |
515
|
0
|
|
|
|
|
0
|
|
516
|
0
|
|
|
|
|
0
|
my ($package, $subroutine, $line, $i) = ('', '', 0); |
517
|
0
|
|
|
|
|
0
|
do { |
518
|
0
|
|
0
|
|
|
0
|
($package, undef, $line, $subroutine) = caller(++$i); |
519
|
0
|
|
|
|
|
0
|
} while($subroutine =~ m|range$|); |
520
|
0
|
|
|
|
|
0
|
# not usually going to happen, but during testing we call |
521
|
|
|
|
|
|
|
# range directly, so have to backtrack. |
522
|
0
|
|
|
|
|
0
|
($package, undef, $line, $subroutine) = caller(--$i) |
523
|
|
|
|
|
|
|
if !$package; |
524
|
|
|
|
|
|
|
return "$package:$line => $subroutine"; |
525
|
|
|
|
|
|
|
} |
526
|
842
|
|
|
842
|
1
|
1604
|
|
527
|
842
|
|
|
|
|
1940
|
my ($package, $subroutine, $line, $i) = ('', '', 0); |
528
|
842
|
|
|
|
|
6631
|
do { |
529
|
842
|
|
|
|
|
5161
|
($package, undef, $line, $subroutine) = caller(++$i); |
530
|
842
|
|
|
|
|
3548
|
} while($package && $package =~ m[^(Google::RestApi)]); |
531
|
|
|
|
|
|
|
return "$package:$line => $subroutine"; |
532
|
|
|
|
|
|
|
} |
533
|
|
|
|
|
|
|
|
534
|
|
|
|
|
|
|
# taken from https://metacpan.org/source/DOUGW/Spreadsheet-ParseExcel-0.65/lib/Spreadsheet/ParseExcel/Utility.pm |
535
|
|
|
|
|
|
|
state $check = compile(StrMatch[qr/^[A-Z]+$/]); |
536
|
842
|
|
|
842
|
|
1586
|
my ($a) = $check->(@_); |
537
|
842
|
|
|
|
|
1098
|
|
538
|
1724
|
|
|
|
|
15839
|
my $result = 0; |
539
|
|
|
|
|
|
|
my $incr = 0; |
540
|
|
|
|
|
|
|
for (my $i = length($a); $i > 0 ; $i--) { |
541
|
|
|
|
|
|
|
my $char = substr($a, $i - 1); |
542
|
842
|
50
|
|
|
|
12148
|
my $curr += ord(lc($char)) - ord('a') + 1; |
543
|
|
|
|
|
|
|
$curr *= $incr if ($incr); |
544
|
842
|
|
|
|
|
3135
|
$result += $curr; |
545
|
|
|
|
|
|
|
$incr += 26; |
546
|
|
|
|
|
|
|
} |
547
|
|
|
|
|
|
|
|
548
|
842
|
|
|
842
|
|
1867
|
return $result; |
549
|
842
|
|
66
|
|
|
1101
|
} |
550
|
4143
|
|
|
|
|
60587
|
|
551
|
|
|
|
|
|
|
my $cell = shift; |
552
|
842
|
|
|
|
|
17048
|
|
553
|
|
|
|
|
|
|
my ($col, $row) = $cell =~ /^([A-Z]*)(\d*)$/; |
554
|
|
|
|
|
|
|
$col = _col_a2i($col) if $col; |
555
|
|
|
|
|
|
|
|
556
|
|
|
|
|
|
|
$col ||= 0; |
557
|
337
|
|
|
337
|
|
493
|
$row ||= 0; |
558
|
337
|
|
|
|
|
2473
|
|
559
|
|
|
|
|
|
|
return [$col, $row]; |
560
|
337
|
|
|
|
|
4107
|
} |
561
|
337
|
|
|
|
|
538
|
|
562
|
337
|
|
|
|
|
905
|
# back to object calls... |
563
|
337
|
|
|
|
|
728
|
|
564
|
337
|
|
|
|
|
704
|
# returns [[col, row], [col, row]] for a full range. |
565
|
337
|
50
|
|
|
|
573
|
# returns [col, row] for a col or col(a2:a) or row(a2:2). |
566
|
337
|
|
|
|
|
449
|
# passing $RANGE_EXPANDED will always return the former. |
567
|
337
|
|
|
|
|
629
|
my $self = shift; |
568
|
|
|
|
|
|
|
|
569
|
|
|
|
|
|
|
state $check = compile(Bool, { optional => 1 }); |
570
|
337
|
|
|
|
|
669
|
my ($expand_range) = $check->(@_); |
571
|
|
|
|
|
|
|
|
572
|
|
|
|
|
|
|
my $range = $self->range(); |
573
|
|
|
|
|
|
|
($range) = $range =~ /!(.+)/; |
574
|
365
|
|
|
365
|
0
|
591
|
$range or LOGDIE "Unable to convert range to array: ", $self->range(); |
575
|
|
|
|
|
|
|
|
576
|
365
|
|
|
|
|
1241
|
my ($start_cell, $end_cell) = split(':', $range); |
577
|
365
|
100
|
|
|
|
977
|
$end_cell = undef if defined $end_cell && $start_cell eq $end_cell; |
578
|
|
|
|
|
|
|
$start_cell = cell_to_array($start_cell); |
579
|
365
|
|
100
|
|
|
778
|
$end_cell = cell_to_array($end_cell) if $end_cell; |
580
|
365
|
|
100
|
|
|
828
|
|
581
|
|
|
|
|
|
|
$end_cell = $start_cell if !$end_cell && $expand_range; |
582
|
365
|
|
|
|
|
841
|
|
583
|
|
|
|
|
|
|
return $end_cell ? [$start_cell, $end_cell] : $start_cell; |
584
|
|
|
|
|
|
|
} |
585
|
|
|
|
|
|
|
|
586
|
|
|
|
|
|
|
# returns [{col =>, row =>}, {col =>, row => }] for an expanded range. |
587
|
|
|
|
|
|
|
# returns {col =>, row =>} for a cell or col(a2:a) or row(a2:2). |
588
|
|
|
|
|
|
|
# passing $RANGE_EXPANDED will always return the former. |
589
|
|
|
|
|
|
|
my $self = shift; |
590
|
|
|
|
|
|
|
|
591
|
306
|
|
|
306
|
1
|
534
|
my $range = $self->range_to_array(@_); |
592
|
|
|
|
|
|
|
$range = [ $range ] if !ref($range->[1]); |
593
|
306
|
|
|
|
|
490
|
my @ranges = map { { col => $_->[0], row => $_->[1] } } @$range; |
594
|
306
|
|
|
|
|
1837
|
|
595
|
|
|
|
|
|
|
return $ranges[1] ? \@ranges : $ranges[0]; |
596
|
306
|
|
|
|
|
3584
|
} |
597
|
306
|
|
|
|
|
1296
|
|
598
|
306
|
50
|
|
|
|
686
|
my $self = shift; |
599
|
|
|
|
|
|
|
|
600
|
306
|
|
|
|
|
885
|
my $array = $self->range_to_array(); |
601
|
306
|
100
|
100
|
|
|
1071
|
$array = [$array, $array] if !ref($array->[1]); |
602
|
306
|
|
|
|
|
601
|
|
603
|
306
|
100
|
|
|
|
621
|
my %range = ( |
604
|
|
|
|
|
|
|
sheetId => $self->worksheet()->worksheet_id(), |
605
|
306
|
100
|
100
|
|
|
1106
|
startColumnIndex => max($array->[0]->[0] - 1, 0), |
606
|
|
|
|
|
|
|
startRowIndex => max($array->[0]->[1] - 1, 0), |
607
|
306
|
100
|
|
|
|
932
|
endColumnIndex => $array->[1]->[0], |
608
|
|
|
|
|
|
|
endRowIndex => $array->[1]->[1], |
609
|
|
|
|
|
|
|
); |
610
|
|
|
|
|
|
|
delete $range{endColumnIndex} |
611
|
|
|
|
|
|
|
if $range{endColumnIndex} < $range{startColumnIndex}; |
612
|
|
|
|
|
|
|
delete $range{endRowIndex} |
613
|
|
|
|
|
|
|
if $range{endRowIndex} < $range{startRowIndex}; |
614
|
190
|
|
|
190
|
1
|
297
|
|
615
|
|
|
|
|
|
|
return \%range; |
616
|
190
|
|
|
|
|
496
|
} |
617
|
190
|
100
|
|
|
|
608
|
|
618
|
190
|
|
|
|
|
412
|
my $self = shift; |
|
312
|
|
|
|
|
1177
|
|
619
|
|
|
|
|
|
|
|
620
|
190
|
100
|
|
|
|
742
|
state $check = compile(DimColRow); |
621
|
|
|
|
|
|
|
my ($dims) = $check->(@_); |
622
|
|
|
|
|
|
|
$dims = dims_any($dims); |
623
|
|
|
|
|
|
|
|
624
|
98
|
|
|
98
|
1
|
210
|
my $array = $self->range_to_array(); |
625
|
|
|
|
|
|
|
$array = [$array, $array] if !ref($array->[1]); |
626
|
98
|
|
|
|
|
266
|
my $start = $dims =~ /^col/i ? $array->[0]->[0] : $array->[0]->[1]; |
627
|
98
|
100
|
|
|
|
346
|
my $end = $dims =~ /^col/i ? $array->[1]->[0] : $array->[1]->[1]; |
628
|
|
|
|
|
|
|
|
629
|
98
|
|
|
|
|
211
|
my %range = ( |
630
|
|
|
|
|
|
|
sheetId => $self->worksheet()->worksheet_id(), |
631
|
|
|
|
|
|
|
startIndex => max($start - 1, 0), |
632
|
|
|
|
|
|
|
endIndex => $end, |
633
|
|
|
|
|
|
|
dimension => $dims, |
634
|
|
|
|
|
|
|
); |
635
|
|
|
|
|
|
|
|
636
|
|
|
|
|
|
|
return \%range; |
637
|
98
|
50
|
|
|
|
273
|
} |
638
|
|
|
|
|
|
|
|
639
|
98
|
50
|
|
|
|
192
|
my $self = shift; |
640
|
|
|
|
|
|
|
|
641
|
98
|
|
|
|
|
711
|
state $check = compile(Int, DimColRow); |
642
|
|
|
|
|
|
|
my ($offset, $dim) = $check->(@_); |
643
|
|
|
|
|
|
|
|
644
|
|
|
|
|
|
|
# it's an a1:b2 range. col/row/cell handle this in the subclass. |
645
|
0
|
|
|
0
|
1
|
0
|
my $range = $self->range_to_hash($RANGE_EXPANDED); |
646
|
|
|
|
|
|
|
my $other_dim = $dim =~ /col/i ? 'row' : 'col'; |
647
|
0
|
|
|
|
|
0
|
# TODO: this is really brain dead, figure this out with a bit of arithmetic... |
648
|
0
|
|
|
|
|
0
|
my @ranges = map { |
649
|
0
|
|
|
|
|
0
|
my $outside = $_; |
650
|
|
|
|
|
|
|
map { |
651
|
0
|
|
|
|
|
0
|
my $inside = $_; |
652
|
0
|
0
|
|
|
|
0
|
my $r = { $dim => $inside, $other_dim => $outside }; $r; # for some reason have to stage the ref to a scalar. |
653
|
0
|
0
|
|
|
|
0
|
} ($range->[0]->{$other_dim} .. $range->[1]->{$other_dim}); |
654
|
0
|
0
|
|
|
|
0
|
} ($range->[0]->{$dim} .. $range->[1]->{$dim}); |
655
|
|
|
|
|
|
|
|
656
|
0
|
|
|
|
|
0
|
my $offset_range = $ranges[$offset] or return; |
657
|
|
|
|
|
|
|
my $new_cell = $self->worksheet()->range_cell($offset_range); |
658
|
|
|
|
|
|
|
$new_cell->share_values($self); |
659
|
|
|
|
|
|
|
|
660
|
|
|
|
|
|
|
return $new_cell; |
661
|
|
|
|
|
|
|
} |
662
|
|
|
|
|
|
|
|
663
|
0
|
|
|
|
|
0
|
# doesn't seem to be used anywhere internally. |
664
|
|
|
|
|
|
|
my $self = shift; |
665
|
|
|
|
|
|
|
|
666
|
|
|
|
|
|
|
state $check = compile_named( |
667
|
6
|
|
|
6
|
0
|
386
|
col => Int, { default => 0 }, |
668
|
|
|
|
|
|
|
row => Int, { default => 0 }, |
669
|
6
|
|
|
|
|
11
|
top => Int, { default => 0 }, |
670
|
6
|
|
|
|
|
1146
|
left => Int, { default => 0 }, |
671
|
|
|
|
|
|
|
bottom => Int, { default => 0 }, |
672
|
|
|
|
|
|
|
right => Int, { default => 0 }, |
673
|
6
|
|
|
|
|
143
|
); |
674
|
6
|
100
|
|
|
|
22
|
my $p = $check->(@_); |
675
|
|
|
|
|
|
|
|
676
|
|
|
|
|
|
|
my $range = $self->range_to_hash(); |
677
|
12
|
|
|
|
|
18
|
if (ref($range) eq 'ARRAY') { |
678
|
|
|
|
|
|
|
foreach my $dim (@$range) { |
679
|
24
|
|
|
|
|
26
|
$dim->{$_} += $p->{$_} foreach (qw(col row)); |
680
|
24
|
|
|
|
|
42
|
} |
|
24
|
|
|
|
|
45
|
|
681
|
12
|
|
|
|
|
32
|
$range->[0]->{row} += $p->{top}; |
682
|
6
|
|
|
|
|
28
|
$range->[0]->{col} += $p->{left}; |
683
|
|
|
|
|
|
|
$range->[1]->{row} += $p->{bottom}; |
684
|
6
|
100
|
|
|
|
23
|
$range->[1]->{col} += $p->{right}; |
685
|
4
|
|
|
|
|
11
|
} else { |
686
|
4
|
|
|
|
|
15
|
$range->{$_} += $p->{$_} foreach (qw(col row)); |
687
|
|
|
|
|
|
|
$range->{row} += $p->{top}; |
688
|
4
|
|
|
|
|
35
|
$range->{col} += $p->{left}; |
689
|
|
|
|
|
|
|
$range->{row} += $p->{bottom}; |
690
|
|
|
|
|
|
|
$range->{col} += $p->{right}; |
691
|
|
|
|
|
|
|
} |
692
|
|
|
|
|
|
|
|
693
|
0
|
|
|
0
|
0
|
0
|
return (ref($self))->new(worksheet => $self->worksheet(), range => $range); |
694
|
|
|
|
|
|
|
} |
695
|
0
|
|
|
|
|
0
|
|
696
|
|
|
|
|
|
|
|
697
|
|
|
|
|
|
|
1; |
698
|
|
|
|
|
|
|
|
699
|
|
|
|
|
|
|
|
700
|
|
|
|
|
|
|
=head1 NAME |
701
|
|
|
|
|
|
|
|
702
|
|
|
|
|
|
|
Google::RestApi::SheetsApi4::Range - Represents a range in a Worksheet. |
703
|
0
|
|
|
|
|
0
|
|
704
|
|
|
|
|
|
|
=head1 DESCRIPTION |
705
|
0
|
|
|
|
|
0
|
|
706
|
0
|
0
|
|
|
|
0
|
A Range object that represents a range in a remote spreadsheet. These are |
707
|
0
|
|
|
|
|
0
|
normally short-lived objects used to set up and execute a remote action. |
708
|
0
|
|
|
|
|
0
|
Keep in mind that the remote spreadsheet can be concurrently updated by |
709
|
|
|
|
|
|
|
many people, so a compromise must always be reached between holding a copy |
710
|
0
|
|
|
|
|
0
|
of the local cell values versus the number of network calls to the Google |
711
|
0
|
|
|
|
|
0
|
API to keep the values current. |
712
|
0
|
|
|
|
|
0
|
|
713
|
0
|
|
|
|
|
0
|
A range can be specified in whatever way is most convenient. |
714
|
|
|
|
|
|
|
|
715
|
0
|
|
|
|
|
0
|
For ranges: |
716
|
0
|
|
|
|
|
0
|
|
717
|
0
|
|
|
|
|
0
|
* A1 notation: A1:B2 |
718
|
0
|
|
|
|
|
0
|
* Hash: [ { col => (A|1), row => 1 }, { col => (B|2), row => 2 } ] |
719
|
0
|
|
|
|
|
0
|
* Array: [ [ (A|1), 1 ], [ (B|2), 2 ] ] |
720
|
|
|
|
|
|
|
|
721
|
|
|
|
|
|
|
For columns: |
722
|
0
|
|
|
|
|
0
|
|
723
|
|
|
|
|
|
|
* A1 notation: A or A:A |
724
|
|
|
|
|
|
|
* Hash: { col => (A|1) } |
725
|
0
|
|
|
0
|
0
|
0
|
* Array: [ A|1 ] |
726
|
2
|
|
|
2
|
1
|
37
|
|
727
|
55
|
|
|
55
|
1
|
118
|
For rows: |
728
|
169
|
|
|
169
|
1
|
558
|
|
729
|
1080
|
|
|
1080
|
1
|
3325
|
* A1 notation: 1 or 1:1 |
730
|
899
|
|
|
899
|
1
|
1984
|
* Hash: { row => 1 } |
731
|
0
|
|
|
0
|
1
|
0
|
* Array: [ <false>, 1 ] |
732
|
6
|
|
|
6
|
1
|
16
|
|
733
|
0
|
|
|
0
|
1
|
|
For cells: |
734
|
0
|
|
|
0
|
0
|
|
|
735
|
0
|
|
|
0
|
1
|
|
* A1 notation: A1 |
736
|
|
|
|
|
|
|
* Hash: { col => (A|1), row => 1 } |
737
|
|
|
|
|
|
|
* Array: [ (A|1), 1 ] |
738
|
|
|
|
|
|
|
|
739
|
|
|
|
|
|
|
See the description and synopsis at L<Google::RestApi::SheetsApi4>. |
740
|
|
|
|
|
|
|
|
741
|
|
|
|
|
|
|
=head1 NAVIGATION |
742
|
|
|
|
|
|
|
|
743
|
|
|
|
|
|
|
=over |
744
|
|
|
|
|
|
|
|
745
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4> |
746
|
|
|
|
|
|
|
|
747
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Spreadsheet> |
748
|
|
|
|
|
|
|
|
749
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Worksheet> |
750
|
|
|
|
|
|
|
|
751
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Range> |
752
|
|
|
|
|
|
|
|
753
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Range::All> |
754
|
|
|
|
|
|
|
|
755
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Range::Col> |
756
|
|
|
|
|
|
|
|
757
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Range::Row> |
758
|
|
|
|
|
|
|
|
759
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Range::Cell> |
760
|
|
|
|
|
|
|
|
761
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::RangeGroup> |
762
|
|
|
|
|
|
|
|
763
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::RangeGroup::Iterator> |
764
|
|
|
|
|
|
|
|
765
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::RangeGroup::Tie> |
766
|
|
|
|
|
|
|
|
767
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::RangeGroup::Tie::Iterator> |
768
|
|
|
|
|
|
|
|
769
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Request::Spreadsheet> |
770
|
|
|
|
|
|
|
|
771
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Request::Spreadsheet::Worksheet> |
772
|
|
|
|
|
|
|
|
773
|
|
|
|
|
|
|
=item * L<Google::RestApi::SheetsApi4::Request::Spreadsheet::Worksheet::Range> |
774
|
|
|
|
|
|
|
|
775
|
|
|
|
|
|
|
=back |
776
|
|
|
|
|
|
|
|
777
|
|
|
|
|
|
|
=head1 SUBROUTINES |
778
|
|
|
|
|
|
|
|
779
|
|
|
|
|
|
|
=over |
780
|
|
|
|
|
|
|
|
781
|
|
|
|
|
|
|
=item new(%args); |
782
|
|
|
|
|
|
|
|
783
|
|
|
|
|
|
|
Creates a new range object for the given worksheet. |
784
|
|
|
|
|
|
|
|
785
|
|
|
|
|
|
|
%args consists of: |
786
|
|
|
|
|
|
|
|
787
|
|
|
|
|
|
|
=over |
788
|
|
|
|
|
|
|
|
789
|
|
|
|
|
|
|
=item * C<worksheet> <Worksheet>: The parent Worksheet object for this range. |
790
|
|
|
|
|
|
|
|
791
|
|
|
|
|
|
|
=item * C<range> <range>: The range for this object, either A1 notation, hash {col => x, row => x} or array [col, row]. |
792
|
|
|
|
|
|
|
|
793
|
|
|
|
|
|
|
=item * C<dim> <col|row>: The major dimension for this range, either 'col' or 'row', defaults to 'row'. |
794
|
|
|
|
|
|
|
|
795
|
|
|
|
|
|
|
=back |
796
|
|
|
|
|
|
|
|
797
|
|
|
|
|
|
|
You would not normally call this directly, you'd use Worksheet::range* methods to create the range object for you. It is recommended and safer to use |
798
|
|
|
|
|
|
|
the Worksheet's methods to create ranges. |
799
|
|
|
|
|
|
|
|
800
|
|
|
|
|
|
|
=item api(%args); |
801
|
|
|
|
|
|
|
|
802
|
|
|
|
|
|
|
Calls the parent Worksheet's 'api' routine with the range added into the URI or content appropriately. This then get's passed to the |
803
|
|
|
|
|
|
|
Spreadsheet's C<api> routine where the spreadsheet ID is tacked on to the URI. This then gets passed to the SheetsApi4's C<api> |
804
|
|
|
|
|
|
|
routine where the Sheets endpoint is tacked on to the URI. This then gets passed to RestApi's api routine for actual execution. |
805
|
|
|
|
|
|
|
|
806
|
|
|
|
|
|
|
You would not normally call this directly unless you were making a Google API call not currently supported by this API framework. |
807
|
|
|
|
|
|
|
|
808
|
|
|
|
|
|
|
=item clear(); |
809
|
|
|
|
|
|
|
|
810
|
|
|
|
|
|
|
Clears the values using Google API's 'A1:clear' call. |
811
|
|
|
|
|
|
|
|
812
|
|
|
|
|
|
|
=item refresh_values(); |
813
|
|
|
|
|
|
|
|
814
|
|
|
|
|
|
|
Immediately refreshes and returns the values from the spreadsheet. |
815
|
|
|
|
|
|
|
|
816
|
|
|
|
|
|
|
=item values(values => <arrayref>, %args); |
817
|
|
|
|
|
|
|
|
818
|
|
|
|
|
|
|
Immediately gets or sets the values using Google API's 'get' or 'update'. |
819
|
|
|
|
|
|
|
|
820
|
|
|
|
|
|
|
values: The array ref of cells to update. |
821
|
|
|
|
|
|
|
|
822
|
|
|
|
|
|
|
'args' are passed to the SheetsApi4's 'api' routine so you may add |
823
|
|
|
|
|
|
|
extra arguments to the 'params' (such as C<valueInputOption>) or 'content' as necessary. |
824
|
|
|
|
|
|
|
|
825
|
|
|
|
|
|
|
=item batch_values(values => <arrayref>); |
826
|
|
|
|
|
|
|
|
827
|
|
|
|
|
|
|
Gets or sets the queued batch values that will be sent to Google API |
828
|
|
|
|
|
|
|
at a later time. Call 'submit_values' to send them later. |
829
|
|
|
|
|
|
|
|
830
|
|
|
|
|
|
|
values: The array ref of cells to update. |
831
|
|
|
|
|
|
|
|
832
|
|
|
|
|
|
|
=item submit_values(%args); |
833
|
|
|
|
|
|
|
|
834
|
|
|
|
|
|
|
Sends the previously queued batch values to Google API, if any. |
835
|
|
|
|
|
|
|
|
836
|
|
|
|
|
|
|
'args' are passed to the SheetsApi4's 'api' routine so you may add |
837
|
|
|
|
|
|
|
extra arguments to the 'params' or 'content' as necessary. |
838
|
|
|
|
|
|
|
|
839
|
|
|
|
|
|
|
=item submit_requests(%args); |
840
|
|
|
|
|
|
|
|
841
|
|
|
|
|
|
|
Sends the previously queued requests (formatting, sheet properties etc) |
842
|
|
|
|
|
|
|
to Google API, if any. |
843
|
|
|
|
|
|
|
|
844
|
|
|
|
|
|
|
'args' are passed to the SheetsApi4's 'api' routine so you may add |
845
|
|
|
|
|
|
|
extra arguments to the 'params' or 'content' as necessary. |
846
|
|
|
|
|
|
|
|
847
|
|
|
|
|
|
|
=item normalize_named(); |
848
|
|
|
|
|
|
|
|
849
|
|
|
|
|
|
|
If this is a 'named range', get the named range property from the |
850
|
|
|
|
|
|
|
spreadsheet and set this object's real range that the named range |
851
|
|
|
|
|
|
|
represents. |
852
|
|
|
|
|
|
|
|
853
|
|
|
|
|
|
|
=item named(); |
854
|
|
|
|
|
|
|
|
855
|
|
|
|
|
|
|
Returns the 'named range' for this range, if any. |
856
|
|
|
|
|
|
|
|
857
|
|
|
|
|
|
|
=item range(); |
858
|
|
|
|
|
|
|
|
859
|
|
|
|
|
|
|
Returns the A1 notation for this range. |
860
|
|
|
|
|
|
|
|
861
|
|
|
|
|
|
|
=item range_to_hash(); |
862
|
|
|
|
|
|
|
|
863
|
|
|
|
|
|
|
Returns the hash representation for this range (e.g. {col => 1, row => 1}). Passing $RANGE_EXPANDED |
864
|
|
|
|
|
|
|
will always return a double cell notations: [{col => 1, row => 1}, {col => 1, row => 1}]. |
865
|
|
|
|
|
|
|
|
866
|
|
|
|
|
|
|
=item range_to_array(); |
867
|
|
|
|
|
|
|
|
868
|
|
|
|
|
|
|
Returns the array representation for this range (e.g. [1, 1]). Passing $RANGE_EXPANDED |
869
|
|
|
|
|
|
|
will always return a double cell notations: [[ 1, 1 ], [ 1, 1]]. |
870
|
|
|
|
|
|
|
|
871
|
|
|
|
|
|
|
=item range_to_index(); |
872
|
|
|
|
|
|
|
|
873
|
|
|
|
|
|
|
Returns the index hash representation for this range, used for formatting |
874
|
|
|
|
|
|
|
requests etc. You would not normally need to call this yourself. |
875
|
|
|
|
|
|
|
|
876
|
|
|
|
|
|
|
=item range_to_dimension(); |
877
|
|
|
|
|
|
|
|
878
|
|
|
|
|
|
|
Returns the dimension hash representation for this range, used for |
879
|
|
|
|
|
|
|
insert requests etc. You would not normally need to call this yourself. |
880
|
|
|
|
|
|
|
|
881
|
|
|
|
|
|
|
=item offset(col => <int>, row => <int>, top => <int>, left => <int>, bottom => <int>, right => <int>); |
882
|
|
|
|
|
|
|
|
883
|
|
|
|
|
|
|
Returns a new range object offset from this range. |
884
|
|
|
|
|
|
|
|
885
|
|
|
|
|
|
|
col: Optionally offset the new range by this many columns. |
886
|
|
|
|
|
|
|
row: Optionally offset the new range by this many rows. |
887
|
|
|
|
|
|
|
top: Optionally offset the new range with the new top. |
888
|
|
|
|
|
|
|
left: Optionally offset the new range with the new left. |
889
|
|
|
|
|
|
|
bottom: Optionally offset the new range with the new bottom. |
890
|
|
|
|
|
|
|
right: Optionally offset the new range with the new right. |
891
|
|
|
|
|
|
|
|
892
|
|
|
|
|
|
|
=item offsets(range<Range>); |
893
|
|
|
|
|
|
|
|
894
|
|
|
|
|
|
|
Returns the offsets of this range from the given range in an |
895
|
|
|
|
|
|
|
array (top, left, bottom, right). |
896
|
|
|
|
|
|
|
|
897
|
|
|
|
|
|
|
=item iterator(%args); |
898
|
|
|
|
|
|
|
|
899
|
|
|
|
|
|
|
Returns an iterator for this range. Any 'args' are passed to the |
900
|
|
|
|
|
|
|
'new' routine for the iterator. |
901
|
|
|
|
|
|
|
|
902
|
|
|
|
|
|
|
=item is_other_inside(range<Range>); |
903
|
|
|
|
|
|
|
|
904
|
|
|
|
|
|
|
Returns a true value if the given range fits entirely inside this range. |
905
|
|
|
|
|
|
|
|
906
|
|
|
|
|
|
|
=item share_values(range<Range>); |
907
|
|
|
|
|
|
|
|
908
|
|
|
|
|
|
|
Share the cell values between this range and the passed range. |
909
|
|
|
|
|
|
|
|
910
|
|
|
|
|
|
|
Normally ranges are short-lived, throw-away objects used to interact |
911
|
|
|
|
|
|
|
with the Google API. Range objects work independently and don't |
912
|
|
|
|
|
|
|
share any common storage of cells. No attempt is made to share a |
913
|
|
|
|
|
|
|
common, local grid that mirrors what's in the remote spreadsheet. |
914
|
|
|
|
|
|
|
The spreadsheet is a shared resource that can be updated concurrently |
915
|
|
|
|
|
|
|
by multiple users, so attempting to mirror it locally is a waste of |
916
|
|
|
|
|
|
|
time unless some kind of listener is provided that can be used to |
917
|
|
|
|
|
|
|
syncronize the remote updates with the local copy (which Google doesn't). |
918
|
|
|
|
|
|
|
|
919
|
|
|
|
|
|
|
However, for some cases, like iterators, we can set up ranges to |
920
|
|
|
|
|
|
|
share values between them for a short time in order to reduce the |
921
|
|
|
|
|
|
|
number of network calls to the Google API while you iterate over |
922
|
|
|
|
|
|
|
the cells. When the iterated cell values are updated, the parent |
923
|
|
|
|
|
|
|
range's value is also updated. |
924
|
|
|
|
|
|
|
|
925
|
|
|
|
|
|
|
So far, this is the only use-case I can see for sharing the cell |
926
|
|
|
|
|
|
|
grid between ranges. |
927
|
|
|
|
|
|
|
|
928
|
|
|
|
|
|
|
=item dimension(); |
929
|
|
|
|
|
|
|
|
930
|
|
|
|
|
|
|
Returns this range's major dimension (col|row). |
931
|
|
|
|
|
|
|
|
932
|
|
|
|
|
|
|
=item has_values(); |
933
|
|
|
|
|
|
|
|
934
|
|
|
|
|
|
|
Returns a true value if this range's values have been set either via |
935
|
|
|
|
|
|
|
'values' or 'batch_values' routines. |
936
|
|
|
|
|
|
|
|
937
|
|
|
|
|
|
|
=item worksheet(); |
938
|
|
|
|
|
|
|
|
939
|
|
|
|
|
|
|
Returns the parent Worksheet object. |
940
|
|
|
|
|
|
|
|
941
|
|
|
|
|
|
|
=item worksheet_name(); |
942
|
|
|
|
|
|
|
|
943
|
|
|
|
|
|
|
Returns the parent Worksheet name. |
944
|
|
|
|
|
|
|
|
945
|
|
|
|
|
|
|
=item worksheet_id(); |
946
|
|
|
|
|
|
|
|
947
|
|
|
|
|
|
|
Returns the parent Worksheet ID. |
948
|
|
|
|
|
|
|
|
949
|
|
|
|
|
|
|
=item spreadsheet(); |
950
|
|
|
|
|
|
|
|
951
|
|
|
|
|
|
|
Returns the parent Spreadsheet object. |
952
|
|
|
|
|
|
|
|
953
|
|
|
|
|
|
|
=item spreadsheet_id(); |
954
|
|
|
|
|
|
|
|
955
|
|
|
|
|
|
|
Returns the parent Spreadsheet ID. |
956
|
|
|
|
|
|
|
|
957
|
|
|
|
|
|
|
=back |
958
|
|
|
|
|
|
|
|
959
|
|
|
|
|
|
|
=head1 AUTHORS |
960
|
|
|
|
|
|
|
|
961
|
|
|
|
|
|
|
=over |
962
|
|
|
|
|
|
|
|
963
|
|
|
|
|
|
|
=item |
964
|
|
|
|
|
|
|
|
965
|
|
|
|
|
|
|
Robin Murray mvsjes@cpan.org |
966
|
|
|
|
|
|
|
|
967
|
|
|
|
|
|
|
=back |
968
|
|
|
|
|
|
|
|
969
|
|
|
|
|
|
|
=head1 COPYRIGHT |
970
|
|
|
|
|
|
|
|
971
|
|
|
|
|
|
|
Copyright (c) 2021, Robin Murray. All rights reserved. |
972
|
|
|
|
|
|
|
|
973
|
|
|
|
|
|
|
This program is free software; you may redistribute it and/or modify it under the same terms as Perl itself. |