Class: FlameChannelParser::Parser
- Inherits:
-
Object
- Object
- FlameChannelParser::Parser
- Defined in:
- lib/parser.rb
Overview
Comnplete parser class for all Flame/Smoke versions
Direct Known Subclasses
Constant Summary collapse
- BASE_VALUE_MATCHER =
/Value ([\-\d\.]+)/
- KF_COUNT_MATCHER =
/Size (\d+)/
- EXTRAP_MATCHER =
/Extrapolation (\w+)/
- CHANNEL_MATCHER =
/Channel (.+)\n/
- NODE_TYPE_MATCHER =
/Node (\w+)/
- NODE_NAME_MATCHER =
/Name (\w+)/
- LITERALS =
%w( linear constant natural hermite cubic bezier cycle revcycle )
Instance Attribute Summary collapse
-
#logger_proc ⇒ Object
Here you can assign a logger proc or a lambda that will be call’ed with progress reports.
Instance Method Summary collapse
-
#channel_is_useful?(channel_name) ⇒ Boolean
Override this method to skip some channels, this will speedup your code alot.
- #extract_key_from(io) ⇒ Object
-
#key_matchers ⇒ Object
Defines a number of regular expression matchers applied to the file as it is being parsed.
-
#log(message) ⇒ Object
This method will be called internally with information on items being processed.
-
#parse(io) ⇒ Object
Parses the setup passed in the IO.
- #parse_channel(io, channel_name, node_type, node_name) ⇒ Object
- #symbolize_literal(v) ⇒ Object
Instance Attribute Details
#logger_proc ⇒ Object
Here you can assign a logger proc or a lambda that will be call’ed with progress reports
6 7 8 |
# File 'lib/parser.rb', line 6 def logger_proc @logger_proc end |
Instance Method Details
#channel_is_useful?(channel_name) ⇒ Boolean
Override this method to skip some channels, this will speedup your code alot
44 45 46 |
# File 'lib/parser.rb', line 44 def channel_is_useful?(channel_name) true end |
#extract_key_from(io) ⇒ Object
111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 |
# File 'lib/parser.rb', line 111 def extract_key_from(io) frame = nil end_matcher = /End/ key = Key.new until io.eof? line = io.gets if line =~ end_matcher return key else key_matchers.each do | property, cast_method, pattern | if line =~ pattern v = symbolize_literal($1.send(cast_method)) key.send("#{property}=", v) end end end end raise "Did not detect any keyframes!" end |
#key_matchers ⇒ Object
Defines a number of regular expression matchers applied to the file as it is being parsed
49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 |
# File 'lib/parser.rb', line 49 def key_matchers #:nodoc: [ # Previously: [:frame, :to_f, /Frame ([\-\d\.]+)/], [:value, :to_f, /Value ([\-\d\.]+)/], [:left_slope, :to_f, /LeftSlope ([\-\d\.]+)/], [:right_slope, :to_f, /RightSlope ([\-\d\.]+)/], [:interpolation, :to_s, /Interpolation (\w+)/], [:break_slope, :to_s, /BreakSlope (\w+)/], # 2012 intoroduces: [:r_handle_x, :to_f, /RHandleX ([\-\d\.]+)/], [:l_handle_x, :to_f, /LHandleX ([\-\d\.]+)/], [:r_handle_y, :to_f, /RHandleY ([\-\d\.]+)/], [:l_handle_y, :to_f, /LHandleY ([\-\d\.]+)/], [:curve_mode, :to_s, /CurveMode (\w+)/], [:curve_order, :to_s, /CurveOrder (\w+)/], ] end |
#log(message) ⇒ Object
This method will be called internally with information on items being processed. The implementation just calls the logger_proc instance variable
38 39 40 |
# File 'lib/parser.rb', line 38 def log() @logger_proc.call() if @do_logs end |
#parse(io) ⇒ Object
Parses the setup passed in the IO. If a block is given to the method it will yield Channel objects one by one instead of accumulating them into an array (useful for big setups)
10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 |
# File 'lib/parser.rb', line 10 def parse(io) @do_logs = (@logger_proc.respond_to?(:call)) channels = [] node_name, node_type = nil, nil until io.eof? line = io.gets if line =~ NODE_TYPE_MATCHER node_type = $1 elsif line =~ NODE_NAME_MATCHER node_name = $1 elsif line =~ CHANNEL_MATCHER && channel_is_useful?($1) log("Parsing channel #{$1.inspect}") channel = parse_channel(io, $1, node_type, node_name) if block_given? yield(channel) else channels << channel end end end return channels unless block_given? end |
#parse_channel(io, channel_name, node_type, node_name) ⇒ Object
81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 |
# File 'lib/parser.rb', line 81 def parse_channel(io, channel_name, node_type, node_name) c = Channel.new(channel_name, node_type, node_name) indent, end_mark = nil, "ENDMARK" while line = io.gets unless indent indent = line.scan(/^(\s+)/)[1] end_mark = "#{indent}End" end if line =~ KF_COUNT_MATCHER num_keyframes = $1.to_i num_keyframes.times do | idx | log("Extracting keyframe %d of %d" % [idx + 1, num_keyframes]) c.push(extract_key_from(io)) end elsif line =~ BASE_VALUE_MATCHER# && empty? c.base_value = $1.to_f elsif line =~ EXTRAP_MATCHER c.extrapolation = symbolize_literal($1) elsif line.strip == end_mark break end end return c end |
#symbolize_literal(v) ⇒ Object
132 133 134 |
# File 'lib/parser.rb', line 132 def symbolize_literal(v) LITERALS.include?(v) ? v.to_sym : v end |