livetext 0.9.14 → 0.9.20

Sign up to get free protection for your applications and to get access to all the features.
Files changed (65) hide show
  1. checksums.yaml +4 -4
  2. data/README.lt3 +2 -2
  3. data/imports/bookish.rb +288 -0
  4. data/imports/calibre.rb +28 -0
  5. data/imports/livemagick.rb +133 -0
  6. data/imports/markdown.rb +44 -0
  7. data/imports/markdown_importable.rb +46 -0
  8. data/imports/pyggish.rb +204 -0
  9. data/imports/tutorial.rb +95 -0
  10. data/lib/cmdargs.rb +93 -0
  11. data/lib/formatline.rb +56 -83
  12. data/lib/handler/icanhaz.rb +35 -0
  13. data/lib/handler.rb +1 -0
  14. data/lib/helpers.rb +173 -4
  15. data/lib/livetext.rb +16 -141
  16. data/lib/parser/file.rb +8 -0
  17. data/lib/parser/import.rb +1 -3
  18. data/lib/parser/mixin.rb +22 -24
  19. data/lib/parser/set.rb +35 -26
  20. data/lib/parser/string.rb +19 -4
  21. data/lib/processor.rb +1 -4
  22. data/lib/standard.rb +68 -97
  23. data/lib/userapi.rb +1 -0
  24. data/livetext.gemspec +2 -1
  25. data/plugin/bookish.rb +26 -22
  26. data/plugin/calibre.rb +1 -1
  27. data/plugin/livemagick.rb +10 -10
  28. data/plugin/markdown.rb +13 -11
  29. data/plugin/pyggish.rb +94 -84
  30. data/plugin/tutorial.rb +10 -5
  31. data/test/all.rb +0 -1
  32. data/test/snapshots/OMIT.txt +9 -8
  33. data/test/snapshots/clusion.txt +84 -0
  34. data/test/snapshots/error_inc_line_num/match-error.txt +1 -1
  35. data/test/snapshots/error_invalid_name/match-error.txt +1 -1
  36. data/test/snapshots/error_line_num/match-error.txt +1 -1
  37. data/test/snapshots/error_mismatched_end/match-error.txt +1 -1
  38. data/test/snapshots/error_missing_end/match-error.txt +1 -1
  39. data/test/snapshots/error_no_such_copy/match-error.txt +1 -1
  40. data/test/snapshots/error_no_such_copy/source.lt3 +0 -1
  41. data/test/snapshots/error_no_such_inc/match-error.txt +1 -1
  42. data/test/snapshots/icanhaz/expected-output.txt +5 -0
  43. data/test/snapshots/icanhaz/match-error.txt +1 -0
  44. data/test/snapshots/icanhaz/simple_import.rb +5 -0
  45. data/test/snapshots/icanhaz/source.lt3 +10 -0
  46. data/test/snapshots/icanhaz2/expected-error.txt +0 -0
  47. data/test/snapshots/icanhaz2/expected-output.txt +6 -0
  48. data/test/snapshots/icanhaz2/simple_canhaz.rb +5 -0
  49. data/test/snapshots/icanhaz2/source.lt3 +6 -0
  50. data/test/snapshots/simple_import/expected-error.txt +0 -0
  51. data/test/snapshots/simple_import/expected-output.txt +7 -0
  52. data/test/snapshots/simple_import/simple_import.rb +5 -0
  53. data/test/snapshots/simple_import/source.lt3 +7 -0
  54. data/test/snapshots/simple_include/source.lt3 +0 -1
  55. data/test/snapshots.rb +85 -37
  56. data/test/unit/all.rb +1 -0
  57. data/test/unit/formatline.rb +650 -0
  58. data/test/unit/parser/importable.rb +1 -1
  59. data/test/unit/parser/mixin.rb +1 -1
  60. data/test/unit/parser/set.rb +19 -12
  61. data/test/unit/parser/string.rb +14 -14
  62. metadata +27 -5
  63. data/test/formatting-tests.rb +0 -35
  64. data/test/formatting.rb +0 -103
  65. data/test/snapshots/formatting-tests.txt +0 -124
@@ -0,0 +1,35 @@
1
+
2
+ class Livetext::Handler::ICanHaz
3
+ include Helpers
4
+
5
+ attr_reader :file
6
+
7
+ def initialize(name)
8
+ @name = name
9
+ @file = find_file(name)
10
+ end
11
+
12
+ def self.get_module(name)
13
+ handler = self.new(name)
14
+ const1 = Object.constants
15
+ @file = handler.file.sub(/.rb$/, "")
16
+ require @file # + ".rb"
17
+ const2 = Object.constants
18
+ names = (const2 - const1)
19
+ abort "Expected ONE new constant: #{names.inspect}" if names.size != 1
20
+ modname = names.first.to_s
21
+ newmod = Object.const_get("::" + modname)
22
+ newmod # return actual module
23
+ end
24
+
25
+ private
26
+
27
+ def cwd_root?
28
+ File.dirname(File.expand_path(".")) == "/"
29
+ end
30
+
31
+ def fname2module(name)
32
+ end
33
+
34
+ end
35
+
data/lib/handler.rb ADDED
@@ -0,0 +1 @@
1
+ require_relative 'handler/icanhaz'
data/lib/helpers.rb CHANGED
@@ -1,10 +1,117 @@
1
1
 
2
2
  module Helpers
3
3
 
4
+ Space = " "
5
+ Sigil = "." # Can't change yet
6
+
7
+ def escape_html(string)
8
+ enc = string.encoding
9
+ unless enc.ascii_compatible?
10
+ if enc.dummy?
11
+ origenc = enc
12
+ enc = Encoding::Converter.asciicompat_encoding(enc)
13
+ string = enc ? string.encode(enc) : string.b
14
+ end
15
+ table = Hash[TABLE_FOR_ESCAPE_HTML__.map {|pair|pair.map {|s|s.encode(enc)}}]
16
+ string = string.gsub(/#{"['&\"<>]".encode(enc)}/, table)
17
+ string.encode!(origenc) if origenc
18
+ return string
19
+ end
20
+ string.gsub(/['&\"<>]/, TABLE_FOR_ESCAPE_HTML__)
21
+ end
22
+
23
+ def find_file(name, ext=".rb")
24
+ paths = [Livetext::Path.sub(/lib/, "imports/"), "./"]
25
+ base = "#{name}#{ext}"
26
+ paths.each do |path|
27
+ file = path + base
28
+ return file if File.exist?(file)
29
+ end
30
+
31
+ raise "No such mixin '#{name}'"
32
+
33
+ # # Really want to search upward??
34
+ # raise "No such mixin '#{name}'" if cwd_root?
35
+ # Dir.chdir("..") { find_file(name) }
36
+ end
37
+
38
+ def self.rx(str, space=nil)
39
+ Regexp.compile("^" + Regexp.escape(str) + "#{space}")
40
+ end
41
+
42
+ Comment = rx(Sigil, Space)
43
+ Dotcmd = rx(Sigil)
44
+ Ddotcmd = /^ *\$\.[A-Za-z]/
45
+
46
+ ## FIXME process_file[!] should call process[_text]
47
+
48
+ def process_file(fname, btrace=false)
49
+ setfile(fname)
50
+ text = File.readlines(fname)
51
+ enum = text.each
52
+ @backtrace = btrace
53
+ @main.source(enum, fname, 0)
54
+ line = nil
55
+ loop do
56
+ line = @main.nextline
57
+ break if line.nil?
58
+ process_line(line)
59
+ end
60
+ val = @main.finalize if @main.respond_to? :finalize
61
+ @body
62
+ end
63
+
64
+ def process_line(line) # FIXME inefficient?
65
+ nomarkup = true
66
+ case line # must apply these in order
67
+ when Comment
68
+ handle_scomment(line)
69
+ when Dotcmd
70
+ handle_dotcmd(line)
71
+ when Ddotcmd
72
+ indent = line.index("$") + 1
73
+ @indentation.push(indent)
74
+ line.sub!(/^ *\$/, "")
75
+ handle_dotcmd(line)
76
+ indentation.pop
77
+ else
78
+ @main._passthru(line)
79
+ end
80
+ end
81
+
82
+ def handle_dotcmd(line, indent = 0)
83
+ indent = @indentation.last # top of stack
84
+ line = line.sub(/# .*$/, "")
85
+ name = get_name(line).to_sym
86
+ result = nil
87
+ case
88
+ when name == :end # special case
89
+ puts @body
90
+ raise EndWithoutOpening()
91
+ when @main.respond_to?(name)
92
+ result = @main.send(name)
93
+ else
94
+ puts @body # earlier correct output, not flushed yet
95
+ raise "Name '#{name}' is unknown"
96
+ return
97
+ end
98
+ result
99
+ end
100
+
101
+ def handle_scomment(line)
102
+ end
103
+
104
+ def get_name(line)
105
+ name, data = line.split(" ", 2)
106
+ name = name[1..-1] # chop off sigil
107
+ name = "dot_" + name if %w[include def].include?(name)
108
+ @main.data = data
109
+ @main.check_disallowed(name)
110
+ name
111
+ end
112
+
4
113
  def check_disallowed(name)
5
- # raise "Illegal name '#{name}'" if _disallowed?(name)
6
- # FIXME use custom exception
7
- raise DisallowedName, name if _disallowed?(name)
114
+ raise DisallowedName(name) if disallowed?(name)
8
115
  end
9
116
 
10
117
  def check_file_exists(file)
@@ -14,7 +121,7 @@ module Helpers
14
121
  def set_variables(pairs)
15
122
  pairs.each do |pair|
16
123
  var, value = *pair
17
- @parent._setvar(var, value)
124
+ @parent.setvar(var, value)
18
125
  end
19
126
  end
20
127
 
@@ -22,4 +129,66 @@ module Helpers
22
129
  File.read(fname)
23
130
  end
24
131
 
132
+ def search_upward(file)
133
+ value = nil
134
+ return file if File.exist?(file)
135
+
136
+ count = 1
137
+ loop do
138
+ front = "../" * count
139
+ count += 1
140
+ here = Pathname.new(front).expand_path.dirname.to_s
141
+ break if here == "/"
142
+ path = front + file
143
+ value = path if File.exist?(path)
144
+ break if value
145
+ end
146
+ STDERR.puts "Cannot find #{file.inspect} from #{Dir.pwd}" unless value
147
+ return value
148
+ rescue
149
+ STDERR.puts "Can't find #{file.inspect} from #{Dir.pwd}"
150
+ return nil
151
+ end
152
+
153
+ def include_file(file)
154
+ @_args = [file]
155
+ dot_include
156
+ end
157
+
158
+ def onoff(arg) # helper
159
+ arg ||= "on"
160
+ raise ExpectedOnOff unless String === arg
161
+ case arg.downcase
162
+ when "on"
163
+ return true
164
+ when "off"
165
+ return false
166
+ else
167
+ raise ExpectedOnOff
168
+ end
169
+ end
170
+
171
+ def setvar(var, val)
172
+ str, sym = var.to_s, var.to_sym
173
+ Livetext::Vars[str] = val
174
+ Livetext::Vars[sym] = val
175
+ @_vars[str] = val
176
+ @_vars[sym] = val
177
+ end
178
+
179
+ def setfile(file)
180
+ if file
181
+ setvar(:File, file)
182
+ dir = File.dirname(File.expand_path(file))
183
+ setvar(:FileDir, dir)
184
+ else
185
+ setvar(:File, "[no file]")
186
+ setvar(:FileDir, "[no dir]")
187
+ end
188
+ end
189
+
190
+ def setfile!(file) # FIXME why does this variant exist?
191
+ setvar(:File, file)
192
+ end
193
+
25
194
  end
data/lib/livetext.rb CHANGED
@@ -1,23 +1,28 @@
1
1
  # Class Livetext skeleton (top level).
2
2
 
3
3
  class Livetext
4
- VERSION = "0.9.14"
4
+ VERSION = "0.9.20"
5
5
  Path = File.expand_path(File.join(File.dirname(__FILE__)))
6
+
7
+ module Handler
8
+ end
6
9
  end
7
10
 
8
11
  # $LOAD_PATH << Livetext::Path
9
12
 
10
13
  require 'fileutils'
11
14
 
12
- require 'helpers'
13
15
  require_relative 'errors'
14
16
  require_relative 'functions'
15
17
  require_relative 'userapi'
16
18
  require_relative 'standard'
17
19
  require_relative 'formatline'
18
20
  require_relative 'processor'
21
+ require_relative 'helpers'
22
+ require_relative 'handler'
19
23
 
20
24
  Plugins = File.expand_path(File.join(File.dirname(__FILE__), "../plugin"))
25
+ Imports = File.expand_path(File.join(File.dirname(__FILE__), "../imports"))
21
26
 
22
27
  TTY = ::File.open("/dev/tty", "w")
23
28
 
@@ -31,16 +36,6 @@ class Livetext
31
36
 
32
37
  Vars = {}
33
38
 
34
- Space = " "
35
- Sigil = "." # Can't change yet
36
-
37
- def self.rx(str, space=nil)
38
- Regexp.compile("^" + Regexp.escape(str) + "#{space}")
39
- end
40
-
41
- Comment = rx(Sigil, Livetext::Space)
42
- Dotcmd = rx(Sigil)
43
- Ddotcmd = /^ *\$\.[A-Za-z]/
44
39
 
45
40
  attr_reader :main
46
41
  attr_accessor :no_puts
@@ -60,16 +55,17 @@ class Livetext
60
55
  call = Array(call)
61
56
  mix.each {|lib| obj.mixin(lib) }
62
57
  call.each {|cmd| obj.main.send(cmd[1..-1]) } # ignores leading dot, no param
63
- vars.each_pair {|var, val| obj._setvar(var, val.to_s) }
58
+ vars.each_pair {|var, val| obj.setvar(var, val.to_s) }
64
59
  obj
65
60
  end
66
61
 
67
62
  def customize(mix: [], call: [], vars: {})
68
63
  mix = Array(mix)
69
64
  call = Array(call)
65
+ # FIXME HF won't this break??
70
66
  mix.each {|lib| mixin(lib) }
71
67
  call.each {|cmd| @main.send(cmd[1..-1]) } # ignores leading dot, no param
72
- vars.each_pair {|var, val| _setvar(var, val.to_s) }
68
+ vars.each_pair {|var, val| setvar(var, val.to_s) }
73
69
  self
74
70
  end
75
71
 
@@ -82,81 +78,17 @@ class Livetext
82
78
  @main = Processor.new(self, output)
83
79
  @indentation = [0]
84
80
  @_vars = Livetext::Vars
85
-
86
- # Other predefined variables (see also _setfile)
87
- _setvar(:User, `whoami`.chomp)
88
- _setvar(:Version, Livetext::VERSION)
89
- end
90
-
91
- def _parse_colon_args(args, hash) # really belongs in livetext
92
- h2 = hash.dup
93
- e = args.each
94
- loop do
95
- arg = e.next.chop.to_sym
96
- raise "_parse_args: #{arg} is unknown" unless hash.keys.include?(arg)
97
- h2[arg] = e.next
98
- end
99
- h2 = h2.reject {|k,v| v.nil? }
100
- h2.each_pair {|k, v| raise "#{k} has no value" if v.empty? }
101
- h2
102
- end
103
-
104
- def _get_arg(name, args) # really belongs in livetext
105
- raise "(#{name}) Expected an array" unless args.is_a? Array
106
- raise "(#{name}) Expected an arg" if args.empty?
107
- raise "(#{name}) Too many args: #{args.inspect}" if args.size > 1
108
- val = args[0]
109
- raise "Expected an argument '#{name}'" if val.nil?
110
- val
81
+ initial_vars
111
82
  end
112
83
 
113
- def mixin(mod)
114
- @main._mixin(mod)
115
- end
116
-
117
- def _setvar(var, val)
118
- str, sym = var.to_s, var.to_sym
119
- Livetext::Vars[str] = val
120
- Livetext::Vars[sym] = val
121
- @_vars[str] = val
122
- @_vars[sym] = val
123
- end
124
-
125
- def _setfile(file)
126
- if file
127
- _setvar(:File, file)
128
- dir = File.dirname(File.expand_path(file))
129
- _setvar(:FileDir, dir)
130
- else
131
- _setvar(:File, "[no file]")
132
- _setvar(:FileDir, "[no dir]")
133
- end
134
- end
135
-
136
- def _setfile!(file)
137
- _setvar(:File, file)
138
- end
139
-
140
- def process_line(line) # FIXME inefficient?
141
- nomarkup = true
142
- case line # must apply these in order
143
- when Comment
144
- handle_scomment(line)
145
- when Dotcmd
146
- handle_dotcmd(line)
147
- when Ddotcmd
148
- indent = line.index("$") + 1
149
- @indentation.push(indent)
150
- line.sub!(/^ *\$/, "")
151
- handle_dotcmd(line)
152
- indentation.pop
153
- else
154
- @main._passthru(line)
155
- end
84
+ def initial_vars
85
+ # Other predefined variables (see also setfile)
86
+ setvar(:User, `whoami`.chomp)
87
+ setvar(:Version, Livetext::VERSION)
156
88
  end
157
89
 
158
90
  def transform(text)
159
- _setfile!("(string)")
91
+ setfile!("(string)")
160
92
  enum = text.each_line
161
93
  front = text.match(/.*?\n/).to_a.first.chomp rescue ""
162
94
  @main.source(enum, "STDIN: '#{front}...'", 0)
@@ -193,62 +125,5 @@ class Livetext
193
125
  self.body
194
126
  end
195
127
 
196
- ## FIXME process_file[!] should call process[_text]
197
-
198
- def process_file(fname, btrace=false)
199
- _setfile(fname)
200
- text = File.readlines(fname)
201
- enum = text.each
202
- @backtrace = btrace
203
- @main.source(enum, fname, 0)
204
- line = nil
205
- loop do
206
- line = @main.nextline
207
- break if line.nil?
208
- process_line(line)
209
- end
210
- val = @main.finalize if @main.respond_to? :finalize
211
- @body
212
- rescue => err
213
- STDERR.puts "[process_file] fname = #{fname.inspect}\n line = #{line.inspect}"
214
- STDERR.puts "ERROR #{err} in process_file"
215
- err.backtrace.each {|x| STDERR.puts " " + x }
216
- # @body = ""
217
- end
218
-
219
- def handle_scomment(line)
220
- end
221
-
222
- def _get_name(line)
223
- name, data = line.split(" ", 2)
224
- name = name[1..-1] # chop off sigil
225
- name = "_" + name if %w[include def].include?(name)
226
- @main.data = data
227
- @main.check_disallowed(name)
228
- name
229
- end
230
-
231
- def handle_dotcmd(line, indent = 0)
232
- indent = @indentation.last # top of stack
233
- line = line.sub(/# .*$/, "")
234
- name = _get_name(line).to_sym
235
- result = nil
236
- case
237
- when name == :end # special case
238
- puts @body
239
- raise EndWithoutOpening()
240
- when @main.respond_to?(name)
241
- result = @main.send(name)
242
- else
243
- @main._error! "Name '#{name}' is unknown"
244
- return
245
- end
246
- result
247
- rescue => err
248
- puts @body # earlier correct output, not flushed yet
249
- STDERR.puts "Error: #{err.inspect}"
250
- STDERR.puts err.backtrace
251
- end
252
-
253
128
  end
254
129
 
@@ -0,0 +1,8 @@
1
+
2
+ require_relative '../livetext'
3
+
4
+ # Hmm. not today?
5
+
6
+ class Livetext::ParseFile
7
+
8
+ end
data/lib/parser/import.rb CHANGED
@@ -4,9 +4,7 @@ require '../livetext/importable'
4
4
  make_exception(:BadVariableName, "Error: invalid variable name")
5
5
  make_exception(:NoEqualSign, "Error: no equal sign found")
6
6
 
7
- # FIXME probably belongs elsewhere?
8
-
9
- class Livetext::ParseImport
7
+ class Livetext::Handler::Import
10
8
  def use_import(name)
11
9
  require name
12
10
  include name
data/lib/parser/mixin.rb CHANGED
@@ -1,40 +1,38 @@
1
1
  require_relative '../livetext'
2
+ require_relative '../helpers'
2
3
  require_relative 'string'
3
4
 
4
- make_exception(:BadVariableName, "Error: invalid variable name")
5
5
  make_exception(:NoEqualSign, "Error: no equal sign found")
6
6
 
7
- # FIXME probably belongs elsewhere?
8
-
9
- class Livetext::ParseMixin # < StringParser
10
-
7
+ class Livetext::ParseMixin
11
8
  include Helpers
12
9
 
13
- def cwd_root?
14
- File.dirname(File.expand_path(".")) == "/"
10
+ def initialize(name)
11
+ @name = name
12
+ @file = find_file(name)
15
13
  end
16
14
 
17
- def find_mixin(name)
18
- file = "#{Plugins}/" + name.downcase + ".rb"
19
- return file if File.exist?(file)
20
-
21
- file = "./#{name}.rb"
22
- return file if File.exist?(file)
15
+ def self.get_module(name)
16
+ parse = self.new(name)
17
+ modname, code = parse.read_mixin
18
+ eval(code) # Avoid in the future
19
+ newmod = Object.const_get("::" + modname)
20
+ # return actual module
21
+ newmod
22
+ end
23
23
 
24
- raise "No such mixin '#{name}'" if cwd_root?
25
- Dir.chdir("..") { find_mixin(name) }
24
+ def read_mixin
25
+ modname = @name.gsub("/","_").capitalize
26
+ meths = grab_file(@file)
27
+ [modname, "module ::#{modname}; #{meths}\nend"]
26
28
  end
27
29
 
28
- def use_mixin(name, file)
29
- modname = name.gsub("/","_").capitalize
30
- meths = grab_file(file)
31
- string = "module ::#{modname}; #{meths}\nend"
32
- eval(string)
33
- newmod = Object.const_get("::" + modname)
34
- self.extend(newmod)
35
- init = "init_#{name}"
36
- self.send(init) if self.respond_to? init
30
+ private
31
+
32
+ def cwd_root?
33
+ File.dirname(File.expand_path(".")) == "/"
37
34
  end
38
35
 
36
+
39
37
  end
40
38
 
data/lib/parser/set.rb CHANGED
@@ -17,22 +17,31 @@ class Livetext::ParseSet < StringParser
17
17
  super
18
18
  end
19
19
 
20
+ def wtf(note="")
21
+ TTY.puts "| PARSER: @i = #@i @len = #@len"
22
+ TTY.puts "|#{note}"
23
+ TTY.puts "| [" + @line.gsub(" ", "_") + "]"
24
+ TTY.print "| " # 0-based (one extra space)
25
+ @i.times { TTY.print "-" }
26
+ TTY.puts "^"
27
+ TTY.puts
28
+ end
29
+
20
30
  def parse
21
31
  pairs = []
22
-
32
+ char = nil
23
33
  loop do
24
- skip_spaces
25
- char = self.peek
26
34
  break if eos? # end of string
35
+ char = skip_spaces
27
36
  raise "Expected alpha to start var name" unless char =~ /[a-z]/i
28
37
  pairs << assignment
29
- skip_spaces
30
- char = self.peek
38
+ char = skip_spaces
31
39
  break if eos? # end of string
32
40
  case char
33
41
  when nil # end of string
34
42
  when ","
35
- self.next # skip comma
43
+ char = grab # skip comma
44
+ char = skip_spaces
36
45
  else
37
46
  raise "Expected comma or end of string (found #{char.inspect})"
38
47
  end
@@ -47,7 +56,7 @@ class Livetext::ParseSet < StringParser
47
56
  var = get_var
48
57
  skip_equal
49
58
  value = get_value
50
- value = FormatLine.var_func_parse(value)
59
+ value = FormatLine.var_func_parse(value) # FIXME broken now?
51
60
  pair = [var, value]
52
61
  pair
53
62
  end
@@ -55,11 +64,11 @@ class Livetext::ParseSet < StringParser
55
64
  def get_var
56
65
  name = ""
57
66
  loop do
58
- char = self.peek
67
+ char = peek
59
68
  break if eos? # end of string
60
69
  case char
61
70
  when /[a-zA-Z_\.0-9]/
62
- name << self.next
71
+ name << grab
63
72
  next
64
73
  when /[ =]/
65
74
  return name
@@ -73,11 +82,10 @@ class Livetext::ParseSet < StringParser
73
82
  def skip_equal
74
83
  found = false
75
84
  skip_spaces
76
- raise NoEqualSign unless self.peek == "="
85
+ raise NoEqualSign unless peek == "="
77
86
  found = true
78
- self.next # skip =... spaces too
79
- self.skip_spaces
80
- peek = self.peek
87
+ grab # skip =
88
+ skip_spaces # skip spaces too
81
89
  return peek # just for testing
82
90
  rescue StopIteration
83
91
  raise NoEqualSign unless found
@@ -85,40 +93,41 @@ class Livetext::ParseSet < StringParser
85
93
  end
86
94
 
87
95
  def escaped
88
- self.next # skip backslash
89
- self.next # return following char
96
+ grab # skip backslash
97
+ grab # return following char
90
98
  end
91
99
 
92
100
  make_exception(:BadQuotedString, "Bad quoted string: %1")
93
101
 
94
102
  def quoted_value
95
- quote = self.next # opening quote...
103
+ quote = grab # opening quote...
96
104
  value = ""
97
105
  char = nil
98
106
  loop do
99
- char = self.peek
100
- break if self.eos?
107
+ char = grab
108
+ break if eos?
101
109
  break if char == quote
110
+ # break if char.nil?
102
111
  char = escaped if char == "\\"
103
112
  value << char
104
- char = self.next
105
113
  end
106
114
  if char == quote
107
- char = self.next
115
+ # char = grab
108
116
  return value
109
117
  end
110
118
  raise BadQuotedString, quote + value
111
119
  end
112
120
 
113
121
  def unquoted_value
114
- # puts "#{__method__}: #{@line.inspect} i = #@i peek = #{self.peek.inspect}"
122
+ char = nil
115
123
  value = ""
116
124
  loop do
117
- char = self.peek
118
- break if self.eos?
125
+ char = peek
126
+ break if eos? # FIXME oops???
127
+ # break if char.nil?
119
128
  break if char == " " || char == ","
120
129
  value << char
121
- char = self.next
130
+ char = grab
122
131
  end
123
132
  value
124
133
  end
@@ -128,8 +137,8 @@ class Livetext::ParseSet < StringParser
128
137
  end
129
138
 
130
139
  def get_value
131
- char = self.peek
132
- value = quote?(char) ? quoted_value : unquoted_value
140
+ char = peek
141
+ value = quote?(char) ? quoted_value : unquoted_value
133
142
  value
134
143
  end
135
144
  end