scarpe 0.2.2 → 0.3.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.yardopts +1 -0
- data/CHANGELOG.md +16 -2
- data/Gemfile.lock +7 -3
- data/README.md +24 -8
- data/Rakefile +1 -1
- data/examples/animate.rb +20 -0
- data/examples/arrow.rb +10 -0
- data/examples/btn_tooltip.rb +7 -0
- data/examples/button_style_changed.rb +7 -0
- data/examples/button_styles_default.rb +6 -0
- data/examples/gen.rb +8 -8
- data/examples/highlander.rb +3 -3
- data/examples/legacy/README.md +6 -0
- data/examples/legacy/not_checked/shoes-contrib/basic/shoes-notes.rb +1 -1
- data/examples/legacy/not_checked/simple/anim-shapes.rb +1 -1
- data/examples/legacy/not_checked/speedometer_app.rb +55 -0
- data/examples/legacy/working/simple/image-icon.rb +3 -0
- data/examples/legacy/{not_checked → working}/simple/image.rb +1 -1
- data/examples/list_box_choose.rb +17 -0
- data/examples/local_assets/local_file_server.rb +82 -0
- data/examples/local_assets/sample.gif +0 -0
- data/examples/local_assets/sample.mp4 +0 -0
- data/examples/local_fonts.rb +2 -2
- data/examples/local_images.rb +2 -3
- data/examples/para/para_text.rb +14 -0
- data/examples/progress.rb +31 -0
- data/examples/radio/radio_groups.rb +2 -2
- data/examples/rect.rb +4 -0
- data/examples/rotate_shapes.rb +17 -0
- data/examples/simpler-menu.rb +21 -0
- data/exe/scarpe +2 -1
- data/lacci/Gemfile +2 -0
- data/lacci/Gemfile.lock +8 -1
- data/lacci/lacci.gemspec +1 -1
- data/lacci/lib/lacci/scarpe_cli.rb +2 -1
- data/lacci/lib/lacci/scarpe_core.rb +2 -1
- data/lacci/lib/lacci/version.rb +1 -1
- data/lacci/lib/scarpe/niente/app.rb +23 -0
- data/lacci/lib/scarpe/niente/display_service.rb +62 -0
- data/lacci/lib/scarpe/niente/drawable.rb +57 -0
- data/lacci/lib/scarpe/niente/logger.rb +29 -0
- data/lacci/lib/scarpe/niente/shoes_spec.rb +87 -0
- data/lacci/lib/scarpe/niente.rb +20 -0
- data/lacci/lib/shoes/app.rb +88 -43
- data/lacci/lib/shoes/background.rb +2 -2
- data/lacci/lib/shoes/border.rb +2 -2
- data/lacci/lib/shoes/builtins.rb +63 -0
- data/lacci/lib/shoes/changelog.rb +52 -0
- data/lacci/lib/shoes/colors.rb +3 -1
- data/lacci/lib/shoes/constants.rb +19 -1
- data/lacci/lib/shoes/display_service.rb +39 -16
- data/lacci/lib/shoes/download.rb +2 -2
- data/lacci/lib/shoes/drawable.rb +380 -0
- data/lacci/lib/shoes/drawables/arc.rb +49 -0
- data/lacci/lib/shoes/drawables/arrow.rb +41 -0
- data/lacci/lib/shoes/drawables/button.rb +73 -0
- data/lacci/lib/shoes/{widgets → drawables}/check.rb +5 -4
- data/lacci/lib/shoes/{widgets → drawables}/document_root.rb +3 -3
- data/lacci/lib/shoes/{widgets → drawables}/edit_box.rb +6 -6
- data/lacci/lib/shoes/{widgets → drawables}/edit_line.rb +6 -6
- data/lacci/lib/shoes/{widgets → drawables}/flow.rb +6 -6
- data/lacci/lib/shoes/{widgets → drawables}/image.rb +6 -6
- data/lacci/lib/shoes/{widgets → drawables}/line.rb +7 -5
- data/lacci/lib/shoes/drawables/link.rb +34 -0
- data/lacci/lib/shoes/drawables/list_box.rb +56 -0
- data/lacci/lib/shoes/drawables/para.rb +118 -0
- data/lacci/lib/shoes/drawables/progress.rb +14 -0
- data/lacci/lib/shoes/drawables/radio.rb +33 -0
- data/lacci/lib/shoes/drawables/rect.rb +17 -0
- data/lacci/lib/shoes/{widgets → drawables}/shape.rb +6 -7
- data/lacci/lib/shoes/{widgets → drawables}/slot.rb +32 -20
- data/lacci/lib/shoes/{widgets → drawables}/span.rb +8 -7
- data/lacci/lib/shoes/{widgets → drawables}/stack.rb +6 -4
- data/lacci/lib/shoes/drawables/star.rb +50 -0
- data/lacci/lib/shoes/drawables/subscription_item.rb +93 -0
- data/lacci/lib/shoes/drawables/text_drawable.rb +63 -0
- data/lacci/lib/shoes/drawables/video.rb +16 -0
- data/lacci/lib/shoes/drawables/widget.rb +69 -0
- data/lacci/lib/shoes/drawables.rb +31 -0
- data/lacci/lib/shoes/errors.rb +28 -0
- data/lacci/lib/shoes/log.rb +2 -2
- data/lacci/lib/shoes/ruby_extensions.rb +15 -0
- data/lacci/lib/shoes/spacing.rb +2 -2
- data/lacci/lib/shoes-spec.rb +93 -0
- data/lacci/lib/shoes.rb +27 -7
- data/lacci/test/test_helper.rb +54 -0
- data/lacci/test/test_lacci.rb +12 -3
- data/lacci/test/test_shoes_errors.rb +49 -0
- data/lib/scarpe/cats_cradle.rb +81 -59
- data/lib/scarpe/errors.rb +77 -0
- data/lib/scarpe/evented_assertions.rb +50 -17
- data/lib/scarpe/shoes_spec.rb +181 -0
- data/lib/scarpe/version.rb +2 -2
- data/lib/scarpe/wv/app.rb +20 -20
- data/lib/scarpe/wv/arc.rb +4 -47
- data/lib/scarpe/wv/arrow.rb +9 -0
- data/lib/scarpe/wv/button.rb +7 -35
- data/lib/scarpe/wv/check.rb +3 -5
- data/lib/scarpe/wv/control_interface.rb +18 -20
- data/lib/scarpe/wv/document_root.rb +81 -4
- data/lib/scarpe/wv/{widget.rb → drawable.rb} +66 -43
- data/lib/scarpe/wv/edit_box.rb +4 -17
- data/lib/scarpe/wv/edit_line.rb +4 -18
- data/lib/scarpe/wv/flow.rb +2 -18
- data/lib/scarpe/wv/image.rb +8 -28
- data/lib/scarpe/wv/line.rb +3 -25
- data/lib/scarpe/wv/link.rb +3 -16
- data/lib/scarpe/wv/list_box.rb +6 -29
- data/lib/scarpe/wv/para.rb +11 -30
- data/lib/scarpe/wv/progress.rb +19 -0
- data/lib/scarpe/wv/radio.rb +9 -10
- data/lib/scarpe/wv/rect.rb +13 -0
- data/lib/scarpe/wv/shape.rb +3 -8
- data/lib/scarpe/wv/slot.rb +8 -25
- data/lib/scarpe/wv/span.rb +3 -27
- data/lib/scarpe/wv/stack.rb +2 -18
- data/lib/scarpe/wv/star.rb +3 -53
- data/lib/scarpe/wv/subscription_item.rb +38 -4
- data/lib/scarpe/wv/text_drawable.rb +32 -0
- data/lib/scarpe/wv/video.rb +15 -15
- data/lib/scarpe/wv/web_wrangler.rb +299 -329
- data/lib/scarpe/wv/webview_local_display.rb +48 -33
- data/lib/scarpe/wv/webview_relay_display.rb +12 -12
- data/lib/scarpe/wv/webview_relay_util.rb +7 -10
- data/lib/scarpe/wv/wv_display_worker.rb +2 -2
- data/lib/scarpe/wv.rb +45 -12
- data/lib/scarpe/wv_local.rb +1 -1
- data/lib/scarpe/wv_relay.rb +1 -1
- data/lib/scarpe.rb +1 -0
- data/logger/debug_web_wrangler.json +1 -1
- data/logger/scarpe_wv_test.json +1 -1
- data/scarpe-components/Gemfile.lock +86 -0
- data/scarpe-components/lib/scarpe/components/base64.rb +3 -7
- data/scarpe-components/lib/scarpe/components/calzini/alert.rb +49 -0
- data/scarpe-components/lib/scarpe/components/calzini/art_widgets.rb +203 -0
- data/scarpe-components/lib/scarpe/components/calzini/button.rb +39 -0
- data/scarpe-components/lib/scarpe/components/calzini/misc.rb +146 -0
- data/scarpe-components/lib/scarpe/components/calzini/para.rb +35 -0
- data/scarpe-components/lib/scarpe/components/calzini/slots.rb +155 -0
- data/scarpe-components/lib/scarpe/components/calzini/text_widgets.rb +65 -0
- data/scarpe-components/lib/scarpe/components/calzini.rb +149 -0
- data/scarpe-components/lib/scarpe/components/errors.rb +20 -0
- data/scarpe-components/lib/scarpe/components/file_helpers.rb +1 -0
- data/scarpe-components/lib/scarpe/components/html.rb +131 -0
- data/scarpe-components/lib/scarpe/components/minitest_export_reporter.rb +75 -0
- data/scarpe-components/lib/scarpe/components/minitest_import_runnable.rb +98 -0
- data/scarpe-components/lib/scarpe/components/minitest_result.rb +86 -0
- data/scarpe-components/lib/scarpe/components/modular_logger.rb +5 -5
- data/scarpe-components/lib/scarpe/components/print_logger.rb +9 -5
- data/scarpe-components/lib/scarpe/components/promises.rb +14 -14
- data/scarpe-components/lib/scarpe/components/segmented_file_loader.rb +36 -17
- data/scarpe-components/lib/scarpe/components/string_helpers.rb +10 -0
- data/scarpe-components/lib/scarpe/components/tiranti.rb +225 -0
- data/scarpe-components/lib/scarpe/components/unit_test_helpers.rb +45 -5
- data/scarpe-components/lib/scarpe/components/version.rb +2 -2
- data/scarpe-components/test/calzini/test_calzini_alert.rb +30 -0
- data/scarpe-components/test/calzini/test_calzini_art_drawables.rb +105 -0
- data/scarpe-components/test/calzini/test_calzini_button.rb +52 -0
- data/scarpe-components/test/calzini/test_calzini_misc.rb +115 -0
- data/scarpe-components/test/calzini/test_calzini_para.rb +37 -0
- data/scarpe-components/test/calzini/test_calzini_slots.rb +130 -0
- data/scarpe-components/test/calzini/test_calzini_text_drawables.rb +41 -0
- data/scarpe-components/test/mtr_data/exception.json +1 -0
- data/scarpe-components/test/mtr_data/fail_with_message.json +1 -0
- data/scarpe-components/test/mtr_data/skipped_no_message.json +1 -0
- data/scarpe-components/test/mtr_data/skipped_w_msg.json +1 -0
- data/scarpe-components/test/mtr_data/succeed_2_asserts.json +1 -0
- data/scarpe-components/test/test_dimensions.rb +26 -0
- data/scarpe-components/test/test_helper.rb +20 -0
- data/scarpe-components/test/test_html.rb +65 -0
- data/scarpe-components/test/test_minitest_result.rb +61 -0
- data/scarpe-components/test/test_promises.rb +5 -4
- data/scarpe-components/test/test_segmented_app_files.rb +8 -6
- data/scarpegen.rb +14 -14
- data/sig/scarpe.rbs +1 -1
- data/templates/basic_class_template.erb +13 -14
- data/templates/class_template_with_event_bind.erb +4 -4
- data/templates/class_template_with_shapes.erb +8 -17
- data/templates/example_template.erb +1 -1
- data/templates/module_template.erb +4 -4
- data/templates/webview_template.erb +3 -2
- metadata +113 -55
- data/examples/legacy/not_checked/shoes-contrib/elements/image-icon.rb +0 -3
- data/lacci/lib/shoes/widget.rb +0 -218
- data/lacci/lib/shoes/widgets/alert.rb +0 -19
- data/lacci/lib/shoes/widgets/arc.rb +0 -51
- data/lacci/lib/shoes/widgets/button.rb +0 -35
- data/lacci/lib/shoes/widgets/font.rb +0 -14
- data/lacci/lib/shoes/widgets/link.rb +0 -25
- data/lacci/lib/shoes/widgets/list_box.rb +0 -25
- data/lacci/lib/shoes/widgets/para.rb +0 -68
- data/lacci/lib/shoes/widgets/radio.rb +0 -35
- data/lacci/lib/shoes/widgets/star.rb +0 -44
- data/lacci/lib/shoes/widgets/subscription_item.rb +0 -60
- data/lacci/lib/shoes/widgets/text_widget.rb +0 -51
- data/lacci/lib/shoes/widgets/video.rb +0 -15
- data/lacci/lib/shoes/widgets.rb +0 -29
- data/lib/scarpe/wv/alert.rb +0 -66
- data/lib/scarpe/wv/background.rb +0 -27
- data/lib/scarpe/wv/border.rb +0 -24
- data/lib/scarpe/wv/control_interface_test.rb +0 -238
- data/lib/scarpe/wv/dimensions.rb +0 -22
- data/lib/scarpe/wv/font.rb +0 -36
- data/lib/scarpe/wv/html.rb +0 -108
- data/lib/scarpe/wv/spacing.rb +0 -41
- data/lib/scarpe/wv/text_widget.rb +0 -30
- /data/examples/legacy/not_checked/{expert → shoes-contrib/basic}/definr.rb +0 -0
- /data/examples/legacy/not_checked/{expert → shoes-contrib/basic}/funnies.rb +0 -0
- /data/examples/legacy/not_checked/shoes-contrib/{elements → basic}/list_box-select-class.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/basic → working/simple}/basic-edit-box.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/basic-fps.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/border-cat.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/check-mate.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/manipulation → working/simple}/clear-slot.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/basic → working/simple}/clock.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/basic → working/simple}/gradient-shoes.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/basic → working/simple}/list_box-shape-report.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/list_box.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/phat-button.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib → working}/simple/simple-calc.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/position → working/simple}/stack-width.rb +0 -0
- /data/examples/legacy/{not_checked/shoes-contrib/elements → working/simple}/width-introspec.rb +0 -0
@@ -1,13 +1,15 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
3
|
+
class Shoes
|
4
4
|
class Stack < Shoes::Slot
|
5
5
|
include Shoes::Background
|
6
6
|
include Shoes::Border
|
7
7
|
include Shoes::Spacing
|
8
8
|
|
9
9
|
# TODO: sort out various margin and padding properties, including putting stuff into spacing
|
10
|
-
|
10
|
+
shoes_styles :width, :height, :scroll
|
11
|
+
|
12
|
+
shoes_events # No Stack-specific events yet
|
11
13
|
|
12
14
|
def initialize(width: nil, height: nil, margin: nil, padding: nil, scroll: false, margin_top: nil, margin_bottom: nil, margin_left: nil,
|
13
15
|
margin_right: nil, **options, &block)
|
@@ -16,8 +18,8 @@ module Shoes
|
|
16
18
|
|
17
19
|
super
|
18
20
|
|
19
|
-
|
20
|
-
# Create the display-side
|
21
|
+
create_display_drawable
|
22
|
+
# Create the display-side drawable *before* running the block, which will add child drawables with their display drawables
|
21
23
|
Shoes::App.instance.with_slot(self, &block) if block_given?
|
22
24
|
end
|
23
25
|
end
|
@@ -0,0 +1,50 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Shoes
|
4
|
+
class Star < Shoes::Drawable
|
5
|
+
shoes_styles :left, :top, :draw_context
|
6
|
+
|
7
|
+
shoes_style(:points) { |val| convert_to_integer(val, "points") }
|
8
|
+
shoes_style(:outer) { |val| convert_to_float(val, "outer") }
|
9
|
+
shoes_style(:inner) { |val| convert_to_float(val, "inner") }
|
10
|
+
|
11
|
+
shoes_events # No Star-specific events yet
|
12
|
+
|
13
|
+
def initialize(left, top, points = 10, outer = 100, inner = 50)
|
14
|
+
super
|
15
|
+
self.left = left
|
16
|
+
self.top = top
|
17
|
+
self.points = points
|
18
|
+
self.outer = outer
|
19
|
+
self.inner = inner
|
20
|
+
|
21
|
+
@draw_context = Shoes::App.instance.current_draw_context
|
22
|
+
|
23
|
+
create_display_drawable
|
24
|
+
end
|
25
|
+
|
26
|
+
def self.convert_to_integer(value, attribute_name)
|
27
|
+
begin
|
28
|
+
value = Integer(value)
|
29
|
+
raise Shoes::Errors::InvalidAttributeValueError, "Negative num '#{value}' not allowed for attribute '#{attribute_name}'" if value < 0
|
30
|
+
|
31
|
+
value
|
32
|
+
rescue ArgumentError
|
33
|
+
error_message = "Invalid value '#{value}' provided for attribute '#{attribute_name}'. The value should be a number."
|
34
|
+
raise Shoes::Errors::InvalidAttributeValueError, error_message
|
35
|
+
end
|
36
|
+
end
|
37
|
+
|
38
|
+
def self.convert_to_float(value, attribute_name)
|
39
|
+
begin
|
40
|
+
value = Float(value)
|
41
|
+
raise Shoes::Errors::InvalidAttributeValueError, "Negative num '#{value}' not allowed for attribute '#{attribute_name}'" if value < 0
|
42
|
+
|
43
|
+
value
|
44
|
+
rescue ArgumentError
|
45
|
+
error_message = "Invalid value '#{value}' provided for attribute '#{attribute_name}'. The value should be a number."
|
46
|
+
raise Shoes::Errors::InvalidAttributeValueError, error_message
|
47
|
+
end
|
48
|
+
end
|
49
|
+
end
|
50
|
+
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# Certain Shoes calls like motion and keydown are basically an
|
4
|
+
# event subscription, with no other visible presence. However,
|
5
|
+
# they have a place in the drawable tree and can be deleted.
|
6
|
+
#
|
7
|
+
# Depending on the display library they may not have any
|
8
|
+
# direct visual (or similar) presence there either.
|
9
|
+
#
|
10
|
+
# Inheriting from Drawable gives these a parent slot and a
|
11
|
+
# linkable_id automatically.
|
12
|
+
#
|
13
|
+
# Events not yet implemented: start, finish events for slots -
|
14
|
+
# start is first draw, finish is drawable destroyed
|
15
|
+
class Shoes::SubscriptionItem < Shoes::Drawable
|
16
|
+
shoes_styles :shoes_api_name, :args
|
17
|
+
shoes_events :animate, :every, :timer, :hover, :leave, :motion, :click, :release, :keypress
|
18
|
+
|
19
|
+
def initialize(args: [], shoes_api_name:, &block)
|
20
|
+
super
|
21
|
+
|
22
|
+
@callback = block
|
23
|
+
|
24
|
+
case shoes_api_name
|
25
|
+
when "animate"
|
26
|
+
@unsub_id = bind_self_event("animate") do |frame|
|
27
|
+
@callback.call(frame)
|
28
|
+
end
|
29
|
+
when "every"
|
30
|
+
@unsub_id = bind_self_event("every") do |count|
|
31
|
+
@callback.call(count)
|
32
|
+
end
|
33
|
+
when "timer"
|
34
|
+
@unsub_id = bind_self_event("timer") do
|
35
|
+
@callback.call
|
36
|
+
end
|
37
|
+
when "hover"
|
38
|
+
# Hover passes the Shoes drawable as the block param
|
39
|
+
@unsub_id = bind_self_event("hover") do
|
40
|
+
@callback&.call(self)
|
41
|
+
end
|
42
|
+
when "leave"
|
43
|
+
# Leave passes the Shoes drawable as the block param
|
44
|
+
@unsub_id = bind_self_event("leave") do
|
45
|
+
@callback&.call(self)
|
46
|
+
end
|
47
|
+
when "motion"
|
48
|
+
# Shoes sends back x, y, mods as the args.
|
49
|
+
# Shoes3 uses the strings "control" "shift" and
|
50
|
+
# "control_shift" as the mods arg.
|
51
|
+
@unsub_id = bind_self_event("motion") do |x, y, ctrl_key, shift_key, **_kwargs|
|
52
|
+
mods = [ctrl_key ? "control" : nil, shift_key ? "shift" : nil].compact.join("_")
|
53
|
+
@callback&.call(x, y, mods)
|
54
|
+
end
|
55
|
+
when "click"
|
56
|
+
# Click has block params button, left, top
|
57
|
+
# button is the button number, left and top are coords
|
58
|
+
@unsub_id = bind_self_event("click") do |button, x, y, **_kwargs|
|
59
|
+
@callback&.call(button, x, y)
|
60
|
+
end
|
61
|
+
when "release"
|
62
|
+
# Click has block params button, left, top
|
63
|
+
# button is the button number, left and top are coords
|
64
|
+
@unsub_id = bind_self_event("release") do |button, x, y, **_kwargs|
|
65
|
+
@callback&.call(button, x, y)
|
66
|
+
end
|
67
|
+
when "keypress"
|
68
|
+
# Keypress passes the key string or symbol to the handler
|
69
|
+
# Do anything special for serialisation here?
|
70
|
+
@unsub_id = bind_self_event("keypress") do |key|
|
71
|
+
@callback&.call(key)
|
72
|
+
end
|
73
|
+
else
|
74
|
+
raise "Unknown Shoes event #{shoes_api_name.inspect} passed to SubscriptionItem!"
|
75
|
+
end
|
76
|
+
|
77
|
+
@unsub_id = bind_self_event(shoes_api_name) do |*args|
|
78
|
+
@callback&.call(*args)
|
79
|
+
end
|
80
|
+
|
81
|
+
# This won't create a visible display drawable, but will turn into
|
82
|
+
# an invisible drawable and a stream of events.
|
83
|
+
create_display_drawable
|
84
|
+
end
|
85
|
+
|
86
|
+
def destroy
|
87
|
+
# TODO: we need a better way to do this automatically. See https://github.com/scarpe-team/scarpe/issues/291
|
88
|
+
unsub_shoes_event(@unsub_id) if @unsub_id
|
89
|
+
@unsub_id = nil
|
90
|
+
|
91
|
+
super
|
92
|
+
end
|
93
|
+
end
|
@@ -0,0 +1,63 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Shoes
|
4
|
+
# TextDrawable is the parent class of various classes of
|
5
|
+
# text that can go inside a para. This includes normal
|
6
|
+
# text, but also links, italic text, bold text, etc.
|
7
|
+
class TextDrawable < Shoes::Drawable
|
8
|
+
class << self
|
9
|
+
# rubocop:disable Lint/MissingSuper
|
10
|
+
def inherited(subclass)
|
11
|
+
Shoes::Drawable.drawable_classes ||= []
|
12
|
+
Shoes::Drawable.drawable_classes << subclass
|
13
|
+
|
14
|
+
Shoes::Drawable.drawable_default_styles ||= {}
|
15
|
+
Shoes::Drawable.drawable_default_styles[subclass] = {}
|
16
|
+
end
|
17
|
+
# rubocop:enable Lint/MissingSuper
|
18
|
+
end
|
19
|
+
|
20
|
+
shoes_events # No TextDrawable-specific events yet
|
21
|
+
end
|
22
|
+
|
23
|
+
class << self
|
24
|
+
def default_text_drawable_with(element)
|
25
|
+
class_name = element.capitalize
|
26
|
+
|
27
|
+
drawable_class = Class.new(Shoes::TextDrawable) do
|
28
|
+
# Can we just change content to text to match the Shoes API?
|
29
|
+
shoes_style :content
|
30
|
+
|
31
|
+
def initialize(content)
|
32
|
+
super
|
33
|
+
|
34
|
+
@content = content
|
35
|
+
|
36
|
+
create_display_drawable
|
37
|
+
end
|
38
|
+
|
39
|
+
def text
|
40
|
+
self.content
|
41
|
+
end
|
42
|
+
|
43
|
+
def to_s
|
44
|
+
self.content
|
45
|
+
end
|
46
|
+
|
47
|
+
def text=(new_text)
|
48
|
+
self.content = new_text
|
49
|
+
end
|
50
|
+
end
|
51
|
+
Shoes.const_set class_name, drawable_class
|
52
|
+
drawable_class.class_eval do
|
53
|
+
shoes_style :content
|
54
|
+
|
55
|
+
shoes_events # No specific events
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
Shoes.default_text_drawable_with(:code)
|
62
|
+
Shoes.default_text_drawable_with(:em)
|
63
|
+
Shoes.default_text_drawable_with(:strong)
|
@@ -0,0 +1,16 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Shoes
|
4
|
+
class Video < Shoes::Drawable
|
5
|
+
shoes_styles :url
|
6
|
+
shoes_events # No specific events yet
|
7
|
+
|
8
|
+
def initialize(url)
|
9
|
+
super
|
10
|
+
@url = url
|
11
|
+
create_display_drawable
|
12
|
+
end
|
13
|
+
|
14
|
+
# other methods
|
15
|
+
end
|
16
|
+
end
|
@@ -0,0 +1,69 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
# A Shoes::Widget is mostly a Slot (related to the
|
4
|
+
# old Shoes concepts of Canvas) that creates drawables
|
5
|
+
# inside itself. When a subclass of Widget is created,
|
6
|
+
# it adds a method to create new objects of that type
|
7
|
+
# on Shoes::App and all Shoes slots.
|
8
|
+
#
|
9
|
+
# The hardest part with a Shoes::Widget is that it
|
10
|
+
# should work fine even if initialize() doesn't call
|
11
|
+
# super, which would make it hard to set up a Shoes
|
12
|
+
# linkable_id, create a display widget, etc.
|
13
|
+
#
|
14
|
+
# It would be possible to add an extra method to set
|
15
|
+
# these up and call it on every created drawable
|
16
|
+
# in case a Widget's initialize method doesn't call
|
17
|
+
# super, which happens quite often. But then we wouldn't
|
18
|
+
# support automatic setting of styles (e.g. padding)
|
19
|
+
# for the widget object itself, which is mostly a Flow.
|
20
|
+
# We also couldn't support default styles -- I can't tell
|
21
|
+
# whether Shoes supports these things either.
|
22
|
+
#
|
23
|
+
# But there's another way to do all of this. When a
|
24
|
+
# subclass of Widget defines an initialize method,
|
25
|
+
# we can catch the method_added hook, save a copy of
|
26
|
+
# that initialize method, and substitute our own
|
27
|
+
# initialize that calls super. We have to be a little
|
28
|
+
# careful -- if the widget's initialize *does* call
|
29
|
+
# super that shouldn't be an error. But that's
|
30
|
+
# workable by defining an extra method with the
|
31
|
+
# copied-method name that does nothing.
|
32
|
+
|
33
|
+
##### TODO: when this is subclassed, grab :initialize out
|
34
|
+
# of the subclass, put it into :initialize_widget, and
|
35
|
+
# replace with an initialize that creates the display
|
36
|
+
# widget propertly, sets the linkable_id, etc.
|
37
|
+
|
38
|
+
class Shoes::Widget < Shoes::Slot
|
39
|
+
include Shoes::Background
|
40
|
+
include Shoes::Border
|
41
|
+
include Shoes::Spacing
|
42
|
+
|
43
|
+
shoes_events
|
44
|
+
|
45
|
+
def self.method_added(name)
|
46
|
+
# We're only looking for the initialize() method, and only on subclasses
|
47
|
+
# of Shoes::Widget, not Shoes::Widget itself.
|
48
|
+
return if self == ::Shoes::Widget || name != :initialize
|
49
|
+
|
50
|
+
# Need to avoid infinite adding of initialize() if we're re-adding the default initialize
|
51
|
+
return if @midway_through_adding_initialize
|
52
|
+
|
53
|
+
# Take the user-provided initialize method and save a copy named __widget_initialize
|
54
|
+
alias_method :__widget_initialize, :initialize
|
55
|
+
|
56
|
+
# And add the default initialize back where it belongs
|
57
|
+
@midway_through_adding_initialize = true
|
58
|
+
define_method(:initialize) do |*args, **kwargs, &block|
|
59
|
+
super(*args, **kwargs, &block)
|
60
|
+
@options = kwargs
|
61
|
+
create_display_drawable
|
62
|
+
__widget_initialize(*args, **kwargs, &block)
|
63
|
+
|
64
|
+
# Do Widgets do this?
|
65
|
+
Shoes::App.instance.with_slot(self, &block) if block
|
66
|
+
end
|
67
|
+
@midway_through_adding_initialize = false
|
68
|
+
end
|
69
|
+
end
|
@@ -0,0 +1,31 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require "shoes/drawables/slot"
|
4
|
+
require "shoes/drawables/stack"
|
5
|
+
require "shoes/drawables/flow"
|
6
|
+
require "shoes/drawables/widget"
|
7
|
+
require "shoes/drawables/document_root"
|
8
|
+
|
9
|
+
require "shoes/drawables/text_drawable"
|
10
|
+
|
11
|
+
require "shoes/drawables/subscription_item"
|
12
|
+
|
13
|
+
require "shoes/drawables/arc"
|
14
|
+
require "shoes/drawables/line"
|
15
|
+
require "shoes/drawables/rect"
|
16
|
+
require "shoes/drawables/shape"
|
17
|
+
require "shoes/drawables/star"
|
18
|
+
require "shoes/drawables/arrow"
|
19
|
+
|
20
|
+
require "shoes/drawables/button"
|
21
|
+
require "shoes/drawables/check"
|
22
|
+
require "shoes/drawables/edit_box"
|
23
|
+
require "shoes/drawables/edit_line"
|
24
|
+
require "shoes/drawables/image"
|
25
|
+
require "shoes/drawables/link"
|
26
|
+
require "shoes/drawables/list_box"
|
27
|
+
require "shoes/drawables/para"
|
28
|
+
require "shoes/drawables/radio"
|
29
|
+
require "shoes/drawables/span"
|
30
|
+
require "shoes/drawables/video"
|
31
|
+
require "shoes/drawables/progress"
|
@@ -0,0 +1,28 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Shoes; end
|
4
|
+
class Shoes::Errors
|
5
|
+
class InvalidAttributeValueError < Shoes::Error; end
|
6
|
+
|
7
|
+
class TooManyInstancesError < Shoes::Error; end
|
8
|
+
|
9
|
+
class NoSuchListItemError < Shoes::Error; end
|
10
|
+
|
11
|
+
class DuplicateCreateDrawableError < Shoes::Error; end
|
12
|
+
|
13
|
+
class MultipleDrawablesFoundError < Shoes::Error; end
|
14
|
+
|
15
|
+
class NoDrawablesFoundError < Shoes::Error; end
|
16
|
+
|
17
|
+
class NoSuchStyleError < Shoes::Error; end
|
18
|
+
|
19
|
+
class NoLinkableIdError < Shoes::Error; end
|
20
|
+
|
21
|
+
class BadLinkableIdError < Shoes::Error; end
|
22
|
+
|
23
|
+
class UnregisteredShoesEvent < Shoes::Error; end
|
24
|
+
|
25
|
+
class SingletonError < Shoes::Error; end
|
26
|
+
|
27
|
+
class MultipleShoesSpecRunsError < Shoes::Error; end
|
28
|
+
end
|
data/lacci/lib/shoes/log.rb
CHANGED
@@ -8,7 +8,7 @@
|
|
8
8
|
# If used alone, this will fail because the @instance is nil. It needs
|
9
9
|
# an implementation to be plugged in.
|
10
10
|
|
11
|
-
|
11
|
+
class Shoes
|
12
12
|
LOG_LEVELS = [:debug, :info, :warn, :error, :fatal].freeze
|
13
13
|
|
14
14
|
# Include this module to get a @log instance variable to log as your
|
@@ -28,7 +28,7 @@ module Shoes
|
|
28
28
|
attr_reader :current_log_config
|
29
29
|
|
30
30
|
def instance=(impl_object)
|
31
|
-
raise(Shoes::
|
31
|
+
raise(Shoes::Errors::TooManyInstancesError, "Already have an instance for Shoes::Log!") if @instance
|
32
32
|
|
33
33
|
@instance = impl_object
|
34
34
|
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
|
2
|
+
class Range
|
3
|
+
def rand
|
4
|
+
conv = (Integer === self.end && Integer === self.begin ? :to_i : :to_f)
|
5
|
+
((Kernel.rand * (self.end - self.begin)) + self.begin).send(conv)
|
6
|
+
end
|
7
|
+
end
|
8
|
+
|
9
|
+
unless Time.respond_to? :today
|
10
|
+
def Time.today
|
11
|
+
t = Time.now
|
12
|
+
t - (t.to_i % 86_400)
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
data/lacci/lib/shoes/spacing.rb
CHANGED
@@ -1,9 +1,9 @@
|
|
1
1
|
# frozen_string_literal: true
|
2
2
|
|
3
|
-
|
3
|
+
class Shoes
|
4
4
|
module Spacing
|
5
5
|
def self.included(includer)
|
6
|
-
includer.
|
6
|
+
includer.shoes_styles :margin, :padding, :margin_top, :margin_left, :margin_right, :margin_bottom, :options
|
7
7
|
end
|
8
8
|
end
|
9
9
|
end
|
@@ -0,0 +1,93 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
class Shoes
|
4
|
+
# A Scarpe-compatible display service can set Shoes::Spec.instance to
|
5
|
+
# a ShoesSpec testing class, and use it to run Shoes-Spec code.
|
6
|
+
# A Shoes application should never do this. It's intended to be used
|
7
|
+
# by display services.
|
8
|
+
module Spec
|
9
|
+
def self.instance
|
10
|
+
@instance
|
11
|
+
end
|
12
|
+
|
13
|
+
def self.instance=(spec_inst)
|
14
|
+
if @instance && @instance != spec_inst
|
15
|
+
raise "Lacci can only use a single ShoesSpec implementation at one time!"
|
16
|
+
end
|
17
|
+
|
18
|
+
@instance = spec_inst
|
19
|
+
end
|
20
|
+
end
|
21
|
+
|
22
|
+
# ShoesSpec testing objects can optionally inherit from this object,
|
23
|
+
# which shows the ShoesSpec testing API.
|
24
|
+
#
|
25
|
+
# @see {Shoes::Spec.instance=}
|
26
|
+
class SpecInstance
|
27
|
+
# Once a Shoes app has been created, this method can be called to
|
28
|
+
# execute Shoes-Spec testing code for that application. Shoes-Spec
|
29
|
+
# uses Minitest for most of its APIs, and Minitest generally reports
|
30
|
+
# results with a class name and test name. If those aren't passed
|
31
|
+
# explicitly, the SpecInstance can choose reasonable defaults.
|
32
|
+
#
|
33
|
+
# The test code should be set up to run automatically from the
|
34
|
+
# display service's existing hooks. For instance, the code might
|
35
|
+
# run in response to the first heartbeat, if the display service
|
36
|
+
# uses heartbeats.
|
37
|
+
#
|
38
|
+
# The test code will export assertion data in its native format.
|
39
|
+
# Multiple display services choose to use the Scarpe-Component
|
40
|
+
# for Minitest data export, which is straightforward to import
|
41
|
+
# into the Shoes-Spec test harness.
|
42
|
+
#
|
43
|
+
# @param code [String] the ShoesSpec code to execute
|
44
|
+
# @param class_name [String|NilClass] the Minitest class name for reporting or nil
|
45
|
+
# @param test_name [String|NilClass] the Minitest test name for reporting or nil
|
46
|
+
# @return [void]
|
47
|
+
def run_shoes_spec_test_code(code, class_name: nil, test_name: nil)
|
48
|
+
raise "Child class should override this!"
|
49
|
+
end
|
50
|
+
end
|
51
|
+
|
52
|
+
# ShoesSpec instances support finder methods like button() that return
|
53
|
+
# a proxy to the corresponding drawable. Those proxies should support
|
54
|
+
# standard Shoes::Drawable methods, including the ones appropriate to
|
55
|
+
# the same drawable object. They should also support certain other
|
56
|
+
# testing-specific methods like "trigger_click" that are used to
|
57
|
+
# simulate display-side events during testing.
|
58
|
+
#
|
59
|
+
# Keep in mind that a proxy will often be in a different process from
|
60
|
+
# the Shoes app. So the proxy can't portably return the object or
|
61
|
+
# display object, though it could possibly return another proxy for such
|
62
|
+
# a thing.
|
63
|
+
class SpecProxy
|
64
|
+
# The proxy will have finder methods for all drawables, such as
|
65
|
+
# button(), edit_line(), etc. How to document those?
|
66
|
+
|
67
|
+
# Trigger a click on a button or button-like drawable. Not every
|
68
|
+
# drawable will support this operation.
|
69
|
+
def trigger_click()
|
70
|
+
raise "Child class should override this!"
|
71
|
+
end
|
72
|
+
|
73
|
+
# Trigger a hover over a hoverable drawable. Not every
|
74
|
+
# drawable will support this operation. A drawable that supports
|
75
|
+
# hover should support leave and vice-versa.
|
76
|
+
def trigger_hover()
|
77
|
+
raise "Child class should override this!"
|
78
|
+
end
|
79
|
+
|
80
|
+
# Trigger ending hover over a hoverable drawable. Not every
|
81
|
+
# drawable will support this operation. A drawable that supports
|
82
|
+
# hover should support leave and vice-versa.
|
83
|
+
def trigger_leave()
|
84
|
+
raise "Child class should override this!"
|
85
|
+
end
|
86
|
+
|
87
|
+
# Trigger a change in value for a drawable like a list_box
|
88
|
+
# with multiple values.
|
89
|
+
def trigger_change(value)
|
90
|
+
raise "Child class should override this!"
|
91
|
+
end
|
92
|
+
end
|
93
|
+
end
|
data/lacci/lib/shoes.rb
CHANGED
@@ -12,38 +12,53 @@ if RUBY_VERSION[0..2] < "3.2"
|
|
12
12
|
exit(-1)
|
13
13
|
end
|
14
14
|
|
15
|
-
|
15
|
+
class Shoes; end
|
16
16
|
class Shoes::Error < StandardError; end
|
17
|
+
require_relative "shoes/errors"
|
17
18
|
|
18
19
|
require_relative "shoes/constants"
|
20
|
+
require_relative "shoes/ruby_extensions"
|
19
21
|
|
20
22
|
# Shoes adds some top-level methods and constants that can be used everywhere. Kernel is where they go.
|
21
23
|
module Kernel
|
22
24
|
include Shoes::Constants
|
23
25
|
end
|
24
26
|
|
25
|
-
require_relative "shoes/log"
|
26
27
|
require_relative "shoes/display_service"
|
28
|
+
|
29
|
+
# Pre-declare classes that get referenced outside their own require file
|
30
|
+
class Shoes::Drawable < Shoes::Linkable; end
|
31
|
+
class Shoes::Slot < Shoes::Drawable; end
|
32
|
+
class Shoes::Widget < Shoes::Slot; end
|
33
|
+
|
34
|
+
require_relative "shoes/log"
|
27
35
|
require_relative "shoes/colors"
|
28
36
|
|
37
|
+
require_relative "shoes/builtins"
|
38
|
+
|
29
39
|
require_relative "shoes/background"
|
30
40
|
require_relative "shoes/border"
|
31
41
|
require_relative "shoes/spacing"
|
32
42
|
|
33
|
-
require_relative "shoes/
|
43
|
+
require_relative "shoes/drawable"
|
34
44
|
require_relative "shoes/app"
|
35
|
-
require_relative "shoes/
|
45
|
+
require_relative "shoes/drawables"
|
36
46
|
|
37
47
|
require_relative "shoes/download"
|
38
48
|
|
49
|
+
# No easy way to tell at this point whether
|
50
|
+
# we will later load Shoes-Spec code, e.g.
|
51
|
+
# by running a segmented app with test code.
|
52
|
+
require_relative "shoes-spec"
|
53
|
+
|
39
54
|
# The module containing Shoes in all its glory.
|
40
55
|
# Shoes is a platform-independent GUI library, designed to create
|
41
56
|
# small visual applications in Ruby.
|
42
57
|
#
|
43
|
-
|
58
|
+
class Shoes
|
44
59
|
class << self
|
45
60
|
# Creates a Shoes app with a new window. The block parameter is used to create
|
46
|
-
#
|
61
|
+
# drawables and set up handlers. Arguments are passed to Shoes::App.new internally.
|
47
62
|
#
|
48
63
|
# @incompatibility In Shoes3, this method will return normally.
|
49
64
|
# In Scarpe, after the block is executed, the method will not return and Scarpe
|
@@ -81,11 +96,16 @@ module Shoes
|
|
81
96
|
# more loaders, a Lacci-based display library can accept new file formats as
|
82
97
|
# well, not just raw Shoes .rb files.
|
83
98
|
#
|
84
|
-
# @param
|
99
|
+
# @param relative_path [String] The current-dir-relative path to the file
|
85
100
|
# @return [void]
|
86
101
|
# @see Shoes.add_file_loader
|
87
102
|
def run_app(relative_path)
|
88
103
|
path = File.expand_path relative_path
|
104
|
+
dir = File.dirname(path)
|
105
|
+
|
106
|
+
# Shoes assumes we're starting from the app code's path
|
107
|
+
Dir.chdir(dir)
|
108
|
+
|
89
109
|
loaded = false
|
90
110
|
file_loaders.each do |loader|
|
91
111
|
if loader.call(path)
|
data/lacci/test/test_helper.rb
CHANGED
@@ -3,7 +3,61 @@
|
|
3
3
|
$LOAD_PATH.unshift File.expand_path("../lib", __dir__)
|
4
4
|
require "shoes"
|
5
5
|
|
6
|
+
require "scarpe/components/unit_test_helpers"
|
7
|
+
|
6
8
|
require "minitest/autorun"
|
7
9
|
|
8
10
|
require "minitest/reporters"
|
9
11
|
Minitest::Reporters.use! [Minitest::Reporters::SpecReporter.new]
|
12
|
+
|
13
|
+
# For testing Lacci, it's kind of silly to start a Webview application.
|
14
|
+
# They're slow, unreliable and memory-hungry. So instead we start a
|
15
|
+
# Niente do-nothing application for a simple API test. It's a lot like
|
16
|
+
# mocking.
|
17
|
+
class NienteTest < Minitest::Test
|
18
|
+
include ::Scarpe::Test::Helpers
|
19
|
+
|
20
|
+
SCARPE_EXE = File.expand_path("../../exe/scarpe", __dir__)
|
21
|
+
|
22
|
+
def run_test_niente_code(
|
23
|
+
scarpe_app_code,
|
24
|
+
test_extension: ".rb",
|
25
|
+
**opts
|
26
|
+
)
|
27
|
+
with_tempfile(["scarpe_test_app", test_extension], scarpe_app_code) do |test_app_location|
|
28
|
+
run_test_niente_app(test_app_location, **opts)
|
29
|
+
end
|
30
|
+
end
|
31
|
+
|
32
|
+
def run_test_niente_app(
|
33
|
+
test_app_location,
|
34
|
+
app_test_code: "",
|
35
|
+
timeout: 5.0,
|
36
|
+
class_name: self.class,
|
37
|
+
method_name: self.name,
|
38
|
+
display_service: "niente"
|
39
|
+
)
|
40
|
+
with_tempfiles([
|
41
|
+
#["scarpe_log_config.json", JSON.dump(log_config_for_test)],
|
42
|
+
[["shoes_spec_code", ".rb"], app_test_code],
|
43
|
+
]) do |shoes_spec_path,_|
|
44
|
+
system(
|
45
|
+
"LOCALAPPDATA=\"#{Dir.tmpdir}\" " +
|
46
|
+
"SHOES_SPEC_TEST=\"#{shoes_spec_path}\" " +
|
47
|
+
"SCARPE_DISPLAY_SERVICE=\"#{display_service}\" " +
|
48
|
+
"SHOES_MINITEST_EXPORT_FILE=niente_test.json " +
|
49
|
+
"SHOES_MINITEST_CLASS_NAME=\"#{class_name}\" " +
|
50
|
+
"SHOES_MINITEST_METHOD_NAME=\"#{method_name}\" " +
|
51
|
+
"ruby #{SCARPE_EXE} --dev #{test_app_location}"
|
52
|
+
)
|
53
|
+
end
|
54
|
+
|
55
|
+
# Check if the process exited normally or crashed (segfault, failure, timeout)
|
56
|
+
unless $?.success?
|
57
|
+
assert(false, "Niente app failed with exit code: #{$?.exitstatus}")
|
58
|
+
return
|
59
|
+
end
|
60
|
+
|
61
|
+
|
62
|
+
end
|
63
|
+
end
|
data/lacci/test/test_lacci.rb
CHANGED
@@ -2,8 +2,17 @@
|
|
2
2
|
|
3
3
|
require_relative "test_helper"
|
4
4
|
|
5
|
-
class TestLacci <
|
6
|
-
def
|
7
|
-
|
5
|
+
class TestLacci < NienteTest
|
6
|
+
def test_simple_button_click
|
7
|
+
run_test_niente_code(<<~SHOES_APP, app_test_code: <<~SHOES_SPEC)
|
8
|
+
Shoes.app do
|
9
|
+
@b = button "OK" do
|
10
|
+
@b.text = "Yup"
|
11
|
+
end
|
12
|
+
end
|
13
|
+
SHOES_APP
|
14
|
+
button().trigger_click
|
15
|
+
assert_equal "Yup", button().text
|
16
|
+
SHOES_SPEC
|
8
17
|
end
|
9
18
|
end
|