sequence 0.2.3 → 0.2.4
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 -0
- data/History.txt +10 -0
- data/Makefile +1 -0
- data/README.txt +17 -13
- data/lib/assert.rb +1 -1
- data/lib/sequence.rb +591 -591
- data/lib/sequence/arraylike.rb +1 -1
- data/lib/sequence/buffered.rb +1 -1
- data/lib/sequence/circular.rb +48 -48
- data/lib/sequence/enum.rb +1 -1
- data/lib/sequence/file.rb +1 -8
- data/lib/sequence/functional.rb +2 -2
- data/lib/sequence/generator.rb +1 -0
- data/lib/sequence/indexed.rb +1 -1
- data/lib/sequence/io.rb +1 -1
- data/lib/sequence/list.rb +23 -10
- data/lib/sequence/ofhash.rb +1 -1
- data/lib/sequence/ofobjectivars.rb +1 -1
- data/lib/sequence/ofobjectmethods.rb +1 -1
- data/lib/sequence/position.rb +2 -2
- data/lib/sequence/reversed.rb +1 -1
- data/lib/sequence/shifting.rb +1 -1
- data/lib/sequence/singleitem.rb +1 -1
- data/lib/sequence/stringio.rb +9 -0
- data/lib/sequence/stringlike.rb +4 -2
- data/lib/sequence/subseq.rb +1 -1
- data/lib/sequence/usedata.rb +3 -3
- data/lib/sequence/version.rb +2 -2
- data/lib/sequence/weakrefset.rb +2 -2
- data/sequence.gemspec +5 -4
- data/test/test_all.rb +1 -1
- data/test/test_changes.rb +1 -1
- data/test/test_rexscan.rb +14 -17
- data/test/test_seqrex.rb +23 -16
- metadata +34 -31
checksums.yaml
ADDED
@@ -0,0 +1,4 @@
|
|
1
|
+
---
|
2
|
+
SHA512:
|
3
|
+
metadata.gz: ed3c3847f992a80b9633c47fb04f1c51bf5f99b72b67a9341e2835f31fabb4a7b233011d3fc0c9061e4d70d436d3dadca44ce5c1e5356fabb90815209aad6871
|
4
|
+
data.tar.gz: 8cfa5d3d16cf63995b39747be1a6d0228d3d5bc25a2747cf71edbdc8cfa456c60a334f714e5c70bd0ac2813347b4761b749fe13045adb5305d558d52829ece90
|
data/History.txt
CHANGED
@@ -1,3 +1,13 @@
|
|
1
|
+
=== 0.2.4 / 16sep2011
|
2
|
+
* 1 Major Bugfix:
|
3
|
+
* hopefully should not break StringIO anymore
|
4
|
+
* 3 Minor Bugfixes:
|
5
|
+
* increased 1.9 compatibility
|
6
|
+
* force rewritten regexps created by sequence to be binary encoded (hacky)
|
7
|
+
* in gemspec, seek for files relative to __FILE__
|
8
|
+
* 1 Minor Enhancement:
|
9
|
+
* in Sequence::List.new, coerce subseqs to Sequences if needed
|
10
|
+
|
1
11
|
=== 0.2.3 / 21dec2009
|
2
12
|
* 3 Minor Bugfixes:
|
3
13
|
* use Array#join instead of Array#to_s for 1.9 compatibility
|
data/Makefile
CHANGED
data/README.txt
CHANGED
@@ -1,19 +1,23 @@
|
|
1
1
|
= sequence
|
2
2
|
|
3
|
-
* http://sequence.rubyforge.org/
|
4
3
|
* http://github.com/coatl/sequence
|
4
|
+
* http://sequence.rubyforge.org/
|
5
5
|
|
6
6
|
== DESCRIPTION:
|
7
7
|
Sequence provides a unified api for access to sequential data types, like
|
8
|
-
Strings, Arrays, Files, IOs, and Enumerations.
|
9
|
-
|
10
|
-
|
11
|
-
|
12
|
-
|
13
|
-
|
14
|
-
|
15
|
-
|
16
|
-
|
8
|
+
Strings, Arrays, Files, IOs, and Enumerations. This is the external
|
9
|
+
iterator pattern (ruby's usual iterators are internal). Each sequence
|
10
|
+
encapsulates some data and a current position within it. Some operations
|
11
|
+
apply to data at (or relative to) the position, others are independant
|
12
|
+
of position. The api contains operations for moving the position, and
|
13
|
+
reading and writing data (with or without moving the position) forward
|
14
|
+
or backward from the current position or anywhere.
|
15
|
+
|
16
|
+
Its perhaps most unusual feature is the ability to scan for Regexps in
|
17
|
+
not just Strings, but Files and any other type of sequence.
|
18
|
+
|
19
|
+
It can also save a position that will remain valid even after data is
|
20
|
+
deleted or inserted elsewhere within the sequence.
|
17
21
|
|
18
22
|
There are also some utility classes for making sequences reversed or
|
19
23
|
circular, turning one-way sequences into two-way, buffering, and making
|
@@ -25,8 +29,8 @@ very much a derivative of his.
|
|
25
29
|
|
26
30
|
|
27
31
|
Sequences always fall into one of two broad categories: string-like and
|
28
|
-
array- like. String-like
|
29
|
-
Array-like
|
32
|
+
array- like. String-like sequences contain only character data, whereas
|
33
|
+
Array-like sequences contain objects.
|
30
34
|
|
31
35
|
|
32
36
|
|
@@ -54,7 +58,7 @@ No unit tests at all for array-like sequences
|
|
54
58
|
|
55
59
|
== LICENSE:
|
56
60
|
|
57
|
-
Copyright (C) 2006,2008 Caleb Clausen
|
61
|
+
Copyright (C) 2006,2008, 2011 Caleb Clausen
|
58
62
|
Distributed under the terms of Ruby's license. (See the file COPYING.)
|
59
63
|
|
60
64
|
== USING:
|
data/lib/assert.rb
CHANGED
data/lib/sequence.rb
CHANGED
@@ -1,4 +1,4 @@
|
|
1
|
-
# Copyright (C) 2006,2008 Caleb Clausen
|
1
|
+
# Copyright (C) 2006,2008, 2011 Caleb Clausen
|
2
2
|
# Distributed under the terms of Ruby's license.
|
3
3
|
# = sequence.rb - external iterators with capabilities like a text editor cursor
|
4
4
|
# $Id$
|
@@ -34,131 +34,131 @@ class Sequence
|
|
34
34
|
# WeakRefSet=::Set
|
35
35
|
# warn "warning: sequence uses Set instead of WeakRefSet; memory leak results"
|
36
36
|
|
37
|
-
|
38
|
-
|
39
|
-
|
40
|
-
|
41
|
-
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
|
51
|
-
|
37
|
+
#include Comparable #bah, endless trouble, no use
|
38
|
+
include Enumerable
|
39
|
+
extend Forwardable
|
40
|
+
|
41
|
+
def initialize; abstract end
|
42
|
+
def to_sequence; self end
|
43
|
+
|
44
|
+
class<<self
|
45
|
+
if nil #borken
|
46
|
+
alias original__new new
|
47
|
+
undef new #bye-bye
|
48
|
+
def new(seq)
|
49
|
+
case seq
|
50
|
+
when File,IO,Array,String,Enumerable
|
51
|
+
seq.to_sequence
|
52
|
+
else
|
53
|
+
if seq.respond_to? :to_str
|
54
|
+
seq.to_str.to_sequence
|
52
55
|
else
|
53
|
-
|
54
|
-
|
55
|
-
else
|
56
|
-
raise ArgumentError
|
57
|
-
end
|
58
|
-
end
|
59
|
-
end
|
56
|
+
raise ArgumentError
|
57
|
+
end
|
60
58
|
end
|
61
|
-
def [](*x) new(*x) end
|
62
59
|
end
|
60
|
+
end
|
61
|
+
def [](*x) new(*x) end
|
62
|
+
end
|
63
63
|
|
64
64
|
|
65
|
-
|
66
|
-
|
67
|
-
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
# read previous element or nil if start of input and move position back
|
72
|
-
def readback1
|
73
|
-
(readback 1)[0]
|
74
|
-
end
|
75
|
-
|
76
|
-
# read element after the pos or nil if eof, leaving position alone
|
77
|
-
def readahead1
|
78
|
-
slice pos
|
79
|
-
end
|
80
|
-
|
81
|
-
# read element before the pos or nil if start of input, leaving position alone
|
82
|
-
def readbehind1
|
83
|
-
slice pos-1 unless pos.zero?
|
84
|
-
end
|
85
|
-
|
65
|
+
public
|
66
|
+
# read next element or nil if eof and advance position
|
67
|
+
def read1
|
68
|
+
(read 1)[0]
|
69
|
+
end
|
86
70
|
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
#means you're at end of file; use #eof? to test for that instead.
|
92
|
-
def read(len)
|
93
|
-
abstract
|
94
|
-
end
|
95
|
-
|
96
|
-
#like read, but position is left alone.
|
97
|
-
def readahead(len)
|
98
|
-
holding{read(len)}
|
99
|
-
end
|
100
|
-
|
101
|
-
#read data behind the current position, leaving position unchanged
|
102
|
-
def readbehind(len)
|
103
|
-
len>pos and len=pos
|
104
|
-
read move( -len)
|
105
|
-
end
|
106
|
-
|
107
|
-
#read data behind the current position, leaving position just before the data read.
|
108
|
-
def readback(len)
|
109
|
-
len>pos and len=pos
|
110
|
-
readahead move( -len )
|
111
|
-
end
|
71
|
+
# read previous element or nil if start of input and move position back
|
72
|
+
def readback1
|
73
|
+
(readback 1)[0]
|
74
|
+
end
|
112
75
|
|
113
|
-
|
114
|
-
|
115
|
-
|
116
|
-
|
117
|
-
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
125
|
-
|
126
|
-
|
127
|
-
|
76
|
+
# read element after the pos or nil if eof, leaving position alone
|
77
|
+
def readahead1
|
78
|
+
slice pos
|
79
|
+
end
|
80
|
+
|
81
|
+
# read element before the pos or nil if start of input, leaving position alone
|
82
|
+
def readbehind1
|
83
|
+
slice pos-1 unless pos.zero?
|
84
|
+
end
|
85
|
+
|
86
|
+
|
87
|
+
# attempt to read up to +len+ elements. the position is left just after the data read.
|
88
|
+
# #read may return less than the whole requested amount if less data than requested in
|
89
|
+
#+len+ is available. This can happen at end of file or if more data is simply unavailable
|
90
|
+
#currently (ie with a Sequence::IO). Don't assume that getting less than you requested
|
91
|
+
#means you're at end of file; use #eof? to test for that instead.
|
92
|
+
def read(len)
|
93
|
+
abstract
|
94
|
+
end
|
95
|
+
|
96
|
+
#like read, but position is left alone.
|
97
|
+
def readahead(len)
|
98
|
+
holding{read(len)}
|
99
|
+
end
|
100
|
+
|
101
|
+
#read data behind the current position, leaving position unchanged
|
102
|
+
def readbehind(len)
|
103
|
+
len>pos and len=pos
|
104
|
+
read move( -len)
|
105
|
+
end
|
106
|
+
|
107
|
+
#read data behind the current position, leaving position just before the data read.
|
108
|
+
def readback(len)
|
109
|
+
len>pos and len=pos
|
110
|
+
readahead move( -len )
|
111
|
+
end
|
112
|
+
|
113
|
+
# read the remaining elements.
|
114
|
+
# if reverse, read everything behind position
|
115
|
+
def read!(reverse=false)
|
116
|
+
if reverse
|
117
|
+
readback pos
|
118
|
+
else
|
119
|
+
read rest_size
|
128
120
|
end
|
121
|
+
end
|
129
122
|
|
123
|
+
def all_data
|
124
|
+
holding_position{|posi|
|
125
|
+
posi.begin!
|
126
|
+
posi.read!
|
127
|
+
}
|
128
|
+
end
|
130
129
|
|
131
130
|
|
132
131
|
|
133
|
-
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
139
|
-
|
140
|
-
|
141
|
-
|
142
|
-
|
143
|
-
|
144
|
-
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
160
|
-
|
161
|
-
|
132
|
+
|
133
|
+
#a StringScanner-like interface for pattern scanning within sequences.
|
134
|
+
#See StringScanner for /(scan|skip|check)(_until)?|match\?|exist\?/.
|
135
|
+
#Some notes on the implementation: scanning is all done at the current
|
136
|
+
#position. Unlike StringScanner, which only scans for Regexp, this
|
137
|
+
#version of #scan, etc allow more pattern types. The pattern type and how
|
138
|
+
#it is treated are determined by whether the underlying data is String-
|
139
|
+
#like (contains only characters/bytes), or Array-like (contains any
|
140
|
+
#Object).
|
141
|
+
|
142
|
+
#If String-like: scan and friends can take a Regexp, String, or
|
143
|
+
#Integer (for a single char) parameter.
|
144
|
+
#If Array-like: scan and friends can take a scalar matcher (something
|
145
|
+
#that responds to ===). Eventually, vector matchers (Reg::Multiple)
|
146
|
+
#will be supported as well here. Literal arrays, as patterns to scan for,
|
147
|
+
#are not supported: too hard, and there's the quoting problem.
|
148
|
+
#if you actually want to scan for items that are equal to a particular
|
149
|
+
#Regexp (for example), instead of items that match it, use duck.rb
|
150
|
+
#to reassign === as ==. Or, if using Reg, create a Reg::Literal by
|
151
|
+
#calling #lit on the pattern (which has the same effect).
|
152
|
+
|
153
|
+
#when scanning string-like sequences, anchors in Regexps are treated somewhat
|
154
|
+
#specially:
|
155
|
+
#when scanning forward, ^ and \A match at the current position, and
|
156
|
+
#$ and \Z match at the end of the sequence.
|
157
|
+
#when scanning backward, ^ and \A match at the beginning of sequence and $ and
|
158
|
+
#\Z match at the current position.
|
159
|
+
#^ and $ still match at begining and end of lines, as usual.
|
160
|
+
|
161
|
+
#when scanning or matching backwards, ^ has some special problems:
|
162
162
|
# these won't work....
|
163
163
|
#an anchor that might or might not match will break the current implementation in some cases...
|
164
164
|
#regexes that should match overall even if the anchor in them doesn't...
|
@@ -179,321 +179,321 @@ class Sequence
|
|
179
179
|
|
180
180
|
|
181
181
|
=begin can't be abstract... defined in modules
|
182
|
-
|
183
|
-
|
184
|
-
|
182
|
+
def scan(pat)
|
183
|
+
abstract
|
184
|
+
end
|
185
185
|
|
186
|
-
|
187
|
-
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
186
|
+
def scan_until(pat)
|
187
|
+
abstract
|
188
|
+
end
|
189
|
+
def scanback(pat)
|
190
|
+
abstract
|
191
|
+
end
|
192
192
|
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
197
|
-
|
198
|
-
|
199
|
-
|
200
|
-
|
201
|
-
|
202
|
-
|
203
|
-
|
193
|
+
def scanback_until(pat)
|
194
|
+
abstract
|
195
|
+
end
|
196
|
+
|
197
|
+
def match(pat)
|
198
|
+
abstract
|
199
|
+
end
|
200
|
+
|
201
|
+
def matchback(pat)
|
202
|
+
abstract
|
203
|
+
end
|
204
204
|
=end
|
205
205
|
|
206
|
-
|
207
|
-
|
208
|
-
|
209
|
-
|
210
|
-
def skip_until(pat) match= scan_until(pat) and match.length end
|
211
|
-
def check_until(pat) holding{scan_until(pat)} end
|
212
|
-
def exist?(pat) holding{skip_until(pat)} end
|
206
|
+
def skip(pat) match= scan(pat) and match.length end
|
207
|
+
def check(pat) holding{scan(pat)} end
|
208
|
+
def match?(pat) holding{skip(pat)} end
|
213
209
|
|
214
|
-
|
215
|
-
|
216
|
-
|
217
|
-
|
218
|
-
def skipback_until(pat) match= scanback_until(pat) and match.length end
|
219
|
-
def checkback_until(pat) holding{scanback_until(pat)} end
|
220
|
-
def existback?(pat) holding{skipback_until(pat) }end
|
221
|
-
|
222
|
-
def skip_literal(lits)
|
223
|
-
sz=lits.size
|
224
|
-
lits==readahead(sz) and move sz
|
225
|
-
end
|
226
|
-
alias skip_literals skip_literal
|
227
|
-
|
228
|
-
def skip_until_literal(lits)
|
229
|
-
sz=lits.size
|
230
|
-
first=lits[0]
|
231
|
-
holding?{
|
232
|
-
until eof?
|
233
|
-
skip_until(first)
|
234
|
-
lits==readahead(sz) and break pos
|
235
|
-
end
|
236
|
-
}
|
237
|
-
end
|
238
|
-
alias skip_until_literals skip_until_literal
|
239
|
-
|
240
|
-
attr_accessor :last_match
|
241
|
-
attr_writer :maxmatchlen
|
242
|
-
|
243
|
-
def _default_maxmatchlen; 1024 end
|
244
|
-
|
245
|
-
def maxmatchlen(backwards)
|
246
|
-
size=self.size
|
247
|
-
|
248
|
-
list=[ _default_maxmatchlen,
|
249
|
-
backwards ? pos : size-pos%size
|
250
|
-
]
|
251
|
-
list.push @maxmatchlen if defined? @maxmatchlen
|
252
|
-
list.min
|
253
|
-
end
|
210
|
+
def skip_until(pat) match= scan_until(pat) and match.length end
|
211
|
+
def check_until(pat) holding{scan_until(pat)} end
|
212
|
+
def exist?(pat) holding{skip_until(pat)} end
|
254
213
|
|
255
|
-
|
256
|
-
|
257
|
-
|
258
|
-
#original location. The return value is the result of the block.
|
259
|
-
def holding
|
260
|
-
oldpos=pos
|
261
|
-
begin
|
262
|
-
yield self
|
263
|
-
ensure
|
264
|
-
self.pos=oldpos
|
265
|
-
end
|
266
|
-
end
|
214
|
+
def skipback(pat) match= scanback(pat) and match.length end
|
215
|
+
def checkback(pat) holding{scanback(pat)} end
|
216
|
+
def matchback?(pat) holding{skipback(pat)} end
|
267
217
|
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
oldpos=pos
|
272
|
-
begin
|
273
|
-
result=yield self
|
274
|
-
ensure
|
275
|
-
(self.pos=oldpos) unless result
|
276
|
-
end
|
277
|
-
end
|
218
|
+
def skipback_until(pat) match= scanback_until(pat) and match.length end
|
219
|
+
def checkback_until(pat) holding{scanback_until(pat)} end
|
220
|
+
def existback?(pat) holding{skipback_until(pat) }end
|
278
221
|
|
279
|
-
|
280
|
-
|
281
|
-
|
282
|
-
|
283
|
-
|
284
|
-
|
285
|
-
|
286
|
-
|
222
|
+
def skip_literal(lits)
|
223
|
+
sz=lits.size
|
224
|
+
lits==readahead(sz) and move sz
|
225
|
+
end
|
226
|
+
alias skip_literals skip_literal
|
227
|
+
|
228
|
+
def skip_until_literal(lits)
|
229
|
+
sz=lits.size
|
230
|
+
first=lits[0]
|
231
|
+
holding?{
|
232
|
+
until eof?
|
233
|
+
skip_until(first)
|
234
|
+
lits==readahead(sz) and break pos
|
287
235
|
end
|
236
|
+
}
|
237
|
+
end
|
238
|
+
alias skip_until_literals skip_until_literal
|
288
239
|
|
240
|
+
attr_accessor :last_match
|
241
|
+
attr_writer :maxmatchlen
|
289
242
|
|
290
|
-
|
291
|
-
|
292
|
-
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
|
298
|
-
|
299
|
-
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
304
|
-
|
305
|
-
|
306
|
-
|
307
|
-
|
243
|
+
def _default_maxmatchlen; 1024 end
|
244
|
+
|
245
|
+
def maxmatchlen(backwards)
|
246
|
+
size=self.size
|
247
|
+
|
248
|
+
list=[ _default_maxmatchlen,
|
249
|
+
backwards ? pos : size-pos%size
|
250
|
+
]
|
251
|
+
list.push @maxmatchlen if defined? @maxmatchlen
|
252
|
+
list.min
|
253
|
+
end
|
254
|
+
|
255
|
+
#hold current position while executing a block. The block is passed the current
|
256
|
+
#sequence as its parameter. you can move the position around or call methods
|
257
|
+
#like read that do it, but after the block returns, the position is reset to the
|
258
|
+
#original location. The return value is the result of the block.
|
259
|
+
def holding
|
260
|
+
oldpos=pos
|
261
|
+
begin
|
262
|
+
yield self
|
263
|
+
ensure
|
264
|
+
self.pos=oldpos
|
308
265
|
end
|
309
|
-
|
310
|
-
|
311
|
-
|
312
|
-
|
313
|
-
|
314
|
-
|
315
|
-
|
316
|
-
|
317
|
-
|
266
|
+
end
|
267
|
+
|
268
|
+
#like #holding, but position is reset only if block returns false or nil (or
|
269
|
+
#raises an exception).
|
270
|
+
def holding?
|
271
|
+
oldpos=pos
|
272
|
+
begin
|
273
|
+
result=yield self
|
274
|
+
ensure
|
275
|
+
(self.pos=oldpos) unless result
|
318
276
|
end
|
319
|
-
|
277
|
+
end
|
320
278
|
|
321
|
-
|
322
|
-
|
323
|
-
|
279
|
+
#like #holding, but block is instance_eval'd in the sequence.
|
280
|
+
def holding! &block
|
281
|
+
oldpos=pos
|
282
|
+
begin
|
283
|
+
instance_eval self, &block
|
284
|
+
ensure
|
285
|
+
self.pos=oldpos
|
324
286
|
end
|
325
|
-
|
287
|
+
end
|
288
|
+
|
326
289
|
|
327
|
-
|
328
|
-
|
329
|
-
|
330
|
-
|
290
|
+
def holding_position
|
291
|
+
pos=position
|
292
|
+
begin
|
293
|
+
result=yield self
|
294
|
+
ensure
|
295
|
+
self.position=pos
|
296
|
+
pos.close
|
331
297
|
end
|
298
|
+
end
|
332
299
|
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
300
|
+
def holding_position?
|
301
|
+
pos=position
|
302
|
+
begin
|
303
|
+
result=yield self
|
304
|
+
ensure
|
305
|
+
self.position=pos unless result
|
306
|
+
pos.close
|
337
307
|
end
|
338
|
-
|
339
|
-
|
340
|
-
|
341
|
-
|
308
|
+
end
|
309
|
+
|
310
|
+
def holding_position! &block
|
311
|
+
pos=position
|
312
|
+
begin
|
313
|
+
result=instance_eval self,&block
|
314
|
+
ensure
|
315
|
+
self.position=pos
|
316
|
+
pos.close
|
342
317
|
end
|
343
|
-
|
344
|
-
|
318
|
+
end
|
319
|
+
|
320
|
+
|
321
|
+
# number of elements from the beginning (0 is at the beginning).
|
322
|
+
def pos()
|
345
323
|
abstract
|
346
|
-
|
347
|
-
|
348
|
-
# A negative +len+ will go in reverse.
|
349
|
-
# The (positive) amount actually moved is returned (<+len+ if reached beginning/end).
|
350
|
-
def move(len)
|
351
|
-
oldpos=pos
|
352
|
-
newpos=oldpos+len
|
353
|
-
newpos<0 and newpos=0
|
354
|
-
goto newpos
|
355
|
-
return (pos-oldpos).abs
|
356
|
-
end
|
357
|
-
# move to end of the remaining elements.
|
358
|
-
# reverse=true to move to beginning instead of end
|
359
|
-
# The amount moved is returned.
|
360
|
-
def move!(reverse=false)
|
361
|
-
reverse ? begin! : end!
|
362
|
-
end
|
324
|
+
end
|
325
|
+
def rest_size; size - pos end
|
363
326
|
|
364
|
-
|
365
|
-
|
366
|
-
|
367
|
-
|
368
|
-
|
369
|
-
if value.size.zero?
|
370
|
-
defined?(@prop) &&@prop&&@prop.clone
|
371
|
-
else
|
372
|
-
if (value = value[0]).nil?
|
373
|
-
defined?(@prop) &&@prop&&remove_instance_variable(:@prop)
|
374
|
-
else
|
375
|
-
(@prop||={}).replace(value)
|
376
|
-
end
|
377
|
-
end
|
378
|
-
else
|
379
|
-
if value.size.zero?
|
380
|
-
defined?(@prop) &&@prop&&@prop[name]
|
381
|
-
else
|
382
|
-
(@prop||={})[name] = value[0]
|
383
|
-
end
|
384
|
-
end
|
385
|
-
end
|
327
|
+
# this checks to see if p is a valid numeric position.
|
328
|
+
def pos?(p)
|
329
|
+
sz=size
|
330
|
+
(-sz..sz)===p
|
331
|
+
end
|
386
332
|
|
387
|
-
|
388
|
-
|
389
|
-
|
390
|
-
|
391
|
-
|
392
|
-
def position(_pos=pos)
|
393
|
-
Position.new(self,_pos)
|
394
|
-
end
|
395
|
-
# Set the position to a Position +p+ (from #position).
|
396
|
-
def position=(p)
|
397
|
-
self.pos = p.pos
|
398
|
-
self.prop(nil,p.prop)
|
399
|
-
p
|
400
|
-
end
|
333
|
+
# Set #pos to be +p+. When +p+ is negative, it is set from the end.
|
334
|
+
def pos=(p)
|
335
|
+
position?(p) and p=p.pos unless Integer===p
|
336
|
+
self._pos=_normalize_pos p
|
337
|
+
end
|
401
338
|
|
402
|
-
|
403
|
-
|
404
|
-
|
405
|
-
|
406
|
-
when Integer; (-size..size)===p
|
407
|
-
when Position; equal? p.data
|
408
|
-
else equal? p
|
409
|
-
end
|
410
|
-
end
|
411
|
-
|
412
|
-
#make a new sequence out of a subrange of current sequence data.
|
413
|
-
#the subseq and parent seq share data, so changes in one
|
414
|
-
#will be reflected in the other.
|
415
|
-
def subseq(*args)
|
416
|
-
assert !closed?
|
417
|
-
first,len,only1=_parse_slice_args(*args)
|
418
|
-
SubSeq.new(self,first,len)
|
419
|
-
end
|
420
|
-
|
421
|
-
#make a new sequence that reverses the order of data.
|
422
|
-
#reversed and parent sequence share data.
|
423
|
-
def reversed
|
424
|
-
Reversed.new self
|
425
|
-
end
|
426
|
-
|
427
|
-
# Close the sequence. This will also close/invalidate every child
|
428
|
-
# position or derived sequence.
|
429
|
-
def close
|
430
|
-
defined? @change_listeners and @change_listeners.each { |p|
|
431
|
-
Sequence===p and p.close
|
432
|
-
}
|
433
|
-
# this should make just about any operation fail
|
434
|
-
instance_variables.each { |v| remove_instance_variable(v) }
|
435
|
-
nil
|
436
|
-
end
|
437
|
-
# Is this sequence closed?
|
438
|
-
def closed?
|
439
|
-
instance_variables.empty?
|
440
|
-
end
|
339
|
+
#go to an absolute position; identical to #pos=
|
340
|
+
def goto p
|
341
|
+
self.pos= p
|
342
|
+
end
|
441
343
|
|
442
|
-
=
|
443
|
-
|
444
|
-
|
445
|
-
|
446
|
-
|
447
|
-
|
448
|
-
|
449
|
-
|
450
|
-
|
451
|
-
=
|
344
|
+
def _pos=(p)
|
345
|
+
abstract
|
346
|
+
end
|
347
|
+
# move position +len+ elements, relative to the current position.
|
348
|
+
# A negative +len+ will go in reverse.
|
349
|
+
# The (positive) amount actually moved is returned (<+len+ if reached beginning/end).
|
350
|
+
def move(len)
|
351
|
+
oldpos=pos
|
352
|
+
newpos=oldpos+len
|
353
|
+
newpos<0 and newpos=0
|
354
|
+
goto newpos
|
355
|
+
return (pos-oldpos).abs
|
356
|
+
end
|
357
|
+
# move to end of the remaining elements.
|
358
|
+
# reverse=true to move to beginning instead of end
|
359
|
+
# The amount moved is returned.
|
360
|
+
def move!(reverse=false)
|
361
|
+
reverse ? begin! : end!
|
362
|
+
end
|
452
363
|
|
453
|
-
|
454
|
-
|
455
|
-
|
456
|
-
|
457
|
-
|
458
|
-
|
364
|
+
# Get (if no +value+) and set properties. Normally, +name+
|
365
|
+
# should be a symbol. If +name+ is +nil+, it wil get/set using a hash
|
366
|
+
# representing all of the properties.
|
367
|
+
def prop(name=nil,*value) # :args: (name[,value])
|
368
|
+
if name.nil?
|
369
|
+
if value.size.zero?
|
370
|
+
defined?(@prop) &&@prop&&@prop.clone
|
459
371
|
else
|
460
|
-
|
372
|
+
if (value = value[0]).nil?
|
373
|
+
defined?(@prop) &&@prop&&remove_instance_variable(:@prop)
|
374
|
+
else
|
375
|
+
(@prop||={}).replace(value)
|
376
|
+
end
|
461
377
|
end
|
462
|
-
|
463
|
-
|
464
|
-
|
465
|
-
if ::Sequence===other
|
466
|
-
List[self, other]
|
378
|
+
else
|
379
|
+
if value.size.zero?
|
380
|
+
defined?(@prop) &&@prop&&@prop[name]
|
467
381
|
else
|
468
|
-
|
382
|
+
(@prop||={})[name] = value[0]
|
469
383
|
end
|
470
384
|
end
|
385
|
+
end
|
471
386
|
|
472
|
-
|
473
|
-
|
474
|
-
|
475
|
-
|
476
|
-
|
477
|
-
|
478
|
-
|
479
|
-
|
480
|
-
|
481
|
-
|
482
|
-
|
483
|
-
|
484
|
-
|
485
|
-
|
486
|
-
|
387
|
+
# #position returns a Sequence::Position to represent a location within this sequence.
|
388
|
+
# The argument allows you to specify a numeric location for the position; default is
|
389
|
+
# currrent position. If the element that a
|
390
|
+
# Position is anchored to is deleted, that Position may become invalid
|
391
|
+
# or have an unknown behavior.
|
392
|
+
def position(_pos=pos)
|
393
|
+
Position.new(self,_pos)
|
394
|
+
end
|
395
|
+
# Set the position to a Position +p+ (from #position).
|
396
|
+
def position=(p)
|
397
|
+
self.pos = p.pos
|
398
|
+
self.prop(nil,p.prop)
|
399
|
+
p
|
400
|
+
end
|
401
|
+
|
402
|
+
# this queries whether a particular #position +p+ is valid (is a child or self).
|
403
|
+
# numeric positions and also be tested
|
404
|
+
def position?(p)
|
405
|
+
case p
|
406
|
+
when Integer; (-size..size)===p
|
407
|
+
when Position; equal? p.data
|
408
|
+
else equal? p
|
487
409
|
end
|
488
|
-
|
489
|
-
|
490
|
-
|
491
|
-
|
410
|
+
end
|
411
|
+
|
412
|
+
#make a new sequence out of a subrange of current sequence data.
|
413
|
+
#the subseq and parent seq share data, so changes in one
|
414
|
+
#will be reflected in the other.
|
415
|
+
def subseq(*args)
|
416
|
+
assert !closed?
|
417
|
+
first,len,only1=_parse_slice_args(*args)
|
418
|
+
SubSeq.new(self,first,len)
|
419
|
+
end
|
420
|
+
|
421
|
+
#make a new sequence that reverses the order of data.
|
422
|
+
#reversed and parent sequence share data.
|
423
|
+
def reversed
|
424
|
+
Reversed.new self
|
425
|
+
end
|
426
|
+
|
427
|
+
# Close the sequence. This will also close/invalidate every child
|
428
|
+
# position or derived sequence.
|
429
|
+
def close
|
430
|
+
defined? @change_listeners and @change_listeners.each { |p|
|
431
|
+
Sequence===p and p.close
|
432
|
+
}
|
433
|
+
# this should make just about any operation fail
|
434
|
+
instance_variables.each { |v| remove_instance_variable(v) }
|
435
|
+
nil
|
436
|
+
end
|
437
|
+
# Is this sequence closed?
|
438
|
+
def closed?
|
439
|
+
instance_variables.empty?
|
440
|
+
end
|
441
|
+
|
442
|
+
=begin who needs it?
|
443
|
+
# Compare +other+ (a Position or Integer) to the current position. return +1
|
444
|
+
# for the self is after, -1 for self being before, and 0 for it being at
|
445
|
+
# same location, nil (or false) if other is not a position of self.
|
446
|
+
def <=>(other)
|
447
|
+
if other.respond_to? :to_i then pos<=>other
|
448
|
+
elsif position?(other) then pos<=>other.pos
|
449
|
+
end
|
450
|
+
end
|
451
|
+
=end
|
452
|
+
|
453
|
+
#if passed an integer arg, return a new position decreased by len. if passed
|
454
|
+
# a position, return the distance (number
|
455
|
+
# or elements) from +other+ (a #position) to +self+. This can be +, -, or 0.
|
456
|
+
def -(other)
|
457
|
+
if position?(other)
|
458
|
+
pos-other.pos
|
459
|
+
else
|
460
|
+
position(pos-other)
|
492
461
|
end
|
493
|
-
|
494
|
-
|
495
|
-
|
462
|
+
end
|
463
|
+
# Returns a new #position increased by +len+ (positive or negative).
|
464
|
+
def +(other)
|
465
|
+
if ::Sequence===other
|
466
|
+
List[self, other]
|
467
|
+
else
|
468
|
+
position(pos+other)
|
496
469
|
end
|
470
|
+
end
|
471
|
+
|
472
|
+
# Return a new #position for next location or +nil+ if we are at the end.
|
473
|
+
def succ
|
474
|
+
self+1 unless eof?
|
475
|
+
end
|
476
|
+
# Return a new #position for previous location or +nil+ if we are at the beginning.
|
477
|
+
def pred
|
478
|
+
self-1 unless pos.zero?
|
479
|
+
end
|
480
|
+
# Return a new #position for the beginning.
|
481
|
+
def begin
|
482
|
+
position(0)
|
483
|
+
end
|
484
|
+
# Return a new #position for the end.
|
485
|
+
def end
|
486
|
+
position(size)
|
487
|
+
end
|
488
|
+
|
489
|
+
#go to beginning
|
490
|
+
def begin!
|
491
|
+
self._pos=0
|
492
|
+
end
|
493
|
+
#go to end
|
494
|
+
def end!
|
495
|
+
self._pos=size
|
496
|
+
end
|
497
497
|
|
498
498
|
#-------------------------------------
|
499
499
|
#is position within len elements of the beginning?
|
@@ -507,216 +507,216 @@ class Sequence
|
|
507
507
|
at+len>=size
|
508
508
|
end
|
509
509
|
|
510
|
-
|
511
|
-
|
512
|
-
|
513
|
-
|
514
|
-
|
515
|
-
|
516
|
-
#has any data been seen so far, or are we still at the beginning?
|
517
|
-
def was_data?
|
518
|
-
pos.nonzero?
|
519
|
-
end
|
520
|
-
|
510
|
+
#is there any more data after the position?
|
511
|
+
def more_data?
|
512
|
+
#!eof?
|
513
|
+
(size-pos).nonzero?
|
514
|
+
end
|
521
515
|
|
522
|
-
|
523
|
-
|
524
|
-
|
525
|
-
|
526
|
-
def length; size end
|
516
|
+
#has any data been seen so far, or are we still at the beginning?
|
517
|
+
def was_data?
|
518
|
+
pos.nonzero?
|
519
|
+
end
|
527
520
|
|
528
|
-
#are we at past the end of the sequence data, with no more data ever to arrive?
|
529
|
-
def eof?
|
530
|
-
abstract
|
531
|
-
end
|
532
521
|
|
533
|
-
|
534
|
-
|
535
|
-
|
536
|
-
|
522
|
+
# Returns the number of elements.
|
523
|
+
def size
|
524
|
+
abstract
|
525
|
+
end
|
526
|
+
def length; size end
|
537
527
|
|
538
|
-
|
539
|
-
|
540
|
-
|
541
|
-
|
542
|
-
|
543
|
-
#return last element of data
|
544
|
-
def last
|
545
|
-
slice( -1)
|
546
|
-
end
|
528
|
+
#are we at past the end of the sequence data, with no more data ever to arrive?
|
529
|
+
def eof?
|
530
|
+
abstract
|
531
|
+
end
|
547
532
|
|
548
|
-
|
549
|
-
|
550
|
-
|
551
|
-
|
552
|
-
elsif pos>size
|
553
|
-
pos=size
|
554
|
-
end
|
555
|
-
|
556
|
-
assert((0..size)===pos)
|
557
|
-
pos
|
558
|
-
end
|
533
|
+
# is there any data in the sequence?
|
534
|
+
def empty?
|
535
|
+
size==0
|
536
|
+
end
|
559
537
|
|
560
|
-
|
561
|
-
|
562
|
-
|
563
|
-
|
564
|
-
case r=args.first
|
565
|
-
when Range
|
566
|
-
asize==1 or raise ArgumentError
|
567
|
-
first,last=r.first,r.last
|
568
|
-
first=_normalize_pos(first,size)
|
569
|
-
last=_normalize_pos(last,size)
|
570
|
-
len=last-first
|
571
|
-
r.exclude_end? or len+=1
|
572
|
-
when Integer
|
573
|
-
asize<=2 or raise ArgumentError
|
574
|
-
first=_normalize_pos(r,size)
|
575
|
-
len=args[1] || (only1=1)
|
576
|
-
when nil
|
577
|
-
asize==0 or raise ArgumentError
|
578
|
-
first=nil
|
579
|
-
len=only1=1
|
580
|
-
else raise ArgumentError
|
581
|
-
end
|
582
|
-
return first,len,only1
|
583
|
-
end
|
538
|
+
#return first element of data
|
539
|
+
def first
|
540
|
+
slice 0
|
541
|
+
end
|
584
542
|
|
585
|
-
|
586
|
-
|
587
|
-
|
588
|
-
|
589
|
-
# #read (positive or negative). The sequence's position is left alone.
|
590
|
-
def slice(*args) #index|range=nil,len=nil
|
591
|
-
first,len,only1=_parse_slice_args( *args)
|
592
|
-
pos==first and first=nil
|
593
|
-
holding {
|
594
|
-
self.pos = first if first
|
595
|
-
only1 ? read1 : read(len)
|
596
|
-
}
|
597
|
-
end
|
598
|
-
def [](*a) slice(*a) end
|
599
|
-
def slice1(idx) slice(idx) end
|
600
|
-
|
601
|
-
# Like #slice except the element(s) are deleted.
|
602
|
-
def slice!(*args) #index|range, len
|
603
|
-
first,len,only1=_parse_slice_args( *args)
|
604
|
-
result=slice(first,len)
|
605
|
-
delete(first,len)
|
606
|
-
only1 ? result.first : result
|
607
|
-
end
|
608
|
-
def slice1!(idx) slice!(idx) end
|
543
|
+
#return last element of data
|
544
|
+
def last
|
545
|
+
slice( -1)
|
546
|
+
end
|
609
547
|
|
610
|
-
|
611
|
-
|
612
|
-
|
613
|
-
|
614
|
-
|
615
|
-
|
616
|
-
def []=(*a) modify(*a) end
|
617
|
-
|
618
|
-
def delete(*args) #index|range, len
|
619
|
-
modify( *args<<new_data)
|
620
|
-
nil
|
621
|
-
end
|
622
|
-
|
623
|
-
def insert index, replacedata
|
624
|
-
modify index,0, replacedata
|
625
|
-
end
|
626
|
-
|
627
|
-
def overwrite index, replacedata
|
628
|
-
modify index,replacedata.size, replacedata
|
629
|
-
end
|
630
|
-
|
631
|
-
def pop count=nil
|
632
|
-
slice!(count ? -count...size : -1)
|
633
|
-
end
|
634
|
-
|
635
|
-
def shift count=nil
|
636
|
-
slice!(count ? 0...count : 0 )
|
637
|
-
end
|
638
|
-
|
639
|
-
def <<(x) push x; return self end
|
640
|
-
|
641
|
-
#push/unshift in stringlike/arraylike
|
642
|
-
|
643
|
-
def append stuff
|
644
|
-
insert(size, stuff)
|
645
|
-
self
|
548
|
+
def _normalize_pos(pos,size=self.size)
|
549
|
+
if pos<0
|
550
|
+
pos+=size
|
551
|
+
pos<0 and pos=0
|
552
|
+
elsif pos>size
|
553
|
+
pos=size
|
646
554
|
end
|
647
|
-
|
648
|
-
|
649
|
-
|
650
|
-
|
651
|
-
|
652
|
-
|
653
|
-
|
654
|
-
|
655
|
-
|
656
|
-
|
657
|
-
|
658
|
-
|
659
|
-
|
660
|
-
|
661
|
-
|
662
|
-
|
663
|
-
|
664
|
-
|
665
|
-
|
666
|
-
|
667
|
-
|
668
|
-
|
669
|
-
|
670
|
-
|
671
|
-
|
672
|
-
|
673
|
-
|
674
|
-
|
675
|
-
|
676
|
-
|
677
|
-
|
678
|
-
|
679
|
-
|
555
|
+
|
556
|
+
assert((0..size)===pos)
|
557
|
+
pos
|
558
|
+
end
|
559
|
+
|
560
|
+
def _parse_slice_args(*args)
|
561
|
+
asize=args.size
|
562
|
+
assert !closed?
|
563
|
+
size=self.size
|
564
|
+
case r=args.first
|
565
|
+
when Range
|
566
|
+
asize==1 or raise ArgumentError
|
567
|
+
first,last=r.first,r.last
|
568
|
+
first=_normalize_pos(first,size)
|
569
|
+
last=_normalize_pos(last,size)
|
570
|
+
len=last-first
|
571
|
+
r.exclude_end? or len+=1
|
572
|
+
when Integer
|
573
|
+
asize<=2 or raise ArgumentError
|
574
|
+
first=_normalize_pos(r,size)
|
575
|
+
len=args[1] || (only1=1)
|
576
|
+
when nil
|
577
|
+
asize==0 or raise ArgumentError
|
578
|
+
first=nil
|
579
|
+
len=only1=1
|
580
|
+
else raise ArgumentError
|
581
|
+
end
|
582
|
+
return first,len,only1
|
583
|
+
end
|
584
|
+
|
585
|
+
# Provides random access for the sequence like what is in Array/String.
|
586
|
+
# +index+ can be +nil+ (start at the current location) or a numeric
|
587
|
+
# (for #pos=) or a range.
|
588
|
+
# +len+ can be +nil+ (get a single element) or the number of elements to
|
589
|
+
# #read (positive or negative). The sequence's position is left alone.
|
590
|
+
def slice(*args) #index|range=nil,len=nil
|
591
|
+
first,len,only1=_parse_slice_args( *args)
|
592
|
+
pos==first and first=nil
|
593
|
+
holding {
|
594
|
+
self.pos = first if first
|
595
|
+
only1 ? read1 : read(len)
|
596
|
+
}
|
597
|
+
end
|
598
|
+
def [](*a) slice(*a) end
|
599
|
+
def slice1(idx) slice(idx) end
|
600
|
+
|
601
|
+
# Like #slice except the element(s) are deleted.
|
602
|
+
def slice!(*args) #index|range, len
|
603
|
+
first,len,only1=_parse_slice_args( *args)
|
604
|
+
result=slice(first,len)
|
605
|
+
delete(first,len)
|
606
|
+
only1 ? result.first : result
|
607
|
+
end
|
608
|
+
def slice1!(idx) slice!(idx) end
|
609
|
+
|
610
|
+
# Similar to #slice except data is written. +index+ and +len+ have the
|
611
|
+
# same meaning as they do in #slice. +len+ elements are deleted and +replacedata+
|
612
|
+
# is inserted. +replacedata+ is a single item if len is ommitted and 1st param is Fixnum
|
613
|
+
def modify(*args) #index|range, len, replacedata
|
614
|
+
abstract
|
615
|
+
end
|
616
|
+
def []=(*a) modify(*a) end
|
617
|
+
|
618
|
+
def delete(*args) #index|range, len
|
619
|
+
modify( *args<<new_data)
|
620
|
+
nil
|
621
|
+
end
|
622
|
+
|
623
|
+
def insert index, replacedata
|
624
|
+
modify index,0, replacedata
|
625
|
+
end
|
626
|
+
|
627
|
+
def overwrite index, replacedata
|
628
|
+
modify index,replacedata.size, replacedata
|
629
|
+
end
|
630
|
+
|
631
|
+
def pop count=nil
|
632
|
+
slice!(count ? -count...size : -1)
|
633
|
+
end
|
634
|
+
|
635
|
+
def shift count=nil
|
636
|
+
slice!(count ? 0...count : 0 )
|
637
|
+
end
|
638
|
+
|
639
|
+
def <<(x) push x; return self end
|
640
|
+
|
641
|
+
#push/unshift in stringlike/arraylike
|
642
|
+
|
643
|
+
def append stuff
|
644
|
+
insert(size, stuff)
|
645
|
+
self
|
646
|
+
end
|
647
|
+
|
648
|
+
def prepend stuff
|
649
|
+
insert(0, stuff)
|
650
|
+
self
|
651
|
+
end
|
652
|
+
|
653
|
+
def write(data)
|
654
|
+
assert oldpos=pos
|
655
|
+
writeahead(data)
|
656
|
+
assert oldpos==pos
|
657
|
+
move data.size
|
658
|
+
end
|
659
|
+
|
660
|
+
def writeback(data)
|
661
|
+
assert oldpos=pos
|
662
|
+
writebehind(data)
|
663
|
+
assert oldpos==pos
|
664
|
+
move( -data.size)
|
665
|
+
end
|
666
|
+
|
667
|
+
def writeahead(data)
|
668
|
+
raise ArgumentError, "attempted overwrite at end of #{self}" if data.size>rest_size
|
669
|
+
overwrite(pos,data)
|
670
|
+
data.size
|
671
|
+
end
|
672
|
+
|
673
|
+
def writebehind(data)
|
674
|
+
raise ArgumentError, "attempted overwrite at begin of #{self}" if data.size>pos
|
675
|
+
overwrite(pos-data.size,data)
|
676
|
+
data.size
|
677
|
+
end
|
678
|
+
|
679
|
+
def _adjust_pos_on_change pos,first,oldsize,newsize
|
680
680
|
# assert newsize != oldsize
|
681
|
-
|
682
|
-
|
683
|
-
|
684
|
-
|
685
|
-
|
686
|
-
|
687
|
-
end
|
681
|
+
if pos>=first+oldsize
|
682
|
+
oldsize.zero? and pos==first and return pos
|
683
|
+
pos+newsize-oldsize
|
684
|
+
elsif pos>first
|
685
|
+
first
|
686
|
+
else pos
|
688
687
|
end
|
688
|
+
end
|
689
689
|
|
690
|
-
|
691
|
-
|
692
|
-
|
693
|
-
|
694
|
-
|
695
|
-
|
696
|
-
|
697
|
-
def notify_change *args #seq, first, oldsize, newsize
|
698
|
-
args[0]=self
|
699
|
-
defined? @change_listeners and @change_listeners.each{|obj|
|
700
|
-
obj.change_notification(*args)
|
701
|
-
}
|
702
|
-
end
|
703
|
-
|
704
|
-
# Delete +p+ from the list of children (from #position).
|
705
|
-
# Should only be used by child Position.
|
706
|
-
def _delete_position(p) # :nodoc:
|
707
|
-
@change_listeners.delete(p)
|
708
|
-
end
|
690
|
+
def on_change_notify obj
|
691
|
+
Symbol===obj and raise ArgumentError
|
692
|
+
obj.respond_to? :change_notification or raise ArgumentError
|
693
|
+
@change_listeners||=WeakRefSet[]
|
694
|
+
@change_listeners<<obj
|
695
|
+
end
|
709
696
|
|
710
|
-
|
711
|
-
|
712
|
-
|
713
|
-
|
714
|
-
|
715
|
-
|
716
|
-
|
717
|
-
|
718
|
-
|
719
|
-
|
697
|
+
def notify_change *args #seq, first, oldsize, newsize
|
698
|
+
args[0]=self
|
699
|
+
defined? @change_listeners and @change_listeners.each{|obj|
|
700
|
+
obj.change_notification(*args)
|
701
|
+
}
|
702
|
+
end
|
703
|
+
|
704
|
+
# Delete +p+ from the list of children (from #position).
|
705
|
+
# Should only be used by child Position.
|
706
|
+
def _delete_position(p) # :nodoc:
|
707
|
+
@change_listeners.delete(p)
|
708
|
+
end
|
709
|
+
|
710
|
+
# Performs each just to make this class Enumerable.
|
711
|
+
# self is returned (or the break value if the code does a break).
|
712
|
+
def each # :yield: value
|
713
|
+
holding {
|
714
|
+
begin!
|
715
|
+
until eof?
|
716
|
+
yield read1
|
717
|
+
end or self
|
718
|
+
}
|
719
|
+
end
|
720
720
|
|
721
721
|
|
722
722
|
end
|