scarpe-components 0.1.0 → 0.3.0

Sign up to get free protection for your applications and to get access to all the features.
@@ -0,0 +1,225 @@
1
+ # frozen_string_literal: true
2
+
3
+ # In Italian, tiranti are bootstraps -- the literal pull-on-a-boot kind, not a step to something better.
4
+ # Tiranti.rb builds on calzini.rb, but renders a Bootstrap-decorated version of the HTML output.
5
+ # You would ordinarily set either Calzini or Tiranti as the top-level HTML renderer, not both.
6
+ # You'll include both if you use Tiranti, because it falls back to Calzini for a lot of its rendering.
7
+
8
+ require "scarpe/components/calzini"
9
+
10
+ # The Tiranti module expects to be included by a class defining
11
+ # the following methods:
12
+ #
13
+ # * html_id - the HTML ID for the specific rendered DOM object
14
+ # * handler_js_code(event_name) - the JS handler code for this DOM object and event name
15
+ # * (optional) display_properties - the display properties for this object, unless overridden in render()
16
+ module Scarpe::Components::Tiranti
17
+ include Scarpe::Components::Calzini
18
+ extend self
19
+
20
+ # Currently we're using Bootswatch 5
21
+ BOOTSWATCH_THEMES = [
22
+ "cerulean",
23
+ "cosmo",
24
+ "cyborg",
25
+ "darkly",
26
+ "flatly",
27
+ "journal",
28
+ "litera",
29
+ "lumen",
30
+ "lux",
31
+ "materia",
32
+ "minty",
33
+ "morph",
34
+ "pulse",
35
+ "quartz",
36
+ "sandstone",
37
+ "simplex",
38
+ "sketchy",
39
+ "slate",
40
+ "solar",
41
+ "spacelab",
42
+ "superhero",
43
+ "united",
44
+ "vapor",
45
+ "yeti",
46
+ "zephyr",
47
+ ]
48
+
49
+ BOOTSWATCH_THEME = ENV["SCARPE_BOOTSTRAP_THEME"] || "sketchy"
50
+
51
+ def empty_page_element
52
+ <<~HTML
53
+ <html>
54
+ <head id='head-wvroot'>
55
+ <meta charset="utf-8">
56
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no">
57
+ <link rel="stylesheet" href="https://bootswatch.com/5/#{BOOTSWATCH_THEME}/bootstrap.css">
58
+ <link rel="stylesheet" href="https://bootswatch.com/_vendor/bootstrap-icons/font/bootstrap-icons.min.css">
59
+ <style id='style-wvroot'>
60
+ /** Style resets **/
61
+ body {
62
+ height: 100%;
63
+ overflow: hidden;
64
+ }
65
+ </style>
66
+ </head>
67
+ <body id='body-wvroot'>
68
+ <div id='wrapper-wvroot'></div>
69
+
70
+ <script src="https://bootswatch.com/_vendor/bootstrap/dist/js/bootstrap.bundle.min.js"></script>
71
+ </body>
72
+ </html>
73
+ HTML
74
+ end
75
+
76
+ # def render_stack
77
+ # end
78
+ # def render_flow
79
+ # end
80
+
81
+ # How do we want to handle theme-specific colours and primary/secondary buttons in Bootstrap?
82
+ # "Disabled" could be checked in properties. Is there any way we can/should use "outline" buttons?
83
+ def button_element(props)
84
+ HTML.render do |h|
85
+ h.button(id: html_id, type: "button", class: "btn btn-primary", onclick: handler_js_code("click"), style: button_style(props)) do
86
+ props["text"]
87
+ end
88
+ end
89
+ end
90
+
91
+ private
92
+
93
+ def button_style(props)
94
+ styles = drawable_style(props)
95
+
96
+ styles[:"background-color"] = props["color"] if props["color"]
97
+ styles[:"padding-top"] = props["padding_top"] if props["padding_top"]
98
+ styles[:"padding-bottom"] = props["padding_bottom"] if props["padding_bottom"]
99
+ styles[:color] = props["text_color"] if props["text_color"]
100
+ styles[:width] = dimensions_length(props["width"]) if props["width"]
101
+ styles[:height] = dimensions_length(props["height"]) if props["height"]
102
+ styles[:"font-size"] = props["font_size"] if props["font_size"]
103
+
104
+ styles[:top] = dimensions_length(props["top"]) if props["top"]
105
+ styles[:left] = dimensions_length(props["left"]) if props["left"]
106
+ styles[:position] = "absolute" if props["top"] || props["left"]
107
+ styles[:"font-size"] = dimensions_length(text_size(props["size"])) if props["size"]
108
+ styles[:"font-family"] = props["font"] if props["font"]
109
+
110
+ styles
111
+ end
112
+
113
+ public
114
+
115
+ def alert_element(props)
116
+ onclick = handler_js_code(props["event_name"] || "click")
117
+
118
+ HTML.render do |h|
119
+ h.div(id: html_id, class: "modal", tabindex: -1, role: "dialog", style: alert_overlay_style(props)) do
120
+ h.div(class: "modal-dialog", role: "document") do
121
+ h.div(class: "modal-content", style: alert_modal_style) do
122
+ h.div(class: "modal-header") do
123
+ h.h5(class: "modal-title") { "Alert" }
124
+ h.button(type: "button", class: "close", data_dismiss: "modal", aria_label: "Close") do
125
+ h.span(aria_hidden: "true") { "&times;" }
126
+ end
127
+ end
128
+ h.div(class: "modal-body") do
129
+ h.p { props["text"] }
130
+ end
131
+ h.div(class: "modal-footer") do
132
+ h.button(type: "button", onclick:, class: "btn btn-primary") { "OK" }
133
+ #h.button(type: "button", class: "btn btn-secondary") { "Close" }
134
+ end
135
+ end
136
+ end
137
+ end
138
+ end
139
+ end
140
+
141
+ def check_element(props)
142
+ HTML.render do |h|
143
+ h.div class: "form-check" do
144
+ h.input type: :checkbox,
145
+ id: html_id,
146
+ class: "form-check-input",
147
+ onclick: handler_js_code("click"),
148
+ value: props["text"],
149
+ checked: props["checked"],
150
+ style: drawable_style(props)
151
+ end
152
+ end
153
+ end
154
+
155
+ def progress_element(props)
156
+ HTML.render do |h|
157
+ h.div(class: "progress", style: "width: 90%") do
158
+ pct = "%.1f" % ((props["fraction"] || 0.0) * 100.0)
159
+ h.div(
160
+ class: "progress-bar progress-bar-striped progress-bar-animated",
161
+ role: "progressbar",
162
+ "aria-valuenow": pct,
163
+ "aria-valuemin": 0,
164
+ "aria-valuemax": 100,
165
+ style: "width: #{pct}%",
166
+ )
167
+ end
168
+ end
169
+ end
170
+
171
+ # para_element is a bit of a hard one, since it does not-entirely-trivial
172
+ # mapping between display objects and IDs. But we don't want Calzini
173
+ # messing with the display service or display objects.
174
+ def para_element(props, &block)
175
+ tag, opts = para_elt_and_opts(props)
176
+
177
+ HTML.render do |h|
178
+ h.send(tag, **opts, &block)
179
+ end
180
+ end
181
+
182
+ private
183
+
184
+ ELT_AND_SIZE = {
185
+ inscription: [:p, 10],
186
+ ins: [:p, 10],
187
+ para: [:p, 12],
188
+ caption: [:p, 14],
189
+ tagline: [:p, 18],
190
+ subtitle: [:h3, 26],
191
+ title: [:h2, 34],
192
+ banner: [:h1, 48],
193
+ }.freeze
194
+
195
+ def para_elt_and_opts(props)
196
+ elt, size = para_elt_and_size(props)
197
+ size = dimensions_length(size)
198
+
199
+ para_style = drawable_style(props).merge({
200
+ color: rgb_to_hex(props["stroke"]),
201
+ "font-size": para_font_size(props),
202
+ "font-family": props["font"],
203
+ }.compact)
204
+
205
+ opts = (props["html_attributes"] || {}).merge(id: html_id, style: para_style)
206
+
207
+ [elt, opts]
208
+ end
209
+
210
+ def para_elt_and_size(props)
211
+ return [:p, nil] unless props["size"]
212
+
213
+ ps = props["size"].to_s.to_sym
214
+ if ELT_AND_SIZE.key?(ps)
215
+ ELT_AND_SIZE[ps]
216
+ else
217
+ sz = props["size"].to_i
218
+ if sz > 18
219
+ [:h2, sz]
220
+ else
221
+ [:p, sz]
222
+ end
223
+ end
224
+ end
225
+ end
@@ -0,0 +1,257 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "tempfile"
4
+ require "json"
5
+ require "fileutils"
6
+
7
+ require "scarpe/components/file_helpers"
8
+
9
+ module Scarpe::Test; end
10
+
11
+ # We want test failures set up once *total*, not per Minitest::Test. So an instance var
12
+ # doesn't do it.
13
+ ALREADY_SET_UP_LOGGED_TEST_FAILURES = { setup: false }
14
+
15
+ # General helpers for general usage.
16
+ # Helpers here should *not* use Webview-specific functionality.
17
+ # The intention is that these are helpers for various Scarpe display
18
+ # services that do *not* necessarily use Webview.
19
+
20
+ module Scarpe::Test::Helpers
21
+ # Very useful for tests
22
+ include Scarpe::Components::FileHelpers
23
+
24
+ # Temporarily set env vars for the block of code inside. The old environment
25
+ # variable values will be restored after the block finishes.
26
+ #
27
+ # @param envs [Hash<String,String>] A hash of environment variable names and values
28
+ def with_env_vars(envs)
29
+ old_env = {}
30
+ envs.each do |k, v|
31
+ old_env[k] = ENV[k]
32
+ ENV[k] = v
33
+ end
34
+ yield
35
+ ensure
36
+ old_env.each { |k, v| ENV[k] = v }
37
+ end
38
+ end
39
+
40
+ # This test will save extensive logs in case of test failure.
41
+ # Note that it defines setup/teardown methods. If you want
42
+ # multiple setup/teardowns from multiple places to happen you
43
+ # may need to explictly call (e.g. with logged_test_setup/teardown)
44
+ # to ensure everything you want happens.
45
+ module Scarpe::Test::LoggedTest
46
+ def self.included(includer)
47
+ class << includer
48
+ attr_accessor :logger_dir
49
+ end
50
+ end
51
+
52
+ def file_id
53
+ "#{self.class.name}_#{self.name}"
54
+ end
55
+
56
+ # This should be called by the test during setup to make sure that
57
+ # failure logs will be saved if this test fails. It makes sure the
58
+ # log config will save all logs from all sources, but keeps a copy
59
+ # of the old log config to restore after the test is finished.
60
+ #
61
+ # @return [void]
62
+ def logged_test_setup
63
+ # Make sure test failures will be saved at the end of the run.
64
+ # Delete stale test failures and logging only the *first* time this is called.
65
+ set_up_test_failures
66
+
67
+ @normal_log_config = Shoes::Log.current_log_config
68
+ Shoes::Log.configure_logger(log_config_for_test)
69
+
70
+ Shoes::Log.logger("LoggedScarpeTest").info("Test: #{self.class.name}##{self.name}")
71
+ end
72
+
73
+ # If you include this module and don't override setup/teardown, everything will
74
+ # work fine. But if you need more setup/teardown steps, you can do that too.
75
+ #
76
+ # The setup method guarantees that just including this module will do setup
77
+ # automatically. If you override it, be sure to call `super` or `logged_test_setup`.
78
+ #
79
+ # @return [void]
80
+ def setup
81
+ logged_test_setup
82
+ end
83
+
84
+ # After the test has finished, this will restore the old log configuration.
85
+ # It will also save the logfiles, but only if the test failed, not if it
86
+ # succeeded or was skipped.
87
+ #
88
+ # @return [void]
89
+ def logged_test_teardown
90
+ # Restore previous log config
91
+ Shoes::Log.configure_logger(@normal_log_config)
92
+
93
+ if self.failure
94
+ save_failure_logs
95
+ else
96
+ remove_unsaved_logs
97
+ end
98
+ end
99
+
100
+ # Make sure that, by default, #logged_test_teardown will be called for teardown.
101
+ # If a class overrides teardown, it should also call `super` or `logged_test_teardown`
102
+ # to make sure this still happens.
103
+ #
104
+ # @return [void]
105
+ def teardown
106
+ logged_test_teardown
107
+ end
108
+
109
+ # Set additional LoggedTest configuration for specific logs to separate or save.
110
+ # This is normally going to be display-service-specific log components.
111
+ # Note that this only really works with the modular logger or another logger
112
+ # that does something useful with the log config. The simple print logger
113
+ # doesn't do a lot with it.
114
+ def extra_log_config=(additional_log_config)
115
+ @additional_log_config = additional_log_config
116
+ end
117
+
118
+ # This is the log config that LoggedTests use. It makes sure all components keep all
119
+ # logs, but also splits the logs into several different files for later ease of scanning.
120
+ #
121
+ # TODO: this shouldn't directly include any Webview entries like WebviewAPI or
122
+ # CatsCradle. Those should be overridden in Webview.
123
+ #
124
+ # @return [Hash] the log config
125
+ def log_config_for_test
126
+ {
127
+ "default" => ["debug", "logger/test_failure_#{file_id}.log"],
128
+ "DisplayService" => ["debug", "logger/test_failure_display_service_#{file_id}.log"],
129
+ }.merge(@additional_log_config || {})
130
+ end
131
+
132
+ # The list of logfiles that should be saved. Normally this is called internally by the
133
+ # class, not externally from elsewhere.
134
+ #
135
+ # This could be a lot simpler except I want to only update the file list in one place,
136
+ # log_config_for_test(). Having a single spot should (I hope) make it a lot friendlier to
137
+ # add more logfiles for different components, logged API objects, etc.
138
+ def saved_log_files
139
+ lc = log_config_for_test
140
+ log_outfiles = lc.values.map { |_level, loc| loc }
141
+ log_outfiles.select { |s| s.start_with?("logger/") }.map { |s| s.delete_prefix("logger/") }
142
+ end
143
+
144
+ # Make sure that test failure logs will be noticed, and a message will be printed,
145
+ # if any logged tests fail. This needs to be called at least once in any Minitest-enabled
146
+ # process using logged tests.
147
+ #
148
+ # @return [void]
149
+ def set_up_test_failures
150
+ return if ALREADY_SET_UP_LOGGED_TEST_FAILURES[:setup]
151
+
152
+ log_dir = self.class.logger_dir
153
+ raise(Scarpe::MustOverrideMethod, "Must set logger directory!") unless log_dir
154
+ raise(Scarpe::NoSuchFile, "Can't find logger directory!") unless File.directory?(log_dir)
155
+
156
+ ALREADY_SET_UP_LOGGED_TEST_FAILURES[:setup] = true
157
+ # Delete stale test failures, if any, before starting the first failure-logged test
158
+ Dir["#{log_dir}/test_failure*.log"].each { |fn| File.unlink(fn) }
159
+
160
+ Minitest.after_run do
161
+ # Print test failure notice to console
162
+ unless Dir["#{log_dir}/test_failure*.out.log"].empty?
163
+ puts "Some tests have failed! See #{log_dir}/test_failure*.out.log for test logs!"
164
+ end
165
+
166
+ # Remove un-saved test logs
167
+ Dir["#{log_dir}/test_failure*.log"].each do |f|
168
+ next if f.include?(".out.log")
169
+
170
+ File.unlink(f) if File.exist?(f)
171
+ end
172
+ end
173
+ end
174
+
175
+ # Failure log output location for a given file path. This is normally used internally to this
176
+ # class, not externally.
177
+ #
178
+ # @return [String] the output path
179
+ def logfail_out_loc(filepath)
180
+ # Add a .out prefix before final .log
181
+ out_loc = filepath.gsub(%r{.log\Z}, ".out.log")
182
+
183
+ if out_loc == filepath
184
+ raise Shoes::Errors::InvalidAttributeValueError, "Something is wrong! Could not figure out failure-log output path for #{filepath.inspect}!"
185
+ end
186
+
187
+ if File.exist?(out_loc)
188
+ raise Scarpe::DuplicateFileError, "Duplicate test file #{out_loc.inspect}? This file should *not* already exist!"
189
+ end
190
+
191
+ out_loc
192
+ end
193
+
194
+ # Save the failure logs in the appropriate place(s). This is normally used internally, not externally.
195
+ #
196
+ # @return [void]
197
+ def save_failure_logs
198
+ saved_log_files.each do |log_file|
199
+ full_loc = File.expand_path("#{self.class.logger_dir}/#{log_file}")
200
+ # TODO: we'd like to skip 0-length logfiles. But also Logging doesn't flush. For now, ignore.
201
+ next unless File.exist?(full_loc)
202
+
203
+ FileUtils.mv full_loc, logfail_out_loc(full_loc)
204
+ end
205
+ end
206
+
207
+ # Remove unsaved failure logs. This is normally used internally, not externally.
208
+ #
209
+ # @return [void]
210
+ def remove_unsaved_logs
211
+ Dir["#{self.class.logger_dir}/test_failure*.log"].each do |f|
212
+ next if f.include?(".out.log") # Don't delete saved logs
213
+
214
+ File.unlink(f)
215
+ end
216
+ end
217
+ end
218
+
219
+ module Scarpe::Test::HTMLAssertions
220
+ # Assert that `actual_html` is the same as `expected_tag` with `opts`.
221
+ # This uses Scarpe's HTML tag-based renderer to render the tag and options
222
+ # into text, and valides that the text is the same.
223
+ #
224
+ # @see Scarpe::Components::HTML.render
225
+ #
226
+ # @param actual_html [String] the html to compare to
227
+ # @param expected_tag [String,Symbol] the HTML tag, used to send a method call
228
+ # @param opts keyword options passed to the tag method call
229
+ # @yield block passed to the tag method call.
230
+ # @return [void]
231
+ def assert_html(actual_html, expected_tag, **opts, &block)
232
+ expected_html = Scarpe::Components::HTML.render do |h|
233
+ h.public_send(expected_tag, opts, &block)
234
+ end
235
+
236
+ assert_equal expected_html, actual_html
237
+ end
238
+
239
+ # Assert that `actual_html` includes `expected_tag` with `opts`.
240
+ # This uses Scarpe's HTML tag-based renderer to render the tag and options
241
+ # into text, and valides that the full HTML contains that tag.
242
+ #
243
+ # @see Scarpe::Components::HTML.render
244
+ #
245
+ # @param actual_html [String] the html to compare to
246
+ # @param expected_tag [String,Symbol] the HTML tag, used to send a method call
247
+ # @param opts keyword options passed to the tag method call
248
+ # @yield block passed to the tag method call.
249
+ # @return [void]
250
+ def assert_contains_html(actual_html, expected_tag, **opts, &block)
251
+ expected_html = Scarpe::Components::HTML.render do |h|
252
+ h.public_send(expected_tag, opts, &block)
253
+ end
254
+
255
+ assert actual_html.include?(expected_html), "Expected #{actual_html.inspect} to include #{expected_html.inspect}!"
256
+ end
257
+ end
@@ -0,0 +1,7 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Scarpe
4
+ module Components
5
+ VERSION = "0.3.0"
6
+ end
7
+ end
metadata CHANGED
@@ -1,7 +1,7 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: scarpe-components
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.1.0
4
+ version: 0.3.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Marco Concetto Rudilosso
@@ -9,7 +9,7 @@ authors:
9
9
  autorequire:
10
10
  bindir: exe
11
11
  cert_chain: []
12
- date: 2023-08-08 00:00:00.000000000 Z
12
+ date: 2023-11-24 00:00:00.000000000 Z
13
13
  dependencies: []
14
14
  description:
15
15
  email:
@@ -19,9 +19,33 @@ executables: []
19
19
  extensions: []
20
20
  extra_rdoc_files: []
21
21
  files:
22
+ - Gemfile
23
+ - Gemfile.lock
22
24
  - README.md
23
25
  - Rakefile
24
- - lib/scarpe/logger.rb
26
+ - lib/scarpe/components/base64.rb
27
+ - lib/scarpe/components/calzini.rb
28
+ - lib/scarpe/components/calzini/alert.rb
29
+ - lib/scarpe/components/calzini/art_widgets.rb
30
+ - lib/scarpe/components/calzini/button.rb
31
+ - lib/scarpe/components/calzini/misc.rb
32
+ - lib/scarpe/components/calzini/para.rb
33
+ - lib/scarpe/components/calzini/slots.rb
34
+ - lib/scarpe/components/calzini/text_widgets.rb
35
+ - lib/scarpe/components/errors.rb
36
+ - lib/scarpe/components/file_helpers.rb
37
+ - lib/scarpe/components/html.rb
38
+ - lib/scarpe/components/minitest_export_reporter.rb
39
+ - lib/scarpe/components/minitest_import_runnable.rb
40
+ - lib/scarpe/components/minitest_result.rb
41
+ - lib/scarpe/components/modular_logger.rb
42
+ - lib/scarpe/components/print_logger.rb
43
+ - lib/scarpe/components/promises.rb
44
+ - lib/scarpe/components/segmented_file_loader.rb
45
+ - lib/scarpe/components/string_helpers.rb
46
+ - lib/scarpe/components/tiranti.rb
47
+ - lib/scarpe/components/unit_test_helpers.rb
48
+ - lib/scarpe/components/version.rb
25
49
  homepage: https://github.com/scarpe-team/scarpe
26
50
  licenses:
27
51
  - MIT
@@ -44,7 +68,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
44
68
  - !ruby/object:Gem::Version
45
69
  version: '0'
46
70
  requirements: []
47
- rubygems_version: 3.4.1
71
+ rubygems_version: 3.4.10
48
72
  signing_key:
49
73
  specification_version: 4
50
74
  summary: Reusable components for Scarpe display libraries