openstax_kitchen 3.2.0 → 6.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.github/workflows/changelog.yml +27 -0
- data/.github/workflows/rubocop.yml +28 -0
- data/.gitignore +1 -1
- data/CHANGELOG.md +94 -1
- data/Gemfile.lock +26 -18
- data/README.md +16 -0
- data/codecov.yaml +1 -0
- data/docker/ci +0 -1
- data/docker/rubocop +22 -0
- data/lib/kitchen/book_document.rb +1 -1
- data/lib/kitchen/book_element.rb +16 -2
- data/lib/kitchen/chapter_element.rb +10 -13
- data/lib/kitchen/chapter_element_enumerator.rb +1 -1
- data/lib/kitchen/composite_chapter_element.rb +7 -11
- data/lib/kitchen/composite_chapter_element_enumerator.rb +21 -0
- data/lib/kitchen/composite_page_element.rb +15 -10
- data/lib/kitchen/composite_page_element_enumerator.rb +1 -1
- data/lib/kitchen/config.rb +14 -0
- data/lib/kitchen/directions/bake_appendix.rb +3 -1
- data/lib/kitchen/directions/bake_chapter_glossary/main.rb +18 -0
- data/lib/kitchen/directions/bake_chapter_glossary/v1.rb +30 -0
- data/lib/kitchen/directions/bake_chapter_introductions.rb +23 -16
- data/lib/kitchen/directions/bake_chapter_introductions/chapter_introduction.xhtml.erb +0 -0
- data/lib/kitchen/directions/bake_chapter_key_concepts/main.rb +7 -2
- data/lib/kitchen/directions/bake_chapter_key_concepts/v1.rb +12 -7
- data/lib/kitchen/directions/bake_chapter_key_equations.rb +26 -21
- data/lib/kitchen/directions/bake_chapter_references/main.rb +16 -0
- data/lib/kitchen/directions/bake_chapter_references/v1.rb +35 -0
- data/lib/kitchen/directions/bake_chapter_section_exercises/main.rb +2 -2
- data/lib/kitchen/directions/bake_chapter_section_exercises/v1.rb +2 -1
- data/lib/kitchen/directions/bake_chapter_solutions/main.rb +11 -0
- data/lib/kitchen/directions/bake_chapter_solutions/v1.rb +37 -0
- data/lib/kitchen/directions/bake_chapter_summary.rb +56 -43
- data/lib/kitchen/directions/bake_composite_chapters.rb +1 -1
- data/lib/kitchen/directions/bake_composite_pages.rb +1 -1
- data/lib/kitchen/directions/bake_equations.rb +2 -2
- data/lib/kitchen/directions/bake_example.rb +8 -1
- data/lib/kitchen/directions/bake_figure.rb +14 -1
- data/lib/kitchen/directions/bake_first_elements.rb +22 -0
- data/lib/kitchen/directions/bake_footnotes/main.rb +2 -2
- data/lib/kitchen/directions/bake_footnotes/v1.rb +13 -9
- data/lib/kitchen/directions/bake_free_response/free_response.xhtml.erb +10 -0
- data/lib/kitchen/directions/{bake_chapter_review → bake_free_response}/main.rb +3 -3
- data/lib/kitchen/directions/bake_free_response/v1.rb +29 -0
- data/lib/kitchen/directions/bake_further_research.rb +61 -0
- data/lib/kitchen/directions/bake_index/v1.rb +36 -26
- data/lib/kitchen/directions/bake_link_placeholders.rb +1 -1
- data/lib/kitchen/directions/bake_notes/bake_note_subtitle.rb +4 -0
- data/lib/kitchen/directions/bake_notes/bake_numbered_notes/main.rb +43 -0
- data/lib/kitchen/directions/bake_notes/bake_numbered_notes/v1.rb +37 -0
- data/lib/kitchen/directions/bake_notes/bake_numbered_notes/v2.rb +25 -0
- data/lib/kitchen/directions/bake_notes/bake_numbered_notes/v3.rb +32 -0
- data/lib/kitchen/directions/bake_numbered_exercise/main.rb +7 -2
- data/lib/kitchen/directions/bake_numbered_exercise/v1.rb +34 -12
- data/lib/kitchen/directions/bake_numbered_table/bake_table_body.rb +29 -0
- data/lib/kitchen/directions/bake_numbered_table/main.rb +4 -0
- data/lib/kitchen/directions/bake_numbered_table/v1.rb +1 -24
- data/lib/kitchen/directions/bake_numbered_table/v2.rb +31 -0
- data/lib/kitchen/directions/bake_page_abstracts.rb +1 -1
- data/lib/kitchen/directions/bake_preface/main.rb +2 -2
- data/lib/kitchen/directions/bake_preface/v1.rb +3 -2
- data/lib/kitchen/directions/bake_references/main.rb +16 -0
- data/lib/kitchen/directions/bake_references/v1.rb +48 -0
- data/lib/kitchen/directions/bake_suggested_reading.rb +5 -0
- data/lib/kitchen/directions/bake_toc.rb +4 -2
- data/lib/kitchen/directions/book_answer_key_container/eob_answer_key_outer_container.xhtml.erb +9 -0
- data/lib/kitchen/directions/book_answer_key_container/main.rb +11 -0
- data/lib/kitchen/directions/book_answer_key_container/v1.rb +14 -0
- data/lib/kitchen/directions/chapter_review_container/chapter_review.xhtml.erb +9 -0
- data/lib/kitchen/directions/chapter_review_container/main.rb +11 -0
- data/lib/kitchen/directions/chapter_review_container/v1.rb +15 -0
- data/lib/kitchen/directions/eoc_section_title_link_snippet.rb +14 -1
- data/lib/kitchen/directions/move_exercises_to_eoc/main.rb +37 -0
- data/lib/kitchen/directions/{bake_chapter_review_exercises → move_exercises_to_eoc}/v1.rb +8 -10
- data/lib/kitchen/directions/{bake_chapter_review_exercises → move_exercises_to_eoc}/v2.rb +8 -9
- data/lib/kitchen/directions/move_exercises_to_eoc/v3.rb +49 -0
- data/lib/kitchen/directions/{bake_chapter_answer_key → move_solutions_to_answer_key}/main.rb +7 -3
- data/lib/kitchen/directions/{bake_chapter_answer_key → move_solutions_to_answer_key}/strategies/calculus.rb +1 -1
- data/lib/kitchen/directions/move_solutions_to_answer_key/strategies/default.rb +27 -0
- data/lib/kitchen/directions/move_solutions_to_answer_key/strategies/precalculus.rb +84 -0
- data/lib/kitchen/directions/{bake_chapter_answer_key → move_solutions_to_answer_key}/strategies/uphysics.rb +7 -5
- data/lib/kitchen/directions/{bake_chapter_answer_key → move_solutions_to_answer_key}/v1.rb +12 -6
- data/lib/kitchen/document.rb +20 -42
- data/lib/kitchen/element.rb +9 -3
- data/lib/kitchen/element_base.rb +108 -21
- data/lib/kitchen/element_enumerator_base.rb +33 -2
- data/lib/kitchen/element_enumerator_factory.rb +28 -12
- data/lib/kitchen/element_factory.rb +3 -3
- data/lib/kitchen/example_element.rb +8 -11
- data/lib/kitchen/example_element_enumerator.rb +1 -1
- data/lib/kitchen/exercise_element.rb +7 -10
- data/lib/kitchen/exercise_element_enumerator.rb +1 -1
- data/lib/kitchen/figure_element.rb +8 -11
- data/lib/kitchen/figure_element_enumerator.rb +1 -1
- data/lib/kitchen/id_tracker.rb +68 -0
- data/lib/kitchen/metadata_element.rb +8 -2
- data/lib/kitchen/metadata_element_enumerator.rb +1 -1
- data/lib/kitchen/note_element.rb +8 -11
- data/lib/kitchen/note_element_enumerator.rb +1 -1
- data/lib/kitchen/oven.rb +5 -1
- data/lib/kitchen/page_element.rb +27 -12
- data/lib/kitchen/page_element_enumerator.rb +1 -1
- data/lib/kitchen/patches/i18n.rb +34 -0
- data/lib/kitchen/patches/integer.rb +24 -0
- data/lib/kitchen/patches/nokogiri.rb +62 -0
- data/lib/kitchen/patches/nokogiri_profiling.rb +60 -0
- data/lib/kitchen/reference_element.rb +27 -0
- data/lib/kitchen/references_element_enumerator.rb +20 -0
- data/lib/kitchen/search_query.rb +31 -3
- data/lib/kitchen/selector.rb +25 -0
- data/lib/kitchen/selectors/base.rb +39 -0
- data/lib/kitchen/selectors/standard_1.rb +13 -0
- data/lib/kitchen/table_element.rb +8 -11
- data/lib/kitchen/table_element_enumerator.rb +1 -1
- data/lib/kitchen/templates/eob_section_title_template.xhtml.erb +10 -0
- data/lib/kitchen/templates/eoc_section_title_template.xhtml.erb +10 -0
- data/lib/kitchen/term_element.rb +5 -8
- data/lib/kitchen/term_element_enumerator.rb +1 -1
- data/lib/kitchen/unit_element.rb +13 -7
- data/lib/kitchen/unit_element_enumerator.rb +1 -1
- data/lib/kitchen/version.rb +1 -1
- data/lib/locales/en.yml +5 -1
- data/lib/locales/es.yml +33 -0
- data/lib/locales/pl.yml +3 -2
- data/lib/openstax_kitchen.rb +2 -5
- data/openstax_kitchen.gemspec +1 -0
- metadata +66 -25
- data/.github/config.yml +0 -14
- data/lib/kitchen/directions/bake_book_answer_key/eob_solutions_container.xhtml.erb +0 -9
- data/lib/kitchen/directions/bake_book_answer_key/main.rb +0 -11
- data/lib/kitchen/directions/bake_book_answer_key/v1.rb +0 -13
- data/lib/kitchen/directions/bake_chapter_glossary.rb +0 -39
- data/lib/kitchen/directions/bake_chapter_key_concepts/key_concepts.xhtml.erb +0 -16
- data/lib/kitchen/directions/bake_chapter_review/chapter_review.xhtml.erb +0 -9
- data/lib/kitchen/directions/bake_chapter_review/v1.rb +0 -13
- data/lib/kitchen/directions/bake_chapter_review_exercises/main.rb +0 -15
- data/lib/kitchen/directions/bake_chapter_review_exercises/review_exercises.xhtml.erb +0 -10
- data/lib/kitchen/directions/bake_exercises/main.rb +0 -12
- data/lib/kitchen/directions/bake_exercises/v1.rb +0 -169
- data/lib/kitchen/directions/bake_notes/bake_notes.rb +0 -48
- data/lib/kitchen/directions/bake_notes/bake_numbered_notes.rb +0 -63
- data/lib/kitchen/directions/bake_problem_first_elements.rb +0 -16
- data/lib/kitchen/transliterations.rb +0 -21
@@ -1,6 +1,6 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module Kitchen::Directions::
|
3
|
+
module Kitchen::Directions::MoveSolutionsToAnswerKey
|
4
4
|
module Strategies
|
5
5
|
class UPhysics
|
6
6
|
def bake(chapter:, append_to:)
|
@@ -18,7 +18,7 @@ module Kitchen::Directions::BakeChapterAnswerKey
|
|
18
18
|
def bake_section(chapter:, append_to:, klass:)
|
19
19
|
section_solutions_set = []
|
20
20
|
chapter.search(".#{klass}").each do |section|
|
21
|
-
section.search('[data-type="solution"]').each do |solution|
|
21
|
+
section.search('div[data-type="solution"]').each do |solution|
|
22
22
|
section_solutions_set.push(solution.cut)
|
23
23
|
end
|
24
24
|
end
|
@@ -31,9 +31,11 @@ module Kitchen::Directions::BakeChapterAnswerKey
|
|
31
31
|
|
32
32
|
def bake_from_notes(chapter:, append_to:, klass:)
|
33
33
|
solutions = []
|
34
|
-
chapter.notes("
|
35
|
-
|
36
|
-
|
34
|
+
chapter.notes("$.#{klass}").each do |note|
|
35
|
+
note.exercises.each do |exercise|
|
36
|
+
solution = exercise.solution
|
37
|
+
solutions.push(solution.cut) if solution
|
38
|
+
end
|
37
39
|
end
|
38
40
|
return if solutions.empty?
|
39
41
|
|
@@ -1,21 +1,27 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
module Kitchen::Directions::
|
3
|
+
module Kitchen::Directions::MoveSolutionsToAnswerKey
|
4
4
|
class V1
|
5
|
-
def bake(chapter:, metadata_source:, strategy:, append_to:)
|
5
|
+
def bake(chapter:, metadata_source:, strategy:, append_to:, strategy_options: {}, solutions_plural: true)
|
6
6
|
strategy =
|
7
7
|
case strategy
|
8
8
|
when :calculus
|
9
|
-
Strategies::Calculus
|
9
|
+
Strategies::Calculus.new
|
10
10
|
when :uphysics
|
11
|
-
Strategies::UPhysics
|
11
|
+
Strategies::UPhysics.new
|
12
|
+
when :precalculus
|
13
|
+
Strategies::Precalculus.new
|
14
|
+
when :default
|
15
|
+
Strategies::Default.new(strategy_options)
|
12
16
|
else
|
13
17
|
raise 'No such strategy'
|
14
18
|
end
|
15
19
|
|
20
|
+
solutions_or_solution = solutions_plural ? 'solutions' : 'solution'
|
16
21
|
append_to.append(child:
|
17
22
|
<<~HTML
|
18
|
-
<div class="os-eob os-
|
23
|
+
<div class="os-eob os-#{solutions_or_solution}-container" data-type="composite-page" \
|
24
|
+
data-uuid-key=".#{solutions_or_solution}#{chapter.count_in(:book)}">
|
19
25
|
<h2 data-type="document-title">
|
20
26
|
<span class="os-text">#{I18n.t(:chapter)} #{chapter.count_in(:book)}</span>
|
21
27
|
</h2>
|
@@ -26,7 +32,7 @@ module Kitchen::Directions::BakeChapterAnswerKey
|
|
26
32
|
</div>
|
27
33
|
HTML
|
28
34
|
)
|
29
|
-
strategy.
|
35
|
+
strategy.bake(chapter: chapter, append_to: append_to.last_element)
|
30
36
|
end
|
31
37
|
end
|
32
38
|
end
|
data/lib/kitchen/document.rb
CHANGED
@@ -13,6 +13,8 @@ module Kitchen
|
|
13
13
|
attr_accessor :location
|
14
14
|
# @return [Config] the configuration used in this document
|
15
15
|
attr_reader :config
|
16
|
+
# @return [IdTracker] the counter for duplicate IDs
|
17
|
+
attr_reader :id_tracker
|
16
18
|
|
17
19
|
# @!method selectors
|
18
20
|
# The document's selectors
|
@@ -34,7 +36,10 @@ module Kitchen
|
|
34
36
|
# @!method to_html
|
35
37
|
# @see https://www.rubydoc.info/github/sparklemotion/nokogiri/Nokogiri/XML/Node#to_html-instance_method Nokogiri::XML::Node#to_html
|
36
38
|
# @return [String] the document as an HTML string
|
37
|
-
|
39
|
+
# @!method encoding
|
40
|
+
# @see https://www.rubydoc.info/github/sparklemotion/nokogiri/Nokogiri/XML/Document#encoding-instance_method Nokogiri::XML::Document#encoding
|
41
|
+
# @return [String] the document as an HTML string
|
42
|
+
def_delegators :@nokogiri_document, :to_xhtml, :to_s, :to_xml, :to_html, :encoding
|
38
43
|
|
39
44
|
# Return a new instance of Document
|
40
45
|
#
|
@@ -44,8 +49,10 @@ module Kitchen
|
|
44
49
|
@nokogiri_document = nokogiri_document
|
45
50
|
@location = nil
|
46
51
|
@config = config || Config.new
|
47
|
-
@
|
48
|
-
|
52
|
+
@id_tracker = IdTracker.new
|
53
|
+
|
54
|
+
# Nokogiri by default only recognizes the namespaces on the root node. Add all others.
|
55
|
+
raw&.add_all_namespaces! if @config.enable_all_namespaces
|
49
56
|
end
|
50
57
|
|
51
58
|
# Returns an enumerator that iterates over all children of this document
|
@@ -145,38 +152,6 @@ module Kitchen
|
|
145
152
|
end
|
146
153
|
end
|
147
154
|
|
148
|
-
# Keeps track that an element with the given ID has been copied. When such
|
149
|
-
# elements are pasted, this information is used to give those elements unique
|
150
|
-
# IDs that don't duplicate the original element.
|
151
|
-
#
|
152
|
-
# @param id [String] the ID
|
153
|
-
#
|
154
|
-
def record_id_copied(id)
|
155
|
-
return if id.blank?
|
156
|
-
|
157
|
-
@next_paste_count_for_id[id] ||= 1
|
158
|
-
end
|
159
|
-
|
160
|
-
# Returns a unique ID given the ID of an element that was copied and is about
|
161
|
-
# to be pasted
|
162
|
-
#
|
163
|
-
# @param original_id [String]
|
164
|
-
#
|
165
|
-
def modified_id_to_paste(original_id)
|
166
|
-
return nil if original_id.nil?
|
167
|
-
return '' if original_id.blank?
|
168
|
-
|
169
|
-
count = next_count_for_pasted_id(original_id)
|
170
|
-
|
171
|
-
# A count of 0 means the element was cut and this is the first paste, do not
|
172
|
-
# modify the ID; otherwise, use the uniquified ID.
|
173
|
-
if count.zero?
|
174
|
-
original_id
|
175
|
-
else
|
176
|
-
"#{original_id}#{@id_copy_suffix}#{count}"
|
177
|
-
end
|
178
|
-
end
|
179
|
-
|
180
155
|
# Returns the underlying Nokogiri Document object
|
181
156
|
#
|
182
157
|
# @return [Nokogiri::XML::Document]
|
@@ -185,16 +160,19 @@ module Kitchen
|
|
185
160
|
@nokogiri_document
|
186
161
|
end
|
187
162
|
|
188
|
-
|
189
|
-
|
190
|
-
|
191
|
-
|
192
|
-
|
193
|
-
|
194
|
-
|
163
|
+
# Returns the locale for this document, default to `:en` if no locale detected
|
164
|
+
#
|
165
|
+
# @return [Symbol]
|
166
|
+
#
|
167
|
+
def locale
|
168
|
+
raw.root['lang']&.to_sym || begin
|
169
|
+
warn 'No `lang` attribute on this document so cannot detect its locale; defaulting to `:en`'
|
170
|
+
:en
|
195
171
|
end
|
196
172
|
end
|
197
173
|
|
174
|
+
protected
|
175
|
+
|
198
176
|
attr_reader :nokogiri_document
|
199
177
|
|
200
178
|
end
|
data/lib/kitchen/element.rb
CHANGED
@@ -19,8 +19,14 @@ module Kitchen
|
|
19
19
|
short_type: short_type)
|
20
20
|
end
|
21
21
|
|
22
|
-
#
|
23
|
-
#
|
24
|
-
#
|
22
|
+
# Returns true if this class represents the element for the given node; always false
|
23
|
+
# for this generic class
|
24
|
+
#
|
25
|
+
# @param node [Nokogiri::XML::Node] the underlying node
|
26
|
+
# @return [Boolean]
|
27
|
+
#
|
28
|
+
def self.is_the_element_class_for?(_node, **)
|
29
|
+
false
|
30
|
+
end
|
25
31
|
end
|
26
32
|
end
|
data/lib/kitchen/element_base.rb
CHANGED
@@ -3,6 +3,7 @@
|
|
3
3
|
require 'forwardable'
|
4
4
|
require 'securerandom'
|
5
5
|
|
6
|
+
# rubocop:disable Metrics/ClassLength
|
6
7
|
module Kitchen
|
7
8
|
# Abstract base class for all elements. If you are looking for a simple concrete
|
8
9
|
# element class, use `Element`.
|
@@ -92,6 +93,16 @@ module Kitchen
|
|
92
93
|
# @return [Selectors::Base]
|
93
94
|
def_delegators :config, :selectors
|
94
95
|
|
96
|
+
# @!method pantry
|
97
|
+
# Access the pantry for this element's document
|
98
|
+
# @return [Pantry]
|
99
|
+
# @!method :clipboard
|
100
|
+
# Access the clipboard for this element's document
|
101
|
+
# @return [Clipboard]
|
102
|
+
def_delegators :document, :pantry, :clipboard
|
103
|
+
|
104
|
+
def_delegators :document, :id_tracker
|
105
|
+
|
95
106
|
# Creates a new instance
|
96
107
|
#
|
97
108
|
# @param node [Nokogiri::XML::Node] the wrapped element
|
@@ -108,7 +119,9 @@ module Kitchen
|
|
108
119
|
|
109
120
|
@enumerator_class = enumerator_class
|
110
121
|
|
111
|
-
@short_type = short_type ||
|
122
|
+
@short_type = short_type ||
|
123
|
+
self.class.try(:short_type) ||
|
124
|
+
"unknown_type_#{SecureRandom.hex(4)}"
|
112
125
|
|
113
126
|
@document =
|
114
127
|
case document
|
@@ -121,16 +134,53 @@ module Kitchen
|
|
121
134
|
@ancestors = HashWithIndifferentAccess.new
|
122
135
|
@search_query_matches_that_have_been_counted = {}
|
123
136
|
@is_a_clone = false
|
137
|
+
@search_cache = {}
|
138
|
+
end
|
139
|
+
|
140
|
+
# Returns ElementBase descendent type or nil if none found
|
141
|
+
#
|
142
|
+
# @param type [Symbol] the descendant type, e.g. `:page`
|
143
|
+
# @return [Class] the child class for the given type
|
144
|
+
#
|
145
|
+
def self.descendant(type)
|
146
|
+
@types_to_descendants ||=
|
147
|
+
descendants.each_with_object({}) do |descendant, hash|
|
148
|
+
next unless descendant.try(:short_type)
|
149
|
+
|
150
|
+
hash[descendant.short_type] = descendant
|
151
|
+
end
|
152
|
+
|
153
|
+
@types_to_descendants[type]
|
154
|
+
end
|
155
|
+
|
156
|
+
# Returns ElementBase descendent type or Error if none found
|
157
|
+
#
|
158
|
+
# @param type [Symbol] the descendant type, e.g. `:page`
|
159
|
+
# @raise if the type is unknown
|
160
|
+
# @return [Class] the child class for the given type
|
161
|
+
#
|
162
|
+
def self.descendant!(type)
|
163
|
+
descendant(type) || raise("Unknown ElementBase descendant type '#{type}'")
|
164
|
+
end
|
165
|
+
|
166
|
+
# Returns true if this element is the given type
|
167
|
+
#
|
168
|
+
# @param type [Symbol] the descendant type, e.g. `:page`
|
169
|
+
# @raise if the type is unknown
|
170
|
+
# @return [Boolean]
|
171
|
+
#
|
172
|
+
def is?(type)
|
173
|
+
ElementBase.descendant!(type).is_the_element_class_for?(raw, config: config)
|
124
174
|
end
|
125
175
|
|
126
176
|
# Returns true if this class represents the element for the given node
|
127
177
|
#
|
128
178
|
# @param node [Nokogiri::XML::Node] the underlying node
|
179
|
+
# @param config [Kitchen::Config]
|
129
180
|
# @return [Boolean]
|
130
181
|
#
|
131
|
-
def self.is_the_element_class_for?(
|
132
|
-
|
133
|
-
false
|
182
|
+
def self.is_the_element_class_for?(node, config:)
|
183
|
+
Selector.named(short_type).matches?(node, config: config)
|
134
184
|
end
|
135
185
|
|
136
186
|
# Returns true if this element has the given class
|
@@ -298,7 +348,7 @@ module Kitchen
|
|
298
348
|
# search results if the method or callable returns false
|
299
349
|
# @return [ElementEnumerator]
|
300
350
|
#
|
301
|
-
def search(*selector_or_xpath_args, only: nil, except: nil)
|
351
|
+
def search(*selector_or_xpath_args, only: nil, except: nil, reload: false)
|
302
352
|
block_error_if(block_given?)
|
303
353
|
|
304
354
|
ElementEnumerator.factory.build_within(
|
@@ -307,19 +357,29 @@ module Kitchen
|
|
307
357
|
css_or_xpath: selector_or_xpath_args,
|
308
358
|
only: only,
|
309
359
|
except: except
|
310
|
-
)
|
360
|
+
),
|
361
|
+
reload: reload
|
311
362
|
)
|
312
363
|
end
|
313
364
|
|
365
|
+
def raw_search(*selector_or_xpath_args, reload: false)
|
366
|
+
key = selector_or_xpath_args
|
367
|
+
@search_cache[key] = nil if reload || !config.enable_search_cache
|
368
|
+
# cache nil search results with a fake -1 value
|
369
|
+
@search_cache[key] ||= raw.search(*selector_or_xpath_args) || -1
|
370
|
+
@search_cache[key] == -1 ? nil : @search_cache[key]
|
371
|
+
end
|
372
|
+
|
314
373
|
# Yields and returns the first child element that matches the provided
|
315
374
|
# selector or XPath arguments.
|
316
375
|
#
|
317
376
|
# @param selector_or_xpath_args [Array<String>] CSS selectors or XPath arguments
|
377
|
+
# @param reload [Boolean] ignores cache if true
|
318
378
|
# @yieldparam [Element] the matched XML element
|
319
379
|
# @return [Element, nil] the matched XML element or nil if no match found
|
320
380
|
#
|
321
|
-
def first(*selector_or_xpath_args)
|
322
|
-
search(*selector_or_xpath_args).first.tap do |element|
|
381
|
+
def first(*selector_or_xpath_args, reload: false)
|
382
|
+
search(*selector_or_xpath_args, reload: reload).first.tap do |element|
|
323
383
|
yield(element) if block_given?
|
324
384
|
end
|
325
385
|
end
|
@@ -328,12 +388,13 @@ module Kitchen
|
|
328
388
|
# selector or XPath arguments.
|
329
389
|
#
|
330
390
|
# @param selector_or_xpath_args [Array<String>] CSS selectors or XPath arguments
|
391
|
+
# @param reload [Boolean] ignores cache if true
|
331
392
|
# @yieldparam [Element] the matched XML element
|
332
393
|
# @raise [ElementNotFoundError] if no matching element is found
|
333
394
|
# @return [Element] the matched XML element
|
334
395
|
#
|
335
|
-
def first!(*selector_or_xpath_args)
|
336
|
-
search(*selector_or_xpath_args).first!.tap do |element|
|
396
|
+
def first!(*selector_or_xpath_args, reload: false)
|
397
|
+
search(*selector_or_xpath_args, reload: reload).first!.tap do |element|
|
337
398
|
yield(element) if block_given?
|
338
399
|
end
|
339
400
|
end
|
@@ -382,8 +443,13 @@ module Kitchen
|
|
382
443
|
def cut(to: nil)
|
383
444
|
block_error_if(block_given?)
|
384
445
|
|
446
|
+
raw.traverse do |node|
|
447
|
+
next if node.text? || node.document?
|
448
|
+
|
449
|
+
id_tracker.record_id_cut(node[:id])
|
450
|
+
end
|
385
451
|
node.remove
|
386
|
-
|
452
|
+
get_clipboard(to).add(self) if to.present?
|
387
453
|
self
|
388
454
|
end
|
389
455
|
|
@@ -402,9 +468,9 @@ module Kitchen
|
|
402
468
|
the_copy.raw.traverse do |node|
|
403
469
|
next if node.text? || node.document?
|
404
470
|
|
405
|
-
|
471
|
+
id_tracker.record_id_copied(node[:id])
|
406
472
|
end
|
407
|
-
|
473
|
+
get_clipboard(to).add(the_copy) if to.present?
|
408
474
|
the_copy
|
409
475
|
end
|
410
476
|
|
@@ -413,20 +479,22 @@ module Kitchen
|
|
413
479
|
def paste
|
414
480
|
# See `clone` method for a note about namespaces
|
415
481
|
block_error_if(block_given?)
|
416
|
-
|
417
482
|
temp_copy = clone
|
418
483
|
temp_copy.raw.traverse do |node|
|
419
484
|
next if node.text? || node.document?
|
420
485
|
|
421
|
-
|
486
|
+
if node[:id].present?
|
487
|
+
id_tracker.record_id_pasted(node[:id])
|
488
|
+
node[:id] = id_tracker.modified_id_to_paste(node[:id])
|
489
|
+
end
|
422
490
|
end
|
423
491
|
temp_copy.to_s
|
424
492
|
end
|
425
493
|
|
426
494
|
# Copy the element's id
|
427
495
|
def copied_id
|
428
|
-
|
429
|
-
|
496
|
+
id_tracker.record_id_copied(id)
|
497
|
+
id_tracker.modified_id_to_paste(id)
|
430
498
|
end
|
431
499
|
|
432
500
|
# Delete the element
|
@@ -440,6 +508,21 @@ module Kitchen
|
|
440
508
|
Element.new(node: raw.parent, document: document, short_type: "parent(#{short_type})")
|
441
509
|
end
|
442
510
|
|
511
|
+
# returns previous element
|
512
|
+
# skips double indentations that the nokigiri sometimes picks up
|
513
|
+
# nil if there's no previous sibling
|
514
|
+
#
|
515
|
+
def previous
|
516
|
+
prev = raw.previous
|
517
|
+
return prev if prev.nil?
|
518
|
+
|
519
|
+
Element.new(
|
520
|
+
node: prev,
|
521
|
+
document: document,
|
522
|
+
short_type: "previous(#{short_type})"
|
523
|
+
)
|
524
|
+
end
|
525
|
+
|
443
526
|
# TODO: make it clear if all of these methods take Element, Node, or String
|
444
527
|
|
445
528
|
# If child argument given, prepends it before the element's current children.
|
@@ -646,7 +729,8 @@ module Kitchen
|
|
646
729
|
# @!method pages
|
647
730
|
# Returns a pages enumerator
|
648
731
|
def_delegators :as_enumerator, :pages, :chapters, :terms, :figures, :notes, :tables, :examples,
|
649
|
-
:metadatas, :non_introduction_pages, :units, :titles, :exercises, :
|
732
|
+
:metadatas, :non_introduction_pages, :units, :titles, :exercises, :references,
|
733
|
+
:composite_pages, :composite_chapters
|
650
734
|
|
651
735
|
# Returns this element as an enumerator (over only one element, itself)
|
652
736
|
#
|
@@ -671,10 +755,10 @@ module Kitchen
|
|
671
755
|
# @param name_or_object [String, Clipboard] the name of the clipboard or the clipboard itself
|
672
756
|
# @return [Clipboard]
|
673
757
|
#
|
674
|
-
def
|
758
|
+
def get_clipboard(name_or_object)
|
675
759
|
case name_or_object
|
676
760
|
when Symbol
|
677
|
-
|
761
|
+
clipboard(name: name_or_object)
|
678
762
|
when Clipboard
|
679
763
|
name_or_object
|
680
764
|
else
|
@@ -688,7 +772,9 @@ module Kitchen
|
|
688
772
|
# @param string [String] the string to clean
|
689
773
|
def remove_default_namespaces_if_clone(string)
|
690
774
|
if is_a_clone
|
691
|
-
string.gsub('xmlns:default="http://www.w3.org/1999/xhtml"', '')
|
775
|
+
string.gsub('xmlns:default="http://www.w3.org/1999/xhtml"', '')
|
776
|
+
.gsub('xmlns="http://www.w3.org/1999/xhtml"', '')
|
777
|
+
.gsub('default:', '')
|
692
778
|
else
|
693
779
|
string
|
694
780
|
end
|
@@ -701,3 +787,4 @@ module Kitchen
|
|
701
787
|
|
702
788
|
end
|
703
789
|
end
|
790
|
+
# rubocop:enable Metrics/ClassLength
|