Class: Oppen::PrintStack
- Inherits:
-
Object
- Object
- Oppen::PrintStack
- Defined in:
- lib/oppen/print_stack.rb
Overview
Class that represents a stack that builds an output string using the values of the tokens that were pushed into it.
Defined Under Namespace
Classes: PrintStackEntry
Instance Attribute Summary collapse
-
#buffer ⇒ Object
readonly
IO element that builds the output.
-
#config ⇒ Object
readonly
Config containing customization flags.
-
#genspace ⇒ Object
readonly
Callable that generate spaces.
-
#items ⇒ Object
readonly
Array representing the stack of PrintStackEntries.
-
#new_line ⇒ Object
readonly
Delimiter between lines in output.
-
#space ⇒ Integer
readonly
Current available space (Called index in the original paper).
-
#width ⇒ Object
readonly
Maximum allowed width for printing (Called length in the original paper).
Instance Method Summary collapse
-
#erase(count = 0) ⇒ Nil
Erase the last
count
characters. -
#handle_begin(token, token_width) ⇒ Nil
Handle Begin Token.
-
#handle_break(token, token_width, trim_on_break: 0) ⇒ Nil
Handle Break Token.
-
#handle_end ⇒ Nil
Handle End Token.
-
#handle_string(token, token_width) ⇒ Nil
Handle String Token.
-
#indent(amount) ⇒ Nil
Add indentation by
amount
. -
#initialize(width, new_line, config, space, out) ⇒ PrintStack
constructor
A new instance of PrintStack.
-
#output ⇒ String
Returns the output of the print stack.
-
#pop ⇒ PrintStackEntry
Pop a PrintStackEntry from the stack.
-
#print(token, token_width, trim_on_break: 0) ⇒ Nil
Core method responsible for building the print stack and the output string.
-
#print_new_line(amount) ⇒ Nil
Add a new line to the output.
-
#push(print_stack_entry) ⇒ Nil
Push a PrintStackEntry into the stack.
-
#top ⇒ PrintStackEntry
Get the element at the top of the stack.
-
#write(obj) ⇒ Nil
Write a string to the output.
Constructor Details
#initialize(width, new_line, config, space, out) ⇒ PrintStack
Returns a new instance of PrintStack.
44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 |
# File 'lib/oppen/print_stack.rb', line 44 def initialize(width, new_line, config, space, out) @buffer = out @config = config @genspace = if space.respond_to?(:call) raise ArgumentError, 'space argument must be a Proc of arity 1' \ if space.to_proc.arity != 1 space else ->(n) { space * n } end @indent = 0 @items = [] @new_line = new_line @width = width @space = width end |
Instance Attribute Details
#buffer ⇒ Object (readonly)
IO element that builds the output.
22 23 24 |
# File 'lib/oppen/print_stack.rb', line 22 def buffer @buffer end |
#config ⇒ Object (readonly)
Config containing customization flags
25 26 27 |
# File 'lib/oppen/print_stack.rb', line 25 def config @config end |
#genspace ⇒ Object (readonly)
Callable that generate spaces
28 29 30 |
# File 'lib/oppen/print_stack.rb', line 28 def genspace @genspace end |
#items ⇒ Object (readonly)
Array representing the stack of PrintStackEntries.
31 32 33 |
# File 'lib/oppen/print_stack.rb', line 31 def items @items end |
#new_line ⇒ Object (readonly)
Delimiter between lines in output
34 35 36 |
# File 'lib/oppen/print_stack.rb', line 34 def new_line @new_line end |
#space ⇒ Integer (readonly)
Current available space (Called index in the original paper).
42 43 44 |
# File 'lib/oppen/print_stack.rb', line 42 def space @space end |
#width ⇒ Object (readonly)
Maximum allowed width for printing (Called length in the original paper).
37 38 39 |
# File 'lib/oppen/print_stack.rb', line 37 def width @width end |
Instance Method Details
#erase(count = 0) ⇒ Nil
Erase the last count
characters.
258 259 260 261 262 263 |
# File 'lib/oppen/print_stack.rb', line 258 def erase(count = 0) raise ArgumentError, "count = #{count} must be non-negative" if count.negative? buffer.seek(-count, IO::SEEK_CUR) @space += count end |
#handle_begin(token, token_width) ⇒ Nil
Handle Begin Token.
101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 |
# File 'lib/oppen/print_stack.rb', line 101 def handle_begin(token, token_width) if token_width > space type = if token.break_type == Token::BreakType::CONSISTENT Token::BreakType::CONSISTENT else Token::BreakType::INCONSISTENT end if config&.indent_anchor == Config::IndentAnchor::ON_BEGIN indent = token.offset if !items.empty? indent += top.offset end else indent = space - token.offset end push PrintStackEntry.new indent, type else push PrintStackEntry.new 0, Token::BreakType::FITS end end |
#handle_break(token, token_width, trim_on_break: 0) ⇒ Nil
Handle Break Token.
141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 |
# File 'lib/oppen/print_stack.rb', line 141 def handle_break(token, token_width, trim_on_break: 0) block = top case block.break_type in Token::BreakType::FITS @space -= token.width write token in Token::BreakType::CONSISTENT @space = block.offset - token.offset indent = if config&.indent_anchor == Config::IndentAnchor::ON_BEGIN token.offset else width - space end erase(trim_on_break) write token.line_continuation print_new_line indent in Token::BreakType::INCONSISTENT if token_width > space @space = block.offset - token.offset indent = if config&.indent_anchor == Config::IndentAnchor::ON_BEGIN token.offset else width - space end erase(trim_on_break) write token.line_continuation print_new_line indent else @space -= token.width write token end end end |
#handle_end ⇒ Nil
Handle End Token.
128 129 130 |
# File 'lib/oppen/print_stack.rb', line 128 def handle_end pop end |
#handle_string(token, token_width) ⇒ Nil
Handle String Token.
185 186 187 188 189 190 191 192 193 194 |
# File 'lib/oppen/print_stack.rb', line 185 def handle_string(token, token_width) return if token.value.empty? @space = [0, space - token_width].max if @indent.positive? indent @indent @indent = 0 end write token end |
#indent(amount) ⇒ Nil
Called Indent as well in the original paper.
Add indentation by amount
.
272 273 274 275 276 |
# File 'lib/oppen/print_stack.rb', line 272 def indent(amount) raise ArgumentError 'Indenting using negative amount' if amount.negative? write genspace.(amount) if amount.positive? end |
#output ⇒ String
Returns the output of the print stack
66 67 68 69 |
# File 'lib/oppen/print_stack.rb', line 66 def output buffer.truncate(buffer.pos) buffer.string end |
#pop ⇒ PrintStackEntry
Pop a PrintStackEntry from the stack.
208 209 210 211 212 213 214 |
# File 'lib/oppen/print_stack.rb', line 208 def pop if items.empty? raise 'Popping empty stack' end items.pop end |
#print(token, token_width, trim_on_break: 0) ⇒ Nil
Called Print in the original paper.
Core method responsible for building the print stack and the output string.
80 81 82 83 84 85 86 87 88 89 90 91 |
# File 'lib/oppen/print_stack.rb', line 80 def print(token, token_width, trim_on_break: 0) case token in Token::Begin handle_begin token, token_width in Token::End handle_end in Token::Break handle_break token, token_width, trim_on_break: in Token::String handle_string token, token_width end end |
#print_new_line(amount) ⇒ Nil
Called PrintNewLine as well in the original paper.
Add a new line to the output.
234 235 236 237 238 239 240 241 242 |
# File 'lib/oppen/print_stack.rb', line 234 def print_new_line(amount) write new_line if config&.indent_anchor == Config::IndentAnchor::ON_BEGIN @space = width - top.offset - amount @indent = width - space else @indent = amount end end |
#push(print_stack_entry) ⇒ Nil
Push a PrintStackEntry into the stack.
201 202 203 |
# File 'lib/oppen/print_stack.rb', line 201 def push(print_stack_entry) items.append(print_stack_entry) end |
#top ⇒ PrintStackEntry
Get the element at the top of the stack.
219 220 221 222 223 224 225 |
# File 'lib/oppen/print_stack.rb', line 219 def top if items.empty? raise 'Accessing empty stack' end items.last end |
#write(obj) ⇒ Nil
Write a string to the output.
249 250 251 |
# File 'lib/oppen/print_stack.rb', line 249 def write(obj) buffer.write(obj.to_s) end |