From 12d21d42de95584f95baa4c984c20a50fb442a27 Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Mon, 26 Oct 2020 16:40:58 +0100 Subject: [PATCH 01/30] try out different interaction model --- src/makielayout/MakieLayout.jl | 2 +- src/makielayout/lobjects/laxis.jl | 100 +++++++++++++++++++++++++++--- src/makielayout/types.jl | 4 ++ 3 files changed, 98 insertions(+), 8 deletions(-) diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index c59dd631e..d2b0586c9 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -43,9 +43,9 @@ function __init__() end include("geometrybasics_extension.jl") +include("mousestatemachine.jl") include("types.jl") include("helpers.jl") -include("mousestatemachine.jl") include("ticklocators/linear.jl") include("ticklocators/wilkinson.jl") include("defaultattributes.jl") diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index 3e6d6574a..23740019c 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -294,15 +294,29 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) # layout layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] + mousestate = addmousestate!(scene) + + interactions = AbstractInteraction[] + la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, - layoutobservables, attrs, block_limit_linking, decorations) + layoutobservables, attrs, block_limit_linking, decorations, mousestate, interactions) + + + on(mousestate) do state + for i in la.interactions + process_interaction(i, state, la) + end + end + + add_limit_reset!(la) + add_rectanglezoom!(la) - # add action that resets limits on ctrl + click - add_reset_limits!(la) - # add action that allows zooming using mouse scrolling - add_zoom!(la) - # add action that allows panning using a mouse button - add_pan!(la) + # # add action that resets limits on ctrl + click + # add_reset_limits!(la) + # # add action that allows zooming using mouse scrolling + # add_zoom!(la) + # # add action that allows panning using a mouse button + # add_pan!(la) # compute limits that adhere to the limit aspect ratio whenever the targeted @@ -314,6 +328,78 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) la end +####################################### +function process_interaction(@nospecialize args...) + # do nothing in the default case +end + +mutable struct RectangleZoom <: AbstractInteraction + from::Union{Nothing, Point2f0} + to::Union{Nothing, Point2f0} + rectnode::Observable{FRect2D} + poly::Union{Poly, Nothing} +end + +function add_rectanglezoom!(ax) + zoom = RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing) + push!(ax.interactions, zoom) + nothing +end + +function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDragStart}, ax) + r.from = mousestate.prev + r.to = mousestate.pos + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] + nothing +end + +function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDrag}, ax) + r.to = mousestate.pos + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + nothing +end + +function positivize(r::FRect2D) + negwidths = r.widths .< 0 + newori = ifelse.(negwidths, r.origin .+ r.widths, r.origin) + newwidths = ifelse.(negwidths, -r.widths, r.widths) + FRect2D(newori, newwidths) +end + +function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDragStop}, ax) + newlims = positivize(r.rectnode[]) + if !(0 in widths(newlims)) + ax.targetlimits[] = newlims + end + + if !isnothing(r.poly) + delete!(ax.scene, r.poly) + r.poly = nothing + end + nothing +end + + + +struct LimitReset <: AbstractInteraction end + +function add_limit_reset!(ax) + reset = LimitReset() + push!(ax.interactions, reset) + nothing +end + +function process_interaction(l::LimitReset, mousestate::MouseState{MouseLeftClick}, ax) + if ispressed(ax.scene, Keyboard.left_control) + autolimits!(ax) + end + nothing +end + +####################################### + + function AbstractPlotting.plot!( la::LAxis, P::AbstractPlotting.PlotFunc, attributes::AbstractPlotting.Attributes, args...; diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 4ab1108df..a883a15b6 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -46,6 +46,8 @@ mutable struct LineAxis ticklabels::Node{Vector{String}} end +abstract type AbstractInteraction end + abstract type LObject end mutable struct LAxis <: AbstractPlotting.AbstractScene @@ -58,6 +60,8 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene attributes::Attributes block_limit_linking::Node{Bool} decorations::Dict{Symbol, Any} + mousestate::Observable{MouseState} + interactions::Vector{AbstractInteraction} end mutable struct LColorbar <: LObject From fbe8215bc75776e646c61774441acd5a0ddc471b Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Tue, 27 Oct 2020 10:28:10 +0100 Subject: [PATCH 02/30] switch mousestate to mouseevents enum for less complex type system --- src/makielayout/MakieLayout.jl | 2 +- src/makielayout/lobjects/laxis.jl | 71 ++++++----- src/makielayout/lobjects/lbutton.jl | 2 +- src/makielayout/lobjects/lmenu.jl | 4 +- src/makielayout/lobjects/lslider.jl | 4 +- src/makielayout/lobjects/ltextbox.jl | 2 +- src/makielayout/lobjects/ltoggle.jl | 2 +- src/makielayout/mousestatemachine.jl | 177 +++++++++++++++------------ src/makielayout/types.jl | 2 +- 9 files changed, 144 insertions(+), 122 deletions(-) diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index d2b0586c9..680438b94 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -85,7 +85,7 @@ export layoutscene export set_close_to! export xaxis_bottom!, xaxis_top!, yaxis_left!, yaxis_right! export labelslider! -export addmousestate! +export addmouseevents! export hlines!, vlines! diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index 23740019c..03aeb33bb 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -294,17 +294,17 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) # layout layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] - mousestate = addmousestate!(scene) + mouseevents = addmouseevents!(scene) interactions = AbstractInteraction[] la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, - layoutobservables, attrs, block_limit_linking, decorations, mousestate, interactions) + layoutobservables, attrs, block_limit_linking, decorations, mouseevents, interactions) - on(mousestate) do state + on(mouseevents) do event for i in la.interactions - process_interaction(i, state, la) + process_interaction(i, event, la) end end @@ -346,20 +346,34 @@ function add_rectanglezoom!(ax) nothing end -function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDragStart}, ax) - r.from = mousestate.prev - r.to = mousestate.pos - r.rectnode[] = FRect2D(r.from, r.to .- r.from) - r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] - nothing -end +function process_interaction(r::RectangleZoom, event::MouseEvent, ax) -function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDrag}, ax) - r.to = mousestate.pos - r.rectnode[] = FRect2D(r.from, r.to .- r.from) - nothing + if event.type === MouseEventTypes.leftdragstart + r.from = event.prev + r.to = event.pos + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] + + elseif event.type === MouseEventTypes.leftdrag + r.to = event.pos + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + + elseif event.type === MouseEventTypes.leftdragstop + newlims = positivize(r.rectnode[]) + if !(0 in widths(newlims)) + ax.targetlimits[] = newlims + end + + if !isnothing(r.poly) + delete!(ax.scene, r.poly) + r.poly = nothing + end + end + + return nothing end + function positivize(r::FRect2D) negwidths = r.widths .< 0 newori = ifelse.(negwidths, r.origin .+ r.widths, r.origin) @@ -367,19 +381,6 @@ function positivize(r::FRect2D) FRect2D(newori, newwidths) end -function process_interaction(r::RectangleZoom, mousestate::MouseState{MouseLeftDragStop}, ax) - newlims = positivize(r.rectnode[]) - if !(0 in widths(newlims)) - ax.targetlimits[] = newlims - end - - if !isnothing(r.poly) - delete!(ax.scene, r.poly) - r.poly = nothing - end - nothing -end - struct LimitReset <: AbstractInteraction end @@ -387,14 +388,18 @@ struct LimitReset <: AbstractInteraction end function add_limit_reset!(ax) reset = LimitReset() push!(ax.interactions, reset) - nothing + return nothing end -function process_interaction(l::LimitReset, mousestate::MouseState{MouseLeftClick}, ax) - if ispressed(ax.scene, Keyboard.left_control) - autolimits!(ax) +function process_interaction(l::LimitReset, event::MouseEvent, ax) + + if event.type === MouseEventTypes.leftclick + if ispressed(ax.scene, Keyboard.left_control) + autolimits!(ax) + end end - nothing + + return nothing end ####################################### diff --git a/src/makielayout/lobjects/lbutton.jl b/src/makielayout/lobjects/lbutton.jl index ea2ada6ac..e819f4804 100644 --- a/src/makielayout/lobjects/lbutton.jl +++ b/src/makielayout/lobjects/lbutton.jl @@ -61,7 +61,7 @@ function LButton(scene::Scene; bbox = nothing, kwargs...) - mousestate = addmousestate!(scene, button, labeltext) + mousestate = addmouseevents!(scene, button, labeltext) onmouseover(mousestate) do state bcolor[] = buttoncolor_hover[] diff --git a/src/makielayout/lobjects/lmenu.jl b/src/makielayout/lobjects/lmenu.jl index 7d6ac19cc..314e5725c 100644 --- a/src/makielayout/lobjects/lmenu.jl +++ b/src/makielayout/lobjects/lmenu.jl @@ -228,7 +228,7 @@ function LMenu(parent::Scene; bbox = nothing, kwargs...) rowgap!(contentgrid, 0) - mousestates = [addmousestate!(scene, r.rect, t.textobject) for (r, t) in zip(allrects, alltexts)] + mousestates = [addmouseevents!(scene, r.rect, t.textobject) for (r, t) in zip(allrects, alltexts)] for (i, (mousestate, r, t)) in enumerate(zip(mousestates, allrects, alltexts)) onmouseover(mousestate) do state @@ -257,7 +257,7 @@ function LMenu(parent::Scene; bbox = nothing, kwargs...) end # close the menu if the user clicks somewhere else - onmousedownoutside(addmousestate!(scene)) do state + onmousedownoutside(addmouseevents!(scene)) do state if is_open[] is_open[] = !is_open[] end diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 38592fb6e..64a74181a 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -105,7 +105,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) linesegs = linesegments!(subscene, linepoints, color = linecolors, linewidth = linewidth, raw = true)[end] decorations[:linesegments] = linesegs - linestate = addmousestate!(subscene, linesegs) + linestate = addmouseevents!(subscene, linesegs) bsize = @lift($buttonradius * 2f0) @@ -117,7 +117,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) decorations[:button] = button - scenestate = addmousestate!(subscene) + scenestate = addmouseevents!(subscene) onmouseleftup(scenestate) do state bcolor[] = buttoncolor_inactive[] diff --git a/src/makielayout/lobjects/ltextbox.jl b/src/makielayout/lobjects/ltextbox.jl index 18293cf93..2fb51a980 100644 --- a/src/makielayout/lobjects/ltextbox.jl +++ b/src/makielayout/lobjects/ltextbox.jl @@ -118,7 +118,7 @@ function LTextbox(parent::Scene; bbox = nothing, kwargs...) # trigger bbox layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] - mousestate = addmousestate!(scene) + mousestate = addmouseevents!(scene) onmouseleftdown(mousestate) do state focus!(ltextbox) diff --git a/src/makielayout/lobjects/ltoggle.jl b/src/makielayout/lobjects/ltoggle.jl index ee8bf3eda..a169d00c5 100644 --- a/src/makielayout/lobjects/ltoggle.jl +++ b/src/makielayout/lobjects/ltoggle.jl @@ -54,7 +54,7 @@ function LToggle(parent::Scene; bbox = nothing, kwargs...) button = scatter!(parent, buttonpos, markersize = buttonsize, color = buttoncolor, raw = true)[end] decorations[:button] = button - buttonstate = addmousestate!(parent, button, frame) + buttonstate = addmouseevents!(parent, button, frame) onmouseleftclick(buttonstate) do state if animating[] diff --git a/src/makielayout/mousestatemachine.jl b/src/makielayout/mousestatemachine.jl index 8b4d9dede..e605f8248 100644 --- a/src/makielayout/mousestatemachine.jl +++ b/src/makielayout/mousestatemachine.jl @@ -1,47 +1,68 @@ -abstract type AbstractMouseState end +module MouseEventTypes + @enum MouseEventType begin + out + enter + over + leftdown + rightdown + middledown + leftup + rightup + middleup + leftdragstart + rightdragstart + middledragstart + leftdrag + rightdrag + middledrag + leftdragstop + rightdragstop + middledragstop + leftclick + rightclick + middleclick + leftdoubleclick + rightdoubleclick + middledoubleclick + downoutside + end + export MouseEventType +end + +using .MouseEventTypes """ - MouseState{T<:AbstractMouseState} + MouseEvent Describes a mouse state change. Fields: -- `typ`: Symbol describing the mouse state +- `type`: MouseEventType - `t`: Time of the event - `pos`: Mouse position - `tprev`: Time of previous event - `prev`: Previous mouse position """ -struct MouseState{T<:AbstractMouseState} - typ::T +struct MouseEvent + type::MouseEventType t::Float64 pos::Point2f0 tprev::Float64 prev::Point2f0 end -mousestates = (:MouseOut, :MouseEnter, :MouseOver, - :MouseLeftDown, :MouseRightDown, :MouseMiddleDown, - :MouseLeftUp, :MouseRightUp, :MouseMiddleUp, - :MouseLeftDragStart, :MouseRightDragStart, :MouseMiddleDragStart, - :MouseLeftDrag, :MouseRightDrag, :MouseMiddleDrag, - :MouseLeftDragStop, :MouseRightDragStop, :MouseMiddleDragStop, - :MouseLeftClick, :MouseRightClick, :MouseMiddleClick, - :MouseLeftDoubleclick, :MouseRightDoubleclick, :MouseMiddleDoubleclick, - :MouseDownOutside - ) - -for statetype in mousestates - onfunctionname = Symbol("on" * lowercase(String(statetype))) + + +for eventtype in instances(MouseEventType) + onfunctionname = Symbol("onmouse" * String(Symbol(eventtype))) @eval begin - struct $statetype <: AbstractMouseState end """ - Executes the function f whenever the `Node{MouseState}` statenode transitions - to `$($statetype)`. + Executes the function f whenever the `Node{MouseEvent}` statenode transitions + to `$($eventtype)`. """ - function $onfunctionname(f, statenode::Node{MouseState}) + function $onfunctionname(f, statenode::Node{MouseEvent}) on(statenode) do state - if state.typ isa $statetype + if state.type === $eventtype f(state) end end @@ -51,14 +72,10 @@ for statetype in mousestates end -function Base.show(io::IO, ms::MouseState{T}) where T - print(io, "$T(t: $(ms.t), pos: $(ms.pos[1]), $(ms.pos[2]), tprev: $(ms.tprev), prev: $(ms.prev[1]), $(ms.prev[2]))") -end - """ - addmousestate!(scene, elements...) + addmouseevents!(scene, elements...) -Returns an `Observable{MouseState}` which is triggered by all mouse +Returns an `Observable{MouseEvent}` which is triggered by all mouse interactions with the `scene` and optionally restricted to all given plot objects in `elements`. @@ -67,25 +84,25 @@ To react to mouse events, use the onmouse... handlers. Example: ``` -mousestate = addmousestate!(scene, scatterplot) +mouseevents = addmouseevents!(scene, scatterplot) -onmouseleftclick(mousestate) do state - # do something with the mousestate +onmouseleftclick(mouseevents) do event + # do something with the mouseevent end ``` """ -function addmousestate!(scene, elements...) +function addmouseevents!(scene, elements...) Mouse = AbstractPlotting.Mouse dblclick_max_interval = 0.2 - mousestate = Node{MouseState}(MouseState(MouseOut(), 0.0, Point2f0(0, 0), 0.0, Point2f0(0, 0))) + mouseevent = Node{MouseEvent}(MouseEvent(MouseEventTypes.out, 0.0, Point2f0(0, 0), 0.0, Point2f0(0, 0))) is_mouse_over_relevant_area() = isempty(elements) ? AbstractPlotting.is_mouseinside(scene) : mouseover(scene, elements...) # initialize state variables - last_mousestate = Ref{Mouse.DragEnum}(events(scene).mousedrag[]) + last_mouseevent = Ref{Mouse.DragEnum}(events(scene).mousedrag[]) prev = Ref(mouseposition(AbstractPlotting.rootparent(scene))) mouse_downed_inside = Ref(false) mouse_downed_button = Ref{Optional{Mouse.Button}}(nothing) @@ -105,49 +122,49 @@ function addmousestate!(scene, elements...) mouse_inside = is_mouse_over_relevant_area() # movement while mouse is pressed - if last_mousestate[] == Mouse.pressed + if last_mouseevent[] == Mouse.pressed # must have been a registered drag (otherwise could have come from outside) if drag_ongoing[] event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDrag() - Mouse.right => MouseRightDrag() - Mouse.middle => MouseMiddleDrag() + Mouse.left => MouseEventTypes.leftdrag + Mouse.right => MouseEventTypes.rightdrag + Mouse.middle => MouseEventTypes.middledrag x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) end # mouse moved while just having been pressed down - elseif last_mousestate[] == Mouse.down + elseif last_mouseevent[] == Mouse.down # mouse must have been downed inside # that means a drag started if mouse_downed_inside[] drag_ongoing[] = true event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDragStart() - Mouse.right => MouseRightDragStart() - Mouse.middle => MouseMiddleDragStart() + Mouse.left => MouseEventTypes.leftdragstart + Mouse.right => MouseEventTypes.rightdragstart + Mouse.middle => MouseEventTypes.middledragstart x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDrag() - Mouse.right => MouseRightDrag() - Mouse.middle => MouseMiddleDrag() + Mouse.left => MouseEventTypes.leftdrag + Mouse.right => MouseEventTypes.rightdrag + Mouse.middle => MouseEventTypes.middledrag x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) end else if mouse_inside if mouse_was_inside[] - mousestate[] = MouseState(MouseOver(), t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.over, t, pos, tprev[], prev[]) else - mousestate[] = MouseState(MouseEnter(), t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.enter, t, pos, tprev[], prev[]) end else if mouse_was_inside[] - mousestate[] = MouseState(MouseOut(), t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.out, t, pos, tprev[], prev[]) end end end @@ -175,16 +192,16 @@ function addmousestate!(scene, elements...) if mouse_was_inside[] event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDown() - Mouse.right => MouseRightDown() - Mouse.middle => MouseMiddleDown() + Mouse.left => MouseEventTypes.leftdown + Mouse.right => MouseEventTypes.rightdown + Mouse.middle => MouseEventTypes.middledown x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) mouse_downed_inside[] = true else mouse_downed_inside[] = false - mousestate[] = MouseState(MouseDownOutside(), t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.downoutside, t, pos, tprev[], prev[]) end end elseif mousedrag == Mouse.up @@ -199,27 +216,27 @@ function addmousestate!(scene, elements...) if drag_ongoing[] event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDragStop() - Mouse.right => MouseRightDragStop() - Mouse.middle => MouseMiddleDragStop() + Mouse.left => MouseEventTypes.leftdragstop + Mouse.right => MouseEventTypes.rightdragstop + Mouse.middle => MouseEventTypes.middledragstop x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) drag_ongoing[] = false if mouse_was_inside[] # up after drag done over element event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftUp() - Mouse.right => MouseRightUp() - Mouse.middle => MouseMiddleUp() + Mouse.left => MouseEventTypes.leftup + Mouse.right => MouseEventTypes.rightup + Mouse.middle => MouseEventTypes.middleup x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) else # mouse could be not over elements after drag is over - mousestate[] = MouseState(MouseOut(), t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.out, t, pos, tprev[], prev[]) end else if mouse_was_inside[] @@ -232,21 +249,21 @@ function addmousestate!(scene, elements...) mouse_downed_button[] == b_last_click[] event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftDoubleclick() - Mouse.right => MouseRightDoubleclick() - Mouse.middle => MouseMiddleDoubleclick() + Mouse.left => MouseEventTypes.leftdoubleclick + Mouse.right => MouseEventTypes.rightdoubleclick + Mouse.middle => MouseEventTypes.middledoubleclick x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) last_click_was_double[] = true else event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftClick() - Mouse.right => MouseRightClick() - Mouse.middle => MouseMiddleClick() + Mouse.left => MouseEventTypes.leftclick + Mouse.right => MouseEventTypes.rightclick + Mouse.middle => MouseEventTypes.middleclick x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) last_click_was_double[] = false end # save what type the last downed button was @@ -256,13 +273,13 @@ function addmousestate!(scene, elements...) # up after click event = @match mouse_downed_button[] begin - Mouse.left => MouseLeftUp() - Mouse.right => MouseRightUp() - Mouse.middle => MouseMiddleUp() + Mouse.left => MouseEventTypes.leftup + Mouse.right => MouseEventTypes.rightup + Mouse.middle => MouseEventTypes.middleup x => error("No recognized mouse button $x") end - mousestate[] = MouseState(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) end end @@ -271,9 +288,9 @@ function addmousestate!(scene, elements...) end - last_mousestate[] = mousedrag + last_mouseevent[] = mousedrag tprev[] = t end - mousestate + mouseevent end diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index a883a15b6..5dd7b2703 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -60,7 +60,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene attributes::Attributes block_limit_linking::Node{Bool} decorations::Dict{Symbol, Any} - mousestate::Observable{MouseState} + mouseevents::Observable{MouseEvent} interactions::Vector{AbstractInteraction} end From 3619eb0bb5693cd1184a7699db1bb9975371e87f Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Tue, 27 Oct 2020 12:07:46 +0100 Subject: [PATCH 03/30] add scrollzoom --- src/makielayout/lobjects/laxis.jl | 88 ++++++++++++++++++++++++++++++- src/makielayout/types.jl | 6 +++ 2 files changed, 92 insertions(+), 2 deletions(-) diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index 03aeb33bb..53c20945b 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -295,21 +295,33 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] mouseevents = addmouseevents!(scene) + scrollevents = Node(ScrollEvent(0, 0)) + on(scene.events.scroll) do s + if is_mouseinside(scene) + scrollevents[] = ScrollEvent(s[1], s[2]) + end + end interactions = AbstractInteraction[] la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, - layoutobservables, attrs, block_limit_linking, decorations, mouseevents, interactions) + layoutobservables, attrs, block_limit_linking, decorations, + mouseevents, scrollevents,interactions) - on(mouseevents) do event + function process_event(event) for i in la.interactions process_interaction(i, event, la) end end + on(process_event, mouseevents) + on(process_event, scrollevents) + + add_limit_reset!(la) add_rectanglezoom!(la) + add_scrollzoom!(la) # # add action that resets limits on ctrl + click # add_reset_limits!(la) @@ -402,6 +414,78 @@ function process_interaction(l::LimitReset, event::MouseEvent, ax) return nothing end + +struct ScrollZoom <: AbstractInteraction + speed::Float32 + reset_timer::Ref{Any} + prev_xticklabelspace::Ref{Any} + prev_yticklabelspace::Ref{Any} + reset_delay::Float32 +end + +function add_scrollzoom!(ax) + sz = ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2) + push!(ax.interactions, sz) + return nothing +end + +function process_interaction(s::ScrollZoom, event::ScrollEvent, ax) + # use vertical zoom + zoom = event.y + + tlimits = ax.targetlimits + xzoomlock = ax.xzoomlock + yzoomlock = ax.yzoomlock + xzoomkey = ax.xzoomkey + yzoomkey = ax.yzoomkey + + scene = ax.scene + e = events(scene) + cam = camera(scene) + + if zoom != 0 + pa = pixelarea(scene)[] + + # don't let z go negative + z = max(0.1f0, 1f0 - (abs(zoom) * s.speed)) + if zoom > 0 + z = 1/z # sets the old to be a fraction of the new. This ensures zoom in & then out returns to original position. + end + + mp_axscene = Vec4f0((e.mouseposition[] .- pa.origin)..., 0, 1) + + # first to normal -1..1 space + mp_axfraction = (cam.pixel_space[] * mp_axscene)[1:2] .* + # now to 1..-1 if an axis is reversed to correct zoom point + (-2 .* ((ax.xreversed[], ax.yreversed[])) .+ 1) .* + # now to 0..1 + 0.5 .+ 0.5 + + xorigin = tlimits[].origin[1] + yorigin = tlimits[].origin[2] + + xwidth = tlimits[].widths[1] + ywidth = tlimits[].widths[2] + + newxwidth = xzoomlock[] ? xwidth : xwidth * z + newywidth = yzoomlock[] ? ywidth : ywidth * z + + newxorigin = xzoomlock[] ? xorigin : xorigin + mp_axfraction[1] * (xwidth - newxwidth) + newyorigin = yzoomlock[] ? yorigin : yorigin + mp_axfraction[2] * (ywidth - newywidth) + + timed_ticklabelspace_reset(ax, s.reset_timer, s.prev_xticklabelspace, s.prev_yticklabelspace, s.reset_delay) + + tlimits[] = if ispressed(scene, xzoomkey[]) + FRect(newxorigin, yorigin, newxwidth, ywidth) + elseif ispressed(scene, yzoomkey[]) + FRect(xorigin, newyorigin, xwidth, newywidth) + else + FRect(newxorigin, newyorigin, newxwidth, newywidth) + end + + end +end + ####################################### diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 5dd7b2703..31c99f096 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -48,6 +48,11 @@ end abstract type AbstractInteraction end +struct ScrollEvent + x::Float32 + y::Float32 +end + abstract type LObject end mutable struct LAxis <: AbstractPlotting.AbstractScene @@ -61,6 +66,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene block_limit_linking::Node{Bool} decorations::Dict{Symbol, Any} mouseevents::Observable{MouseEvent} + scrollevents::Observable{ScrollEvent} interactions::Vector{AbstractInteraction} end From 54ba88bfc95c3de3f5edd19994f66fa211aa5976 Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Tue, 27 Oct 2020 14:49:41 +0100 Subject: [PATCH 04/30] store mouse data and px, add dragpan, move types --- src/makielayout/lobjects/laxis.jl | 84 ++++++++++++++-------------- src/makielayout/mousestatemachine.jl | 64 +++++++++++---------- src/makielayout/types.jl | 24 ++++++++ 3 files changed, 102 insertions(+), 70 deletions(-) diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index 53c20945b..b32463643 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -318,10 +318,10 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) on(process_event, mouseevents) on(process_event, scrollevents) - - add_limit_reset!(la) - add_rectanglezoom!(la) - add_scrollzoom!(la) + push!(la.interactions, RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing)) + push!(la.interactions, LimitReset()) + push!(la.interactions, ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) + push!(la.interactions, DragPan(Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) # # add action that resets limits on ctrl + click # add_reset_limits!(la) @@ -345,29 +345,17 @@ function process_interaction(@nospecialize args...) # do nothing in the default case end -mutable struct RectangleZoom <: AbstractInteraction - from::Union{Nothing, Point2f0} - to::Union{Nothing, Point2f0} - rectnode::Observable{FRect2D} - poly::Union{Poly, Nothing} -end - -function add_rectanglezoom!(ax) - zoom = RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing) - push!(ax.interactions, zoom) - nothing -end function process_interaction(r::RectangleZoom, event::MouseEvent, ax) if event.type === MouseEventTypes.leftdragstart - r.from = event.prev - r.to = event.pos + r.from = event.prev_data + r.to = event.data r.rectnode[] = FRect2D(r.from, r.to .- r.from) r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] elseif event.type === MouseEventTypes.leftdrag - r.to = event.pos + r.to = event.data r.rectnode[] = FRect2D(r.from, r.to .- r.from) elseif event.type === MouseEventTypes.leftdragstop @@ -394,15 +382,6 @@ function positivize(r::FRect2D) end - -struct LimitReset <: AbstractInteraction end - -function add_limit_reset!(ax) - reset = LimitReset() - push!(ax.interactions, reset) - return nothing -end - function process_interaction(l::LimitReset, event::MouseEvent, ax) if event.type === MouseEventTypes.leftclick @@ -415,20 +394,6 @@ function process_interaction(l::LimitReset, event::MouseEvent, ax) end -struct ScrollZoom <: AbstractInteraction - speed::Float32 - reset_timer::Ref{Any} - prev_xticklabelspace::Ref{Any} - prev_yticklabelspace::Ref{Any} - reset_delay::Float32 -end - -function add_scrollzoom!(ax) - sz = ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2) - push!(ax.interactions, sz) - return nothing -end - function process_interaction(s::ScrollZoom, event::ScrollEvent, ax) # use vertical zoom zoom = event.y @@ -486,6 +451,41 @@ function process_interaction(s::ScrollZoom, event::ScrollEvent, ax) end end +function process_interaction(dp::DragPan, event::MouseEvent, ax) + + if event.type !== MouseEventTypes.rightdrag + return nothing + end + + tlimits = ax.targetlimits + xpanlock = ax.xpanlock + ypanlock = ax.ypanlock + xpankey = ax.xpankey + ypankey = ax.ypankey + panbutton = ax.panbutton + + scene = ax.scene + + movement = AbstractPlotting.to_world(ax.scene, event.px) .- + AbstractPlotting.to_world(ax.scene, event.prev_px) + + xori, yori = Vec2f0(tlimits[].origin) .- movement + + if xpanlock[] || ispressed(scene, ypankey[]) + xori = tlimits[].origin[1] + end + + if ypanlock[] || ispressed(scene, xpankey[]) + yori = tlimits[].origin[2] + end + + timed_ticklabelspace_reset(ax, dp.reset_timer, dp.prev_xticklabelspace, dp.prev_yticklabelspace, dp.reset_delay) + + tlimits[] = FRect(Vec2f0(xori, yori), widths(tlimits[])) + + return nothing +end + ####################################### diff --git a/src/makielayout/mousestatemachine.jl b/src/makielayout/mousestatemachine.jl index e605f8248..d605f9576 100644 --- a/src/makielayout/mousestatemachine.jl +++ b/src/makielayout/mousestatemachine.jl @@ -38,16 +38,20 @@ Describes a mouse state change. Fields: - `type`: MouseEventType - `t`: Time of the event -- `pos`: Mouse position -- `tprev`: Time of previous event -- `prev`: Previous mouse position +- `data`: Mouse position in data coordinates +- `px`: Mouse position in px relative to scene origin +- `prev_t`: Time of previous event +- `prev_data`: Previous mouse position in data coordinates +- `prev_px`: Previous mouse position in data coordinates """ struct MouseEvent type::MouseEventType t::Float64 - pos::Point2f0 - tprev::Float64 - prev::Point2f0 + data::Point2f0 + px::Point2f0 + prev_t::Float64 + prev_data::Point2f0 + prev_px::Point2f0 end @@ -96,19 +100,20 @@ function addmouseevents!(scene, elements...) Mouse = AbstractPlotting.Mouse dblclick_max_interval = 0.2 - mouseevent = Node{MouseEvent}(MouseEvent(MouseEventTypes.out, 0.0, Point2f0(0, 0), 0.0, Point2f0(0, 0))) + mouseevent = Node{MouseEvent}(MouseEvent(MouseEventTypes.out, 0.0, Point2f0(0, 0), Point2f0(0, 0), 0.0, Point2f0(0, 0), Point2f0(0, 0))) is_mouse_over_relevant_area() = isempty(elements) ? AbstractPlotting.is_mouseinside(scene) : mouseover(scene, elements...) # initialize state variables last_mouseevent = Ref{Mouse.DragEnum}(events(scene).mousedrag[]) - prev = Ref(mouseposition(AbstractPlotting.rootparent(scene))) + prev_data = Ref(mouseposition(scene)) + prev_px = Ref(AbstractPlotting.mouseposition_px(scene)) mouse_downed_inside = Ref(false) mouse_downed_button = Ref{Optional{Mouse.Button}}(nothing) drag_ongoing = Ref(false) mouse_was_inside = Ref(false) - tprev = Ref(0.0) + prev_t = Ref(0.0) t_last_click = Ref(0.0) b_last_click = Ref{Optional{Mouse.Button}}(nothing) last_click_was_double = Ref(false) @@ -118,7 +123,8 @@ function addmouseevents!(scene, elements...) on(events(scene).mouseposition) do mp t = time() - pos = mouseposition(AbstractPlotting.rootparent(scene)) + data = mouseposition(scene) + px = AbstractPlotting.mouseposition_px(scene) mouse_inside = is_mouse_over_relevant_area() # movement while mouse is pressed @@ -131,7 +137,7 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledrag x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) end # mouse moved while just having been pressed down elseif last_mouseevent[] == Mouse.down @@ -145,7 +151,7 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledragstart x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) event = @match mouse_downed_button[] begin Mouse.left => MouseEventTypes.leftdrag @@ -153,25 +159,26 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledrag x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) end else if mouse_inside if mouse_was_inside[] - mouseevent[] = MouseEvent(MouseEventTypes.over, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.over, t, data, px, prev_t[], prev_data[], prev_px[]) else - mouseevent[] = MouseEvent(MouseEventTypes.enter, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.enter, t, data, px, prev_t[], prev_data[], prev_px[]) end else if mouse_was_inside[] - mouseevent[] = MouseEvent(MouseEventTypes.out, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.out, t, data, px, prev_t[], prev_data[], prev_px[]) end end end mouse_was_inside[] = mouse_inside - prev[] = pos - tprev[] = t + prev_data[] = data + prev_px[] = px + prev_t[] = t end @@ -179,7 +186,8 @@ function addmouseevents!(scene, elements...) on(events(scene).mousedrag) do mousedrag t = time() - pos = prev[] + data = prev_data[] + px = prev_px[] pressed_buttons = events(scene).mousebuttons[] @@ -197,11 +205,11 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledown x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) mouse_downed_inside[] = true else mouse_downed_inside[] = false - mouseevent[] = MouseEvent(MouseEventTypes.downoutside, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.downoutside, t, data, px, prev_t[], prev_data[], prev_px[]) end end elseif mousedrag == Mouse.up @@ -221,7 +229,7 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledragstop x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) drag_ongoing[] = false if mouse_was_inside[] @@ -233,10 +241,10 @@ function addmouseevents!(scene, elements...) x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) else # mouse could be not over elements after drag is over - mouseevent[] = MouseEvent(MouseEventTypes.out, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(MouseEventTypes.out, t, data, px, prev_t[], prev_data[], prev_px[]) end else if mouse_was_inside[] @@ -254,7 +262,7 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middledoubleclick x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) last_click_was_double[] = true else event = @match mouse_downed_button[] begin @@ -263,7 +271,7 @@ function addmouseevents!(scene, elements...) Mouse.middle => MouseEventTypes.middleclick x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) last_click_was_double[] = false end # save what type the last downed button was @@ -279,7 +287,7 @@ function addmouseevents!(scene, elements...) x => error("No recognized mouse button $x") end - mouseevent[] = MouseEvent(event, t, pos, tprev[], prev[]) + mouseevent[] = MouseEvent(event, t, data, px, prev_t[], prev_data[], prev_px[]) end end @@ -289,7 +297,7 @@ function addmouseevents!(scene, elements...) last_mouseevent[] = mousedrag - tprev[] = t + prev_t[] = t end mouseevent diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 31c99f096..4b786e90e 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -48,6 +48,30 @@ end abstract type AbstractInteraction end +struct LimitReset <: AbstractInteraction end + +mutable struct RectangleZoom <: AbstractInteraction + from::Union{Nothing, Point2f0} + to::Union{Nothing, Point2f0} + rectnode::Observable{FRect2D} + poly::Union{Poly, Nothing} +end + +struct ScrollZoom <: AbstractInteraction + speed::Float32 + reset_timer::Ref{Any} + prev_xticklabelspace::Ref{Any} + prev_yticklabelspace::Ref{Any} + reset_delay::Float32 +end + +struct DragPan <: AbstractInteraction + reset_timer::Ref{Any} + prev_xticklabelspace::Ref{Any} + prev_yticklabelspace::Ref{Any} + reset_delay::Float32 +end + struct ScrollEvent x::Float32 y::Float32 From 7c48e2bb1ce6f15649d3d4bd675b5f9851f8975f Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Tue, 27 Oct 2020 15:49:21 +0100 Subject: [PATCH 05/30] register and deregister --- src/makielayout/lobjects/laxis.jl | 193 +++++++----------------------- src/makielayout/types.jl | 2 +- 2 files changed, 45 insertions(+), 150 deletions(-) diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index b32463643..bb8321046 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -302,15 +302,15 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) end end - interactions = AbstractInteraction[] + interactions = Dict{Symbol, AbstractInteraction}() la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, layoutobservables, attrs, block_limit_linking, decorations, - mouseevents, scrollevents,interactions) + mouseevents, scrollevents, interactions) function process_event(event) - for i in la.interactions + for i in values(la.interactions) process_interaction(i, event, la) end end @@ -318,18 +318,21 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) on(process_event, mouseevents) on(process_event, scrollevents) - push!(la.interactions, RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing)) - push!(la.interactions, LimitReset()) - push!(la.interactions, ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) - push!(la.interactions, DragPan(Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) + register_interaction!(la, + :rectanglezoom, + RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing)) - # # add action that resets limits on ctrl + click - # add_reset_limits!(la) - # # add action that allows zooming using mouse scrolling - # add_zoom!(la) - # # add action that allows panning using a mouse button - # add_pan!(la) + register_interaction!(la, + :limitreset, + LimitReset()) + register_interaction!(la, + :scrollzoom, + ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) + + register_interaction!(la, + :dragpan, + DragPan(Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) # compute limits that adhere to the limit aspect ratio whenever the targeted # limits or the scene size change, because both influence the displayed ratio @@ -341,12 +344,37 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) end ####################################### + +function register_interaction!(ax::LAxis, name::Symbol, interaction::AbstractInteraction) + haskey(ax.interactions, name) && error("Interaction $name already exists.") + registration_setup!(ax, interaction) + push!(ax.interactions, name => interaction) + return interaction +end + +function deregister_interaction!(ax::LAxis, name::Symbol) + !haskey(ax.interactions, name) && error("Interaction $name does not exist.") + interaction = ax.interactions[name] + + deregistration_cleanup!(ax, interaction) + pop!(ax.interactions, name) + return interaction +end + +function registration_setup!(ax, interaction) + # do nothing in the default case +end + +function deregistration_cleanup!(ax, interaction) + # do nothing in the default case +end + function process_interaction(@nospecialize args...) # do nothing in the default case end -function process_interaction(r::RectangleZoom, event::MouseEvent, ax) +function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) if event.type === MouseEventTypes.leftdragstart r.from = event.prev_data @@ -382,7 +410,7 @@ function positivize(r::FRect2D) end -function process_interaction(l::LimitReset, event::MouseEvent, ax) +function process_interaction(l::LimitReset, event::MouseEvent, ax::LAxis) if event.type === MouseEventTypes.leftclick if ispressed(ax.scene, Keyboard.left_control) @@ -394,7 +422,7 @@ function process_interaction(l::LimitReset, event::MouseEvent, ax) end -function process_interaction(s::ScrollZoom, event::ScrollEvent, ax) +function process_interaction(s::ScrollZoom, event::ScrollEvent, ax::LAxis) # use vertical zoom zoom = event.y @@ -777,139 +805,6 @@ function linkyaxes!(a::LAxis, others...) autolimits!(a) end -function add_pan!(ax::LAxis) - - tlimits = ax.targetlimits - xpanlock = ax.xpanlock - ypanlock = ax.ypanlock - xpankey = ax.xpankey - ypankey = ax.ypankey - panbutton = ax.panbutton - - scene = ax.scene - - reset_timer = Ref{Any}(nothing) - prev_xticklabelspace = Ref{Any}(0) - prev_yticklabelspace = Ref{Any}(0) - - startpos = Base.RefValue((0.0, 0.0)) - e = events(scene) - on( - camera(scene), - # Node.((scene, cam, startpos))..., - Node.((scene, startpos))..., - e.mousedrag - ) do scene, startpos, dragging - mp = e.mouseposition[] - if ispressed(scene, panbutton[]) && is_mouseinside(scene) - window_area = pixelarea(scene)[] - if dragging == Mouse.down - startpos[] = mp - elseif dragging == Mouse.pressed && ispressed(scene, panbutton[]) - diff = startpos[] .- mp - startpos[] = mp - pxa = scene.px_area[] - diff_fraction = Vec2f0(diff) ./ Vec2f0(widths(pxa)) - - diff_limits = diff_fraction .* widths(tlimits[]) - - # correct for reversals - reversals = (ax.xreversed[], ax.yreversed[]) - - diff_limits = diff_limits .* (-2 .* reversals .+ 1) - - xori, yori = Vec2f0(tlimits[].origin) .+ Vec2f0(diff_limits) - - if xpanlock[] || ispressed(scene, ypankey[]) - xori = tlimits[].origin[1] - end - - if ypanlock[] || ispressed(scene, xpankey[]) - yori = tlimits[].origin[2] - end - - timed_ticklabelspace_reset(ax, reset_timer, prev_xticklabelspace, prev_yticklabelspace, 0.1) - - tlimits[] = FRect(Vec2f0(xori, yori), widths(tlimits[])) - end - end - return - end -end - -function add_zoom!(ax::LAxis) - - tlimits = ax.targetlimits - xzoomlock = ax.xzoomlock - yzoomlock = ax.yzoomlock - xzoomkey = ax.xzoomkey - yzoomkey = ax.yzoomkey - - scene = ax.scene - - e = events(scene) - cam = camera(scene) - - reset_timer = Ref{Any}(nothing) - prev_xticklabelspace = Ref{Any}(0) - prev_yticklabelspace = Ref{Any}(0) - - on(cam, e.scroll) do x - # @extractvalue cam (zoomspeed, zoombutton, area) - zoomspeed = 0.10f0 - zoombutton = nothing - zoom = Float32(x[2]) - if zoom != 0 && ispressed(scene, zoombutton) && AbstractPlotting.is_mouseinside(scene) - pa = pixelarea(scene)[] - - # don't let z go negative - z = max(0.1f0, 1f0 - (abs(zoom) * zoomspeed)) - if zoom > 0 - z = 1/z # sets the old to be a fraction of the new. This ensures zoom in & then out returns to original position. - end - - mp_axscene = Vec4f0((e.mouseposition[] .- pa.origin)..., 0, 1) - - # first to normal -1..1 space - mp_axfraction = (cam.pixel_space[] * mp_axscene)[1:2] .* - # now to 1..-1 if an axis is reversed to correct zoom point - (-2 .* ((ax.xreversed[], ax.yreversed[])) .+ 1) .* - # now to 0..1 - 0.5 .+ 0.5 - - xorigin = tlimits[].origin[1] - yorigin = tlimits[].origin[2] - - xwidth = tlimits[].widths[1] - ywidth = tlimits[].widths[2] - - newxwidth = xzoomlock[] ? xwidth : xwidth * z - newywidth = yzoomlock[] ? ywidth : ywidth * z - - newxorigin = xzoomlock[] ? xorigin : xorigin + mp_axfraction[1] * (xwidth - newxwidth) - newyorigin = yzoomlock[] ? yorigin : yorigin + mp_axfraction[2] * (ywidth - newywidth) - - timed_ticklabelspace_reset(ax, reset_timer, prev_xticklabelspace, prev_yticklabelspace, 0.1) - - tlimits[] = if AbstractPlotting.ispressed(scene, xzoomkey[]) - FRect(newxorigin, yorigin, newxwidth, ywidth) - elseif AbstractPlotting.ispressed(scene, yzoomkey[]) - FRect(xorigin, newyorigin, xwidth, newywidth) - else - FRect(newxorigin, newyorigin, newxwidth, newywidth) - end - - end - - return - end - - # Also support rubber band selection - rect = select_rectangle(scene) - on(rect) do r - tlimits[] = r - end -end """ Keeps the ticklabelspace static for a short duration and then resets it to its previous diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 4b786e90e..02997860a 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -91,7 +91,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene decorations::Dict{Symbol, Any} mouseevents::Observable{MouseEvent} scrollevents::Observable{ScrollEvent} - interactions::Vector{AbstractInteraction} + interactions::Dict{Symbol, AbstractInteraction} end mutable struct LColorbar <: LObject From bbcc47abceba97968a349660399386548f8a980f Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Tue, 27 Oct 2020 16:19:26 +0100 Subject: [PATCH 06/30] allow activation, deactivation --- src/makielayout/lobjects/laxis.jl | 22 +++++++++++++++++----- src/makielayout/types.jl | 2 +- 2 files changed, 18 insertions(+), 6 deletions(-) diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index bb8321046..95255671c 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -310,8 +310,8 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) function process_event(event) - for i in values(la.interactions) - process_interaction(i, event, la) + for (active, interaction) in values(la.interactions) + active && process_interaction(interaction, event, la) end end @@ -329,7 +329,7 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) register_interaction!(la, :scrollzoom, ScrollZoom(0.1, Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) - + register_interaction!(la, :dragpan, DragPan(Ref{Any}(nothing), Ref{Any}(0), Ref{Any}(0), 0.2)) @@ -348,13 +348,13 @@ end function register_interaction!(ax::LAxis, name::Symbol, interaction::AbstractInteraction) haskey(ax.interactions, name) && error("Interaction $name already exists.") registration_setup!(ax, interaction) - push!(ax.interactions, name => interaction) + push!(ax.interactions, name => (true, interaction)) return interaction end function deregister_interaction!(ax::LAxis, name::Symbol) !haskey(ax.interactions, name) && error("Interaction $name does not exist.") - interaction = ax.interactions[name] + _, interaction = ax.interactions[name] deregistration_cleanup!(ax, interaction) pop!(ax.interactions, name) @@ -369,6 +369,18 @@ function deregistration_cleanup!(ax, interaction) # do nothing in the default case end +function activate_interaction!(ax, name::Symbol) + !haskey(ax.interactions, name) && error("Interaction $name does not exist.") + ax.interactions[name] = (true, ax.interactions[name][2]) + return nothing +end + +function deactivate_interaction!(ax, name::Symbol) + !haskey(ax.interactions, name) && error("Interaction $name does not exist.") + ax.interactions[name] = (false, ax.interactions[name][2]) + return nothing +end + function process_interaction(@nospecialize args...) # do nothing in the default case end diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 02997860a..63334f834 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -91,7 +91,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene decorations::Dict{Symbol, Any} mouseevents::Observable{MouseEvent} scrollevents::Observable{ScrollEvent} - interactions::Dict{Symbol, AbstractInteraction} + interactions::Dict{Symbol, Tuple{Bool, AbstractInteraction}} end mutable struct LColorbar <: LObject From 2444e9ac3712890b4cb738400188bd958bb0c41b Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Tue, 27 Oct 2020 19:17:25 +0100 Subject: [PATCH 07/30] fix slider --- src/makielayout/lobjects/lslider.jl | 20 ++++++++++---------- 1 file changed, 10 insertions(+), 10 deletions(-) diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 64a74181a..1b4d52ff9 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -117,18 +117,18 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) decorations[:button] = button - scenestate = addmouseevents!(subscene) + mouseevents = addmouseevents!(subscene) - onmouseleftup(scenestate) do state + onmouseleftup(mouseevents) do event bcolor[] = buttoncolor_inactive[] end - onmouseleftdrag(scenestate) do state + onmouseleftdrag(mouseevents) do event pad = buttonradius[] + buttonstrokewidth[] dragging[] = true - dif = state.pos - state.prev + dif = event.px - event.prev_px fraction = if horizontal[] dif[1] / (width(sliderbox[]) - 2pad) else @@ -145,35 +145,35 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) end end - onmouseleftdragstop(scenestate) do state + onmouseleftdragstop(mouseevents) do event dragging[] = false # adjust slider to closest legal value sliderfraction[] = sliderfraction[] end - onmouseleftdown(scenestate) do state + onmouseleftdown(mouseevents) do event bcolor[] = color_active[] pad = buttonradius[] + buttonstrokewidth[] - pos = state.pos + pos = event.px dim = horizontal[] ? 1 : 2 frac = (pos[dim] - endpoints[][1][dim] - pad) / (endpoints[][2][dim] - endpoints[][1][dim] - 2pad) selected_index[] = closest_fractionindex(sliderrange[], frac) end - onmouseleftdoubleclick(scenestate) do state + onmouseleftdoubleclick(mouseevents) do event selected_index[] = closest_index(sliderrange[], startvalue[]) end - onmouseenter(scenestate) do state + onmouseenter(mouseevents) do event # bcolor[] = color_active[] linecolors[] = [color_active[], color_inactive[]] button.strokecolor = color_active[] end - onmouseout(scenestate) do state + onmouseout(mouseevents) do event bcolor[] = buttoncolor_inactive[] linecolors[] = [color_active_dimmed[], color_inactive[]] button.strokecolor = color_active_dimmed[] From 281526e889d51e528bf968df9bc933fa03eefa3e Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Tue, 27 Oct 2020 19:18:04 +0100 Subject: [PATCH 08/30] button --- src/makielayout/lobjects/lbutton.jl | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/src/makielayout/lobjects/lbutton.jl b/src/makielayout/lobjects/lbutton.jl index e819f4804..141b01b2d 100644 --- a/src/makielayout/lobjects/lbutton.jl +++ b/src/makielayout/lobjects/lbutton.jl @@ -61,24 +61,24 @@ function LButton(scene::Scene; bbox = nothing, kwargs...) - mousestate = addmouseevents!(scene, button, labeltext) + mouseevents = addmouseevents!(scene, button, labeltext) - onmouseover(mousestate) do state + onmouseover(mouseevents) do state bcolor[] = buttoncolor_hover[] lcolor[] = labelcolor_hover[] end - onmouseout(mousestate) do state + onmouseout(mouseevents) do state bcolor[] = buttoncolor[] lcolor[] = labelcolor[] end - onmouseleftup(mousestate) do state + onmouseleftup(mouseevents) do state bcolor[] = buttoncolor_hover[] lcolor[] = labelcolor_hover[] end - onmouseleftdown(mousestate) do state + onmouseleftdown(mouseevents) do state bcolor[] = buttoncolor_active[] lcolor[] = labelcolor_active[] clicks[] = clicks[] + 1 From 253733d6a3e38e058b171962f2094510b3bf73c0 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Tue, 27 Oct 2020 19:30:26 +0100 Subject: [PATCH 09/30] lmenu --- src/makielayout/lobjects/lmenu.jl | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/src/makielayout/lobjects/lmenu.jl b/src/makielayout/lobjects/lmenu.jl index 314e5725c..fb8ba9384 100644 --- a/src/makielayout/lobjects/lmenu.jl +++ b/src/makielayout/lobjects/lmenu.jl @@ -228,14 +228,14 @@ function LMenu(parent::Scene; bbox = nothing, kwargs...) rowgap!(contentgrid, 0) - mousestates = [addmouseevents!(scene, r.rect, t.textobject) for (r, t) in zip(allrects, alltexts)] + mouseevents_vector = [addmouseevents!(scene, r.rect, t.textobject) for (r, t) in zip(allrects, alltexts)] - for (i, (mousestate, r, t)) in enumerate(zip(mousestates, allrects, alltexts)) - onmouseover(mousestate) do state + for (i, (mouseevents, r, t)) in enumerate(zip(mouseevents_vector, allrects, alltexts)) + onmouseover(mouseevents) do events r.color = cell_color_hover[] end - onmouseout(mousestate) do state + onmouseout(mouseevents) do events if i == 1 r.color = selection_cell_color_inactive[] else @@ -244,7 +244,7 @@ function LMenu(parent::Scene; bbox = nothing, kwargs...) end end - onmouseleftdown(mousestate) do state + onmouseleftdown(mouseevents) do events r.color = cell_color_active[] if is_open[] # first item is already selected @@ -257,7 +257,7 @@ function LMenu(parent::Scene; bbox = nothing, kwargs...) end # close the menu if the user clicks somewhere else - onmousedownoutside(addmouseevents!(scene)) do state + onmousedownoutside(addmouseevents!(scene)) do events if is_open[] is_open[] = !is_open[] end From ae7021ab263614472e7eb80f8ce73f4386cee226 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Wed, 28 Oct 2020 10:48:59 +0100 Subject: [PATCH 10/30] clean up slider, reacts on thick line, not scene --- src/makielayout/defaultattributes.jl | 4 ++-- src/makielayout/lobjects/lslider.jl | 26 ++++++++++++-------------- 2 files changed, 14 insertions(+), 16 deletions(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index 3b693bc47..4e75327d6 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -522,9 +522,9 @@ function default_attributes(::Type{LSlider}, scene) "The color of the slider when the mouse clicks and drags the slider." color_active = COLOR_ACCENT[] "The color of the slider when it is not interacted with." - color_inactive = RGBf0(0.9, 0.9, 0.9) + color_inactive = RGBf0(0.94, 0.94, 0.94) "The color of the button when it is not interacted with." - buttoncolor_inactive = RGBf0(1, 1, 1) + buttoncolor_inactive = RGBf0(0.3, 0.3, 0.3) "Controls if the slider has a horizontal orientation or not." horizontal = true "The line width of the slider button's border." diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 1b4d52ff9..6ab2009ed 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -85,7 +85,8 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) buttonpoint = lift(sliderbox, horizontal, displayed_sliderfraction, buttonradius, buttonstrokewidth) do bb, horizontal, sf, brad, bstw - pad = brad + bstw + # pad = brad #+ bstw + pad = 0f0 if horizontal [Point2f0(left(bb) + pad + (width(bb) - 2pad) * sf, bottom(bb) + height(bb) / 2)] @@ -102,22 +103,23 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) [ca, ci] end - linesegs = linesegments!(subscene, linepoints, color = linecolors, linewidth = linewidth, raw = true)[end] + bsize = @lift($buttonradius * 2f0) + + linesegs = linesegments!(subscene, linepoints, color = linecolors, linewidth = bsize, raw = true)[end] decorations[:linesegments] = linesegs linestate = addmouseevents!(subscene, linesegs) - bsize = @lift($buttonradius * 2f0) bcolor = Node{Any}(buttoncolor_inactive[]) - button = scatter!(subscene, buttonpoint, markersize = bsize, color = bcolor, marker = '⚫', - strokewidth = buttonstrokewidth, strokecolor = color_active_dimmed, raw = true)[end] + button = scatter!(subscene, buttonpoint, markersize = bsize, color = bcolor, marker = FRect2D(0, 0, 1, 1), + strokewidth = 0, strokecolor = color_active_dimmed, raw = true, visible = false)[end] decorations[:button] = button - mouseevents = addmouseevents!(subscene) + mouseevents = addmouseevents!(subscene, linesegs) onmouseleftup(mouseevents) do event bcolor[] = buttoncolor_inactive[] @@ -125,14 +127,12 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) onmouseleftdrag(mouseevents) do event - pad = buttonradius[] + buttonstrokewidth[] - dragging[] = true dif = event.px - event.prev_px fraction = if horizontal[] - dif[1] / (width(sliderbox[]) - 2pad) + dif[1] / width(sliderbox[]) else - dif[2] / (height(sliderbox[]) - 2pad) + dif[2] / height(sliderbox[]) end if fraction != 0.0f0 newfraction = min(max(displayed_sliderfraction[] + fraction, 0f0), 1f0) @@ -153,13 +153,11 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) onmouseleftdown(mouseevents) do event - bcolor[] = color_active[] - - pad = buttonradius[] + buttonstrokewidth[] + # bcolor[] = color_active[] pos = event.px dim = horizontal[] ? 1 : 2 - frac = (pos[dim] - endpoints[][1][dim] - pad) / (endpoints[][2][dim] - endpoints[][1][dim] - 2pad) + frac = (pos[dim] - endpoints[][1][dim]) / (endpoints[][2][dim] - endpoints[][1][dim]) selected_index[] = closest_fractionindex(sliderrange[], frac) end From 1267f57266cb92f9605a940732e431a39c16bdb4 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Wed, 28 Oct 2020 10:53:30 +0100 Subject: [PATCH 11/30] assimilate button color --- src/makielayout/defaultattributes.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index 4e75327d6..b87782f63 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -433,7 +433,7 @@ function default_attributes(::Type{LButton}, scene) "The color of the button border." strokecolor = :transparent "The color of the button." - buttoncolor = RGBf0(0.9, 0.9, 0.9) + buttoncolor = RGBf0(0.94, 0.94, 0.94) "The color of the label." labelcolor = :black "The color of the label when the mouse hovers over the button." From 9a5c3f4a8c96ded31c8fc1980a4a1cf25be5c01c Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Wed, 28 Oct 2020 12:24:29 +0100 Subject: [PATCH 12/30] simplify slider --- src/makielayout/defaultattributes.jl | 10 +--- src/makielayout/lobjects/lslider.jl | 79 +++++----------------------- src/makielayout/types.jl | 1 - 3 files changed, 15 insertions(+), 75 deletions(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index b87782f63..d296d4e22 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -495,8 +495,6 @@ end function default_attributes(::Type{LSlider}, scene) attrs, docdict, defaultdict = @documented_attributes begin - "The line width of the main slider line." - linewidth = 4f0 "The horizontal alignment of the slider in its suggested bounding box." halign = :center "The vertical alignment of the slider in its suggested bounding box." @@ -504,15 +502,13 @@ function default_attributes(::Type{LSlider}, scene) "The width setting of the slider." width = nothing "The height setting of the slider." - height = Auto() + height = 15 "The range of values that the slider can pick from." range = 0:10 "Controls if the parent layout can adjust to this element's width" tellwidth = true "Controls if the parent layout can adjust to this element's height" tellheight = true - "The radius of the slider button." - buttonradius = 9f0 "The start value of the slider or the value that is closest in the slider range." startvalue = 0 "The current value of the slider." @@ -523,12 +519,8 @@ function default_attributes(::Type{LSlider}, scene) color_active = COLOR_ACCENT[] "The color of the slider when it is not interacted with." color_inactive = RGBf0(0.94, 0.94, 0.94) - "The color of the button when it is not interacted with." - buttoncolor_inactive = RGBf0(0.3, 0.3, 0.3) "Controls if the slider has a horizontal orientation or not." horizontal = true - "The line width of the slider button's border." - buttonstrokewidth = 4f0 "The align mode of the slider in its parent GridLayout." alignmode = Inside() end diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 6ab2009ed..287bbb983 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -7,9 +7,8 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) decorations = Dict{Symbol, Any}() @extract attrs ( - halign, valign, linewidth, buttonradius, horizontal, - startvalue, value, color_active, color_active_dimmed, color_inactive, - buttonstrokewidth, buttoncolor_inactive + halign, valign, horizontal, + startvalue, value, color_active, color_active_dimmed, color_inactive ) sliderrange = attrs.range @@ -18,23 +17,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) layoutobservables = LayoutObservables{LSlider}(attrs.width, attrs.height, attrs.tellwidth, attrs.tellheight, halign, valign, attrs.alignmode; suggestedbbox = bbox, protrusions = protrusions) - onany(buttonradius, horizontal, buttonstrokewidth) do br, hori, bstrw - layoutobservables.autosize[] = if hori - (nothing, 2 * (br + bstrw)) - else - (2 * (br + bstrw), nothing) - end - end - - subarea = lift(layoutobservables.computedbbox) do bbox - round_to_IRect2D(bbox) - end - - # the slider gets its own subscene so a click doesn't have to hit the line - # perfectly but can be registered in the whole area that the slider scene has - subscene = Scene(parent, subarea, camera=campixel!) - - sliderbox = lift(bb -> Rect{2, Float32}(zeros(eltype(bb.origin), 2), bb.widths), layoutobservables.computedbbox) + sliderbox = lift(identity, layoutobservables.computedbbox) endpoints = lift(sliderbox, horizontal) do bb, horizontal @@ -82,48 +65,31 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) # initialize slider value with closest from range selected_index[] = closest_index(sliderrange[], startvalue[]) - buttonpoint = lift(sliderbox, horizontal, displayed_sliderfraction, buttonradius, - buttonstrokewidth) do bb, horizontal, sf, brad, bstw - - # pad = brad #+ bstw - pad = 0f0 + middlepoint = lift(sliderbox, horizontal, displayed_sliderfraction) do bb, horizontal, sf if horizontal - [Point2f0(left(bb) + pad + (width(bb) - 2pad) * sf, bottom(bb) + height(bb) / 2)] + Point2f0(left(bb) + width(bb) * sf, bottom(bb) + height(bb) / 2) else - [Point2f0(left(bb) + 0.5f0 * width(bb), bottom(bb) + pad + (height(bb) - 2pad) * sf)] + Point2f0(left(bb) + 0.5f0 * width(bb), bottom(bb) + height(bb) * sf) end end - linepoints = lift(endpoints, buttonpoint) do eps, bp - [eps[1], bp[1], bp[1], eps[2]] + linepoints = lift(endpoints, middlepoint) do eps, middle + [eps[1], middle, middle, eps[2]] end linecolors = lift(color_active_dimmed, color_inactive) do ca, ci [ca, ci] end - bsize = @lift($buttonradius * 2f0) + linewidth = lift(horizontal, sliderbox) do hori, sbox + hori ? height(sbox) : width(sbox) + end - linesegs = linesegments!(subscene, linepoints, color = linecolors, linewidth = bsize, raw = true)[end] + linesegs = linesegments!(parent, linepoints, color = linecolors, linewidth = linewidth, raw = true)[end] decorations[:linesegments] = linesegs - linestate = addmouseevents!(subscene, linesegs) - - - bcolor = Node{Any}(buttoncolor_inactive[]) - - - button = scatter!(subscene, buttonpoint, markersize = bsize, color = bcolor, marker = FRect2D(0, 0, 1, 1), - strokewidth = 0, strokecolor = color_active_dimmed, raw = true, visible = false)[end] - decorations[:button] = button - - - mouseevents = addmouseevents!(subscene, linesegs) - - onmouseleftup(mouseevents) do event - bcolor[] = buttoncolor_inactive[] - end + mouseevents = addmouseevents!(parent, linesegs) onmouseleftdrag(mouseevents) do event @@ -153,8 +119,6 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) onmouseleftdown(mouseevents) do event - # bcolor[] = color_active[] - pos = event.px dim = horizontal[] ? 1 : 2 frac = (pos[dim] - endpoints[][1][dim]) / (endpoints[][2][dim] - endpoints[][1][dim]) @@ -166,32 +130,17 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) end onmouseenter(mouseevents) do event - # bcolor[] = color_active[] linecolors[] = [color_active[], color_inactive[]] - button.strokecolor = color_active[] end onmouseout(mouseevents) do event - bcolor[] = buttoncolor_inactive[] linecolors[] = [color_active_dimmed[], color_inactive[]] - button.strokecolor = color_active_dimmed[] end - onany(buttonradius, horizontal) do br, horizontal - protrusions[] = if horizontal - GridLayoutBase.RectSides{Float32}(br, br, 0, 0) - else - GridLayoutBase.RectSides{Float32}(0, 0, br, br) - end - end - - # trigger protrusions using one observable - buttonradius[] = buttonradius[] - # trigger bbox layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] - LSlider(parent, subscene, layoutobservables, attrs, decorations) + LSlider(parent, layoutobservables, attrs, decorations) end function valueindex(sliderrange, value) diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 63334f834..0742ff474 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -117,7 +117,6 @@ end struct LSlider <: LObject parent::Scene - scene::Scene layoutobservables::LayoutObservables attributes::Attributes decorations::Dict{Symbol, Any} From 08bc4c0503556fa75ecd95f542c7614d771741d2 Mon Sep 17 00:00:00 2001 From: jkrumbiegel <22495855+jkrumbiegel@users.noreply.github.com> Date: Wed, 28 Oct 2020 14:11:18 +0100 Subject: [PATCH 13/30] no abstractinteraction, enable Functions, move to interactions.jl --- src/makielayout/MakieLayout.jl | 1 + src/makielayout/interactions.jl | 197 ++++++++++++++++++++++++++++++ src/makielayout/lobjects/laxis.jl | 187 +--------------------------- src/makielayout/types.jl | 12 +- 4 files changed, 204 insertions(+), 193 deletions(-) create mode 100644 src/makielayout/interactions.jl diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index 680438b94..2083428e8 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -50,6 +50,7 @@ include("ticklocators/linear.jl") include("ticklocators/wilkinson.jl") include("defaultattributes.jl") include("lineaxis.jl") +include("interactions.jl") include("lobjects/laxis.jl") include("lobjects/lcolorbar.jl") include("lobjects/ltext.jl") diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl new file mode 100644 index 000000000..0412b4bc5 --- /dev/null +++ b/src/makielayout/interactions.jl @@ -0,0 +1,197 @@ +# overloadable for other types that might want to offer similar interactions +interactions(ax::LAxis) = ax.interactions + +function register_interaction!(parent, name::Symbol, interaction) + haskey(interactions(parent), name) && error("Interaction $name already exists.") + registration_setup!(parent, interaction) + push!(interactions(parent), name => (true, interaction)) + return interaction +end + +function deregister_interaction!(parent, name::Symbol) + !haskey(interactions(parent), name) && error("Interaction $name does not exist.") + _, interaction = interactions(parent)[name] + + deregistration_cleanup!(parent, interaction) + pop!(interactions(parent), name) + return interaction +end + +function registration_setup!(parent, interaction) + # do nothing in the default case +end + +function deregistration_cleanup!(parent, interaction) + # do nothing in the default case +end + +function activate_interaction!(parent, name::Symbol) + !haskey(interactions(parent), name) && error("Interaction $name does not exist.") + interactions(parent)[name] = (true, interactions(parent)[name][2]) + return nothing +end + +function deactivate_interaction!(parent, name::Symbol) + !haskey(interactions(parent), name) && error("Interaction $name does not exist.") + interactions(parent)[name] = (false, interactions(parent)[name][2]) + return nothing +end + +function process_interaction(@nospecialize args...) + # do nothing in the default case +end + +# a generic fallback for functions to have one really simple path to getting interactivity +# without needing to define a special type first +function process_interaction(f::Function, event, parent) + # in case f is only defined for a specific type of event + if applicable(f, event, parent) + f(event, parent) + end +end + + + +############################################################################ +# LAxis interactions +############################################################################ + +function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) + + if event.type === MouseEventTypes.leftdragstart + r.from = event.prev_data + r.to = event.data + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] + + elseif event.type === MouseEventTypes.leftdrag + r.to = event.data + r.rectnode[] = FRect2D(r.from, r.to .- r.from) + + elseif event.type === MouseEventTypes.leftdragstop + newlims = positivize(r.rectnode[]) + if !(0 in widths(newlims)) + ax.targetlimits[] = newlims + end + + if !isnothing(r.poly) + delete!(ax.scene, r.poly) + r.poly = nothing + end + end + + return nothing +end + + +function positivize(r::FRect2D) + negwidths = r.widths .< 0 + newori = ifelse.(negwidths, r.origin .+ r.widths, r.origin) + newwidths = ifelse.(negwidths, -r.widths, r.widths) + FRect2D(newori, newwidths) +end + + +function process_interaction(l::LimitReset, event::MouseEvent, ax::LAxis) + + if event.type === MouseEventTypes.leftclick + if ispressed(ax.scene, Keyboard.left_control) + autolimits!(ax) + end + end + + return nothing +end + + +function process_interaction(s::ScrollZoom, event::ScrollEvent, ax::LAxis) + # use vertical zoom + zoom = event.y + + tlimits = ax.targetlimits + xzoomlock = ax.xzoomlock + yzoomlock = ax.yzoomlock + xzoomkey = ax.xzoomkey + yzoomkey = ax.yzoomkey + + scene = ax.scene + e = events(scene) + cam = camera(scene) + + if zoom != 0 + pa = pixelarea(scene)[] + + # don't let z go negative + z = max(0.1f0, 1f0 - (abs(zoom) * s.speed)) + if zoom > 0 + z = 1/z # sets the old to be a fraction of the new. This ensures zoom in & then out returns to original position. + end + + mp_axscene = Vec4f0((e.mouseposition[] .- pa.origin)..., 0, 1) + + # first to normal -1..1 space + mp_axfraction = (cam.pixel_space[] * mp_axscene)[1:2] .* + # now to 1..-1 if an axis is reversed to correct zoom point + (-2 .* ((ax.xreversed[], ax.yreversed[])) .+ 1) .* + # now to 0..1 + 0.5 .+ 0.5 + + xorigin = tlimits[].origin[1] + yorigin = tlimits[].origin[2] + + xwidth = tlimits[].widths[1] + ywidth = tlimits[].widths[2] + + newxwidth = xzoomlock[] ? xwidth : xwidth * z + newywidth = yzoomlock[] ? ywidth : ywidth * z + + newxorigin = xzoomlock[] ? xorigin : xorigin + mp_axfraction[1] * (xwidth - newxwidth) + newyorigin = yzoomlock[] ? yorigin : yorigin + mp_axfraction[2] * (ywidth - newywidth) + + timed_ticklabelspace_reset(ax, s.reset_timer, s.prev_xticklabelspace, s.prev_yticklabelspace, s.reset_delay) + + tlimits[] = if ispressed(scene, xzoomkey[]) + FRect(newxorigin, yorigin, newxwidth, ywidth) + elseif ispressed(scene, yzoomkey[]) + FRect(xorigin, newyorigin, xwidth, newywidth) + else + FRect(newxorigin, newyorigin, newxwidth, newywidth) + end + + end +end + +function process_interaction(dp::DragPan, event::MouseEvent, ax) + + if event.type !== MouseEventTypes.rightdrag + return nothing + end + + tlimits = ax.targetlimits + xpanlock = ax.xpanlock + ypanlock = ax.ypanlock + xpankey = ax.xpankey + ypankey = ax.ypankey + panbutton = ax.panbutton + + scene = ax.scene + + movement = AbstractPlotting.to_world(ax.scene, event.px) .- + AbstractPlotting.to_world(ax.scene, event.prev_px) + + xori, yori = Vec2f0(tlimits[].origin) .- movement + + if xpanlock[] || ispressed(scene, ypankey[]) + xori = tlimits[].origin[1] + end + + if ypanlock[] || ispressed(scene, xpankey[]) + yori = tlimits[].origin[2] + end + + timed_ticklabelspace_reset(ax, dp.reset_timer, dp.prev_xticklabelspace, dp.prev_yticklabelspace, dp.reset_delay) + + tlimits[] = FRect(Vec2f0(xori, yori), widths(tlimits[])) + + return nothing +end diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index 95255671c..b64070940 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -302,7 +302,7 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) end end - interactions = Dict{Symbol, AbstractInteraction}() + interactions = Dict{Symbol, Tuple{Bool, Any}}() la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, layoutobservables, attrs, block_limit_linking, decorations, @@ -343,191 +343,6 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) la end -####################################### - -function register_interaction!(ax::LAxis, name::Symbol, interaction::AbstractInteraction) - haskey(ax.interactions, name) && error("Interaction $name already exists.") - registration_setup!(ax, interaction) - push!(ax.interactions, name => (true, interaction)) - return interaction -end - -function deregister_interaction!(ax::LAxis, name::Symbol) - !haskey(ax.interactions, name) && error("Interaction $name does not exist.") - _, interaction = ax.interactions[name] - - deregistration_cleanup!(ax, interaction) - pop!(ax.interactions, name) - return interaction -end - -function registration_setup!(ax, interaction) - # do nothing in the default case -end - -function deregistration_cleanup!(ax, interaction) - # do nothing in the default case -end - -function activate_interaction!(ax, name::Symbol) - !haskey(ax.interactions, name) && error("Interaction $name does not exist.") - ax.interactions[name] = (true, ax.interactions[name][2]) - return nothing -end - -function deactivate_interaction!(ax, name::Symbol) - !haskey(ax.interactions, name) && error("Interaction $name does not exist.") - ax.interactions[name] = (false, ax.interactions[name][2]) - return nothing -end - -function process_interaction(@nospecialize args...) - # do nothing in the default case -end - - -function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) - - if event.type === MouseEventTypes.leftdragstart - r.from = event.prev_data - r.to = event.data - r.rectnode[] = FRect2D(r.from, r.to .- r.from) - r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] - - elseif event.type === MouseEventTypes.leftdrag - r.to = event.data - r.rectnode[] = FRect2D(r.from, r.to .- r.from) - - elseif event.type === MouseEventTypes.leftdragstop - newlims = positivize(r.rectnode[]) - if !(0 in widths(newlims)) - ax.targetlimits[] = newlims - end - - if !isnothing(r.poly) - delete!(ax.scene, r.poly) - r.poly = nothing - end - end - - return nothing -end - - -function positivize(r::FRect2D) - negwidths = r.widths .< 0 - newori = ifelse.(negwidths, r.origin .+ r.widths, r.origin) - newwidths = ifelse.(negwidths, -r.widths, r.widths) - FRect2D(newori, newwidths) -end - - -function process_interaction(l::LimitReset, event::MouseEvent, ax::LAxis) - - if event.type === MouseEventTypes.leftclick - if ispressed(ax.scene, Keyboard.left_control) - autolimits!(ax) - end - end - - return nothing -end - - -function process_interaction(s::ScrollZoom, event::ScrollEvent, ax::LAxis) - # use vertical zoom - zoom = event.y - - tlimits = ax.targetlimits - xzoomlock = ax.xzoomlock - yzoomlock = ax.yzoomlock - xzoomkey = ax.xzoomkey - yzoomkey = ax.yzoomkey - - scene = ax.scene - e = events(scene) - cam = camera(scene) - - if zoom != 0 - pa = pixelarea(scene)[] - - # don't let z go negative - z = max(0.1f0, 1f0 - (abs(zoom) * s.speed)) - if zoom > 0 - z = 1/z # sets the old to be a fraction of the new. This ensures zoom in & then out returns to original position. - end - - mp_axscene = Vec4f0((e.mouseposition[] .- pa.origin)..., 0, 1) - - # first to normal -1..1 space - mp_axfraction = (cam.pixel_space[] * mp_axscene)[1:2] .* - # now to 1..-1 if an axis is reversed to correct zoom point - (-2 .* ((ax.xreversed[], ax.yreversed[])) .+ 1) .* - # now to 0..1 - 0.5 .+ 0.5 - - xorigin = tlimits[].origin[1] - yorigin = tlimits[].origin[2] - - xwidth = tlimits[].widths[1] - ywidth = tlimits[].widths[2] - - newxwidth = xzoomlock[] ? xwidth : xwidth * z - newywidth = yzoomlock[] ? ywidth : ywidth * z - - newxorigin = xzoomlock[] ? xorigin : xorigin + mp_axfraction[1] * (xwidth - newxwidth) - newyorigin = yzoomlock[] ? yorigin : yorigin + mp_axfraction[2] * (ywidth - newywidth) - - timed_ticklabelspace_reset(ax, s.reset_timer, s.prev_xticklabelspace, s.prev_yticklabelspace, s.reset_delay) - - tlimits[] = if ispressed(scene, xzoomkey[]) - FRect(newxorigin, yorigin, newxwidth, ywidth) - elseif ispressed(scene, yzoomkey[]) - FRect(xorigin, newyorigin, xwidth, newywidth) - else - FRect(newxorigin, newyorigin, newxwidth, newywidth) - end - - end -end - -function process_interaction(dp::DragPan, event::MouseEvent, ax) - - if event.type !== MouseEventTypes.rightdrag - return nothing - end - - tlimits = ax.targetlimits - xpanlock = ax.xpanlock - ypanlock = ax.ypanlock - xpankey = ax.xpankey - ypankey = ax.ypankey - panbutton = ax.panbutton - - scene = ax.scene - - movement = AbstractPlotting.to_world(ax.scene, event.px) .- - AbstractPlotting.to_world(ax.scene, event.prev_px) - - xori, yori = Vec2f0(tlimits[].origin) .- movement - - if xpanlock[] || ispressed(scene, ypankey[]) - xori = tlimits[].origin[1] - end - - if ypanlock[] || ispressed(scene, xpankey[]) - yori = tlimits[].origin[2] - end - - timed_ticklabelspace_reset(ax, dp.reset_timer, dp.prev_xticklabelspace, dp.prev_yticklabelspace, dp.reset_delay) - - tlimits[] = FRect(Vec2f0(xori, yori), widths(tlimits[])) - - return nothing -end - -####################################### - function AbstractPlotting.plot!( la::LAxis, P::AbstractPlotting.PlotFunc, diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index 0742ff474..ba2a0a59d 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -46,18 +46,16 @@ mutable struct LineAxis ticklabels::Node{Vector{String}} end -abstract type AbstractInteraction end +struct LimitReset end -struct LimitReset <: AbstractInteraction end - -mutable struct RectangleZoom <: AbstractInteraction +mutable struct RectangleZoom from::Union{Nothing, Point2f0} to::Union{Nothing, Point2f0} rectnode::Observable{FRect2D} poly::Union{Poly, Nothing} end -struct ScrollZoom <: AbstractInteraction +struct ScrollZoom speed::Float32 reset_timer::Ref{Any} prev_xticklabelspace::Ref{Any} @@ -65,7 +63,7 @@ struct ScrollZoom <: AbstractInteraction reset_delay::Float32 end -struct DragPan <: AbstractInteraction +struct DragPan reset_timer::Ref{Any} prev_xticklabelspace::Ref{Any} prev_yticklabelspace::Ref{Any} @@ -91,7 +89,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene decorations::Dict{Symbol, Any} mouseevents::Observable{MouseEvent} scrollevents::Observable{ScrollEvent} - interactions::Dict{Symbol, Tuple{Bool, AbstractInteraction}} + interactions::Dict{Symbol, Tuple{Bool, Any}} end mutable struct LColorbar <: LObject From a2f9667aa855d15356120c035a5dd7ff297b7e64 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 09:18:41 +0100 Subject: [PATCH 14/30] export interaction stuff --- src/makielayout/MakieLayout.jl | 1 + 1 file changed, 1 insertion(+) diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index 2083428e8..1db041edb 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -87,6 +87,7 @@ export set_close_to! export xaxis_bottom!, xaxis_top!, yaxis_left!, yaxis_right! export labelslider! export addmouseevents! +export interactions, register_interaction!, deregister_interaction!, activate_interaction!, deactivate_interaction! export hlines!, vlines! From 479207e8ecf14047f2e7b38d8640b8fe463eba31 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 09:29:03 +0100 Subject: [PATCH 15/30] add docstrings --- src/makielayout/interactions.jl | 31 ++++++++++++++++++++++++++++++- 1 file changed, 30 insertions(+), 1 deletion(-) diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index 0412b4bc5..15939c197 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -1,6 +1,18 @@ # overloadable for other types that might want to offer similar interactions +function interactions end + interactions(ax::LAxis) = ax.interactions +""" + register_interaction!(parent, name::Symbol, interaction) + +Register `interaction` with `parent` under the name `name`. +The parent will call `process_interaction(interaction, event, parent)` +whenever suitable events happen. + +The interaction can be removed with `deregister_interaction!` or temporarily +toggled with `activate_interaction!` / `deactivate_interaction!`. +""" function register_interaction!(parent, name::Symbol, interaction) haskey(interactions(parent), name) && error("Interaction $name already exists.") registration_setup!(parent, interaction) @@ -8,6 +20,11 @@ function register_interaction!(parent, name::Symbol, interaction) return interaction end +""" + deregister_interaction!(parent, name::Symbol) + +Deregister the interaction named `name` registered in `parent`. +""" function deregister_interaction!(parent, name::Symbol) !haskey(interactions(parent), name) && error("Interaction $name does not exist.") _, interaction = interactions(parent)[name] @@ -25,18 +42,30 @@ function deregistration_cleanup!(parent, interaction) # do nothing in the default case end +""" + activate_interaction!(parent, name::Symbol) + +Activate the interaction named `name` registered in `parent`. +""" function activate_interaction!(parent, name::Symbol) !haskey(interactions(parent), name) && error("Interaction $name does not exist.") interactions(parent)[name] = (true, interactions(parent)[name][2]) return nothing end +""" + deactivate_interaction!(parent, name::Symbol) + +Deactivate the interaction named `name` registered in `parent`. +It can be reactivated with `activate_interaction!`. +""" function deactivate_interaction!(parent, name::Symbol) !haskey(interactions(parent), name) && error("Interaction $name does not exist.") interactions(parent)[name] = (false, interactions(parent)[name][2]) return nothing end + function process_interaction(@nospecialize args...) # do nothing in the default case end @@ -53,7 +82,7 @@ end ############################################################################ -# LAxis interactions +# LAxis interactions # ############################################################################ function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) From 5c4a73603b0d1e656c38072a6b10b927fb132744 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 10:21:50 +0100 Subject: [PATCH 16/30] add keyevents, allow restriction by keys --- src/makielayout/interactions.jl | 32 ++++++++++++++++++++++++++++--- src/makielayout/lobjects/laxis.jl | 11 +++++++++-- src/makielayout/types.jl | 8 ++++++++ 3 files changed, 46 insertions(+), 5 deletions(-) diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index 15939c197..d1c03ae9f 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -85,20 +85,36 @@ end # LAxis interactions # ############################################################################ +function _chosen_limits(rz, ax) + + r = positivize(FRect2D(rz.from, rz.to .- rz.from)) + lims = ax.limits[] + # restrict to y change + if rz.restrict_x + r = FRect2D(lims.origin[1], r.origin[2], widths(lims)[1], widths(r)[2]) + end + # restrict to x change + if rz.restrict_y + r = FRect2D(r.origin[1], lims.origin[2], widths(r)[1], widths(lims)[2]) + end + return r +end + function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) if event.type === MouseEventTypes.leftdragstart r.from = event.prev_data r.to = event.data - r.rectnode[] = FRect2D(r.from, r.to .- r.from) + r.rectnode[] = _chosen_limits(r, ax) r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] + r.active = true elseif event.type === MouseEventTypes.leftdrag r.to = event.data - r.rectnode[] = FRect2D(r.from, r.to .- r.from) + r.rectnode[] = _chosen_limits(r, ax) elseif event.type === MouseEventTypes.leftdragstop - newlims = positivize(r.rectnode[]) + newlims = r.rectnode[] if !(0 in widths(newlims)) ax.targetlimits[] = newlims end @@ -107,11 +123,21 @@ function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) delete!(ax.scene, r.poly) r.poly = nothing end + r.active = false end return nothing end +function process_interaction(r::RectangleZoom, event::KeysEvent, ax::LAxis) + r.restrict_y = Keyboard.x in event.keys + r.restrict_x = Keyboard.y in event.keys + r.active || return + + r.rectnode[] = _chosen_limits(r, ax) + return nothing +end + function positivize(r::FRect2D) negwidths = r.widths .< 0 diff --git a/src/makielayout/lobjects/laxis.jl b/src/makielayout/lobjects/laxis.jl index b64070940..14827c907 100644 --- a/src/makielayout/lobjects/laxis.jl +++ b/src/makielayout/lobjects/laxis.jl @@ -296,17 +296,23 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) mouseevents = addmouseevents!(scene) scrollevents = Node(ScrollEvent(0, 0)) + keysevents = Node(KeysEvent(Set())) + on(scene.events.scroll) do s if is_mouseinside(scene) scrollevents[] = ScrollEvent(s[1], s[2]) end end + on(scene.events.keyboardbuttons) do buttons + keysevents[] = KeysEvent(buttons) + end + interactions = Dict{Symbol, Tuple{Bool, Any}}() la = LAxis(parent, scene, xaxislinks, yaxislinks, limits, layoutobservables, attrs, block_limit_linking, decorations, - mouseevents, scrollevents, interactions) + mouseevents, scrollevents, keysevents, interactions) function process_event(event) @@ -317,10 +323,11 @@ function LAxis(parent::Scene; bbox = nothing, kwargs...) on(process_event, mouseevents) on(process_event, scrollevents) + on(process_event, keysevents) register_interaction!(la, :rectanglezoom, - RectangleZoom(nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing)) + RectangleZoom(false, false, false, nothing, nothing, Node(FRect2D(0, 0, 1, 1)), nothing)) register_interaction!(la, :limitreset, diff --git a/src/makielayout/types.jl b/src/makielayout/types.jl index ba2a0a59d..a11ee14d8 100644 --- a/src/makielayout/types.jl +++ b/src/makielayout/types.jl @@ -49,6 +49,9 @@ end struct LimitReset end mutable struct RectangleZoom + active::Bool + restrict_x::Bool + restrict_y::Bool from::Union{Nothing, Point2f0} to::Union{Nothing, Point2f0} rectnode::Observable{FRect2D} @@ -75,6 +78,10 @@ struct ScrollEvent y::Float32 end +struct KeysEvent + keys::Set{AbstractPlotting.Keyboard.Button} +end + abstract type LObject end mutable struct LAxis <: AbstractPlotting.AbstractScene @@ -89,6 +96,7 @@ mutable struct LAxis <: AbstractPlotting.AbstractScene decorations::Dict{Symbol, Any} mouseevents::Observable{MouseEvent} scrollevents::Observable{ScrollEvent} + keysevents::Observable{KeysEvent} interactions::Dict{Symbol, Tuple{Bool, Any}} end From 53a0f428d3bd4b9f6d2f00ec89ca6bea94a6fc77 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:08:57 +0100 Subject: [PATCH 17/30] update slider with rounded ends and less distraction --- src/makielayout/lobjects/lslider.jl | 39 ++++++++++++++++++----------- 1 file changed, 24 insertions(+), 15 deletions(-) diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 287bbb983..b3ad39c54 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -21,14 +21,17 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) endpoints = lift(sliderbox, horizontal) do bb, horizontal + h = height(bb) + w = width(bb) + if horizontal - y = bottom(bb) + height(bb) / 2 - [Point2f0(left(bb), y), - Point2f0(right(bb), y)] + y = bottom(bb) + h / 2 + [Point2f0(left(bb) + h/2, y), + Point2f0(right(bb) - h/2, y)] else - x = left(bb) + width(bb) / 2 - [Point2f0(x, bottom(bb)), - Point2f0(x, top(bb))] + x = left(bb) + w / 2 + [Point2f0(x, bottom(bb) + w/2), + Point2f0(x, top(bb) + h/2)] end end @@ -65,13 +68,8 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) # initialize slider value with closest from range selected_index[] = closest_index(sliderrange[], startvalue[]) - middlepoint = lift(sliderbox, horizontal, displayed_sliderfraction) do bb, horizontal, sf - - if horizontal - Point2f0(left(bb) + width(bb) * sf, bottom(bb) + height(bb) / 2) - else - Point2f0(left(bb) + 0.5f0 * width(bb), bottom(bb) + height(bb) * sf) - end + middlepoint = lift(endpoints, displayed_sliderfraction) do ep, sf + Point2f0(ep[1] .+ sf .* (ep[2] .- ep[1])) end linepoints = lift(endpoints, middlepoint) do eps, middle @@ -86,10 +84,18 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) hori ? height(sbox) : width(sbox) end + endbuttons = scatter!(parent, endpoints, color = linecolors, markersize = linewidth, strokewidth = 0, raw = true)[end] + decorations[:endbuttons] = endbuttons + linesegs = linesegments!(parent, linepoints, color = linecolors, linewidth = linewidth, raw = true)[end] decorations[:linesegments] = linesegs - mouseevents = addmouseevents!(parent, linesegs) + button_magnification = Node(1.0) + buttonsize = @lift($linewidth * $button_magnification) + button = scatter!(parent, middlepoint, color = color_active, strokewidth = 0, markersize = buttonsize, raw = true)[end] + decorations[:button] = button + + mouseevents = addmouseevents!(parent, linesegs, button) onmouseleftdrag(mouseevents) do event @@ -115,6 +121,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) dragging[] = false # adjust slider to closest legal value sliderfraction[] = sliderfraction[] + linecolors[] = [color_active_dimmed[], color_inactive[]] end onmouseleftdown(mouseevents) do event @@ -123,6 +130,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) dim = horizontal[] ? 1 : 2 frac = (pos[dim] - endpoints[][1][dim]) / (endpoints[][2][dim] - endpoints[][1][dim]) selected_index[] = closest_fractionindex(sliderrange[], frac) + # linecolors[] = [color_active[], color_inactive[]] end onmouseleftdoubleclick(mouseevents) do event @@ -130,10 +138,11 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) end onmouseenter(mouseevents) do event - linecolors[] = [color_active[], color_inactive[]] + button_magnification[] = 1.25 end onmouseout(mouseevents) do event + button_magnification[] = 1.0 linecolors[] = [color_active_dimmed[], color_inactive[]] end From 9c3b8676192e0511003a5f5e49251ca83a476138 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:13:37 +0100 Subject: [PATCH 18/30] remove stroke from ltoggle --- src/makielayout/lobjects/ltoggle.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/makielayout/lobjects/ltoggle.jl b/src/makielayout/lobjects/ltoggle.jl index a169d00c5..541700988 100644 --- a/src/makielayout/lobjects/ltoggle.jl +++ b/src/makielayout/lobjects/ltoggle.jl @@ -51,7 +51,7 @@ function LToggle(parent::Scene; bbox = nothing, kwargs...) ms * (1 - rf) end - button = scatter!(parent, buttonpos, markersize = buttonsize, color = buttoncolor, raw = true)[end] + button = scatter!(parent, buttonpos, markersize = buttonsize, color = buttoncolor, strokewidth = 0, raw = true)[end] decorations[:button] = button buttonstate = addmouseevents!(parent, button, frame) From f513c0ff2a9d84b542946cd26e4c7a90031a7eac Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:19:55 +0100 Subject: [PATCH 19/30] add closure version for function registration --- src/makielayout/interactions.jl | 18 ++++++++++++++++++ 1 file changed, 18 insertions(+) diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index d1c03ae9f..85a794535 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -20,6 +20,24 @@ function register_interaction!(parent, name::Symbol, interaction) return interaction end +""" + register_interaction!(interaction::Function, parent, name::Symbol) + +Register `interaction` with `parent` under the name `name`. +The parent will call `process_interaction(interaction, event, parent)` +whenever suitable events happen. +This form with the first `Function` argument is especially intended for `do` syntax. + +The interaction can be removed with `deregister_interaction!` or temporarily +toggled with `activate_interaction!` / `deactivate_interaction!`. +""" +function register_interaction!(interaction::Function, parent, name::Symbol) + haskey(interactions(parent), name) && error("Interaction $name already exists.") + registration_setup!(parent, interaction) + push!(interactions(parent), name => (true, interaction)) + return interaction +end + """ deregister_interaction!(parent, name::Symbol) From e022588cf7641aed884559ea286d82f5c05bdc55 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:20:03 +0100 Subject: [PATCH 20/30] export event stuff --- src/makielayout/MakieLayout.jl | 1 + 1 file changed, 1 insertion(+) diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index 1db041edb..13ccd1551 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -88,6 +88,7 @@ export xaxis_bottom!, xaxis_top!, yaxis_left!, yaxis_right! export labelslider! export addmouseevents! export interactions, register_interaction!, deregister_interaction!, activate_interaction!, deactivate_interaction! +export MouseEventTypes, MouseEvent, ScrollEvent, KeysEvent export hlines!, vlines! From e3ab995e690ea19ca88603b0d3df2aa3a987b5b1 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:30:45 +0100 Subject: [PATCH 21/30] add labelslidergrid! helper function --- src/makielayout/MakieLayout.jl | 2 +- src/makielayout/helpers.jl | 23 ++++++++++++++++++++++- 2 files changed, 23 insertions(+), 2 deletions(-) diff --git a/src/makielayout/MakieLayout.jl b/src/makielayout/MakieLayout.jl index 13ccd1551..03aec2189 100644 --- a/src/makielayout/MakieLayout.jl +++ b/src/makielayout/MakieLayout.jl @@ -85,7 +85,7 @@ export tight_xticklabel_spacing!, tight_yticklabel_spacing!, tight_ticklabel_spa export layoutscene export set_close_to! export xaxis_bottom!, xaxis_top!, yaxis_left!, yaxis_right! -export labelslider! +export labelslider!, labelslidergrid! export addmouseevents! export interactions, register_interaction!, deregister_interaction!, activate_interaction!, deactivate_interaction! export MouseEventTypes, MouseEvent, ScrollEvent, KeysEvent diff --git a/src/makielayout/helpers.jl b/src/makielayout/helpers.jl index 13d4f1694..689a69a9d 100644 --- a/src/makielayout/helpers.jl +++ b/src/makielayout/helpers.jl @@ -426,7 +426,7 @@ layout[1, 1] = ls.layout ``` """ function labelslider!(scene, label, range; format = string, - sliderkw = Dict(), labelkw = Dict(), valuekw = Dict(), layoutkw...) + sliderkw = Dict(), labelkw = Dict(), valuekw = Dict(), layoutkw...) slider = LSlider(scene; range = range, sliderkw...) label = LText(scene, label; labelkw...) valuelabel = LText(scene, lift(format, slider.value); valuekw...) @@ -436,6 +436,27 @@ end +function labelslidergrid!(scene, labels, ranges; formats = [string], + sliderkw = Dict(), labelkw = Dict(), valuekw = Dict(), layoutkw...) + + elements = broadcast(labels, ranges, formats) do label, range, format + slider = LSlider(scene; range = range, sliderkw...) + label = LText(scene, label; halign = :left, labelkw...) + valuelabel = LText(scene, lift(format, slider.value); halign = :right, valuekw...) + (; slider = slider, label = label, valuelabel = valuelabel) + end + + sliders = map(x -> x.slider, elements) + labels = map(x -> x.label, elements) + valuelabels = map(x -> x.valuelabel, elements) + + layout = grid!(hcat(labels, sliders, valuelabels)) + + (sliders = sliders, labels = labels, valuelabels = valuelabels, layout = layout) +end + + + # helper function to create either h or vlines depending on `direction` # this works only with LAxes because it needs to react to limit changes function hvlines!(ax::LAxis, direction::Int, datavals, axmins, axmaxs; attributes...) From 27efedf2a584fd69d0cc27dba0e6419ec908b377 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 11:42:37 +0100 Subject: [PATCH 22/30] add docstring --- src/makielayout/helpers.jl | 23 +++++++++++++++++++++++ 1 file changed, 23 insertions(+) diff --git a/src/makielayout/helpers.jl b/src/makielayout/helpers.jl index 689a69a9d..05d667348 100644 --- a/src/makielayout/helpers.jl +++ b/src/makielayout/helpers.jl @@ -435,7 +435,30 @@ function labelslider!(scene, label, range; format = string, end +""" + labelslidergrid!(scene, labels, ranges; formats = [string], + sliderkw = Dict(), labelkw = Dict(), valuekw = Dict(), layoutkw...) + +Construct a GridLayout with a column of label, a column of sliders and a column of value labels in `scene`. +The argument values are broadcast, so you can use scalars if you want to keep labels, ranges or formats constant across rows. + +Returns a `NamedTuple`: + +`(sliders = sliders, labels = labels, valuelabels = valuelabels, layout = layout)` + +Specify format functions for the value labels with the `formats` keyword. +The sliders are forwarded the keywords from `sliderkw`. +The labels are forwarded the keywords from `labelkw`. +The value labels are forwarded the keywords from `valuekw`. +All other keywords are forwarded to the `GridLayout`. + +Example: +``` +ls = labelslidergrid!(scene, ["Voltage", "Ampere"], Ref(0:0.1:100); format = x -> "\$(x)V") +layout[1, 1] = ls.layout +``` +""" function labelslidergrid!(scene, labels, ranges; formats = [string], sliderkw = Dict(), labelkw = Dict(), valuekw = Dict(), layoutkw...) From ede930b0fe0edbe82ec7bc765f089fc58036e00b Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 13:35:43 +0100 Subject: [PATCH 23/30] update changelog --- CHANGELOG.md | 75 +++++++--------------------------------------------- 1 file changed, 10 insertions(+), 65 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index fb0a8d822..be8aadf93 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,71 +1,16 @@ -# Other branches (not yet merged) +# master -# `master` -- Changed default font to monospace in showlibrary (#190) -- Got `rotate!(Accum, ...)` working (#196) -- Updated Stepper documentation and allowed for custom file formats (#197) +## New features -# v0.9.10 -- Fixed two-dimensional limits when passed to `Scene` (#195) +- Added interactions model to LAxis. Interactions can be added with `register_interaction!` and removed with `deregister_interaction!`, as well as activated or deactivated temporarily with `activate_interaction!` and `deactivate_interaction!`. +- Added `labelslidergrid!`, a function to create an internally aligned grid of sliders with labels and value-labels. -# v0.9.9 -- Added a custom docstring extension which allows the Attributes of a Recipe to be shown in - the help mode (#174). -- Documented a lot of internal features (#174). -- Added a new 3d camera type, `cam3d_cad!`(#161). -- Improved warning text when displaying to text or plotpane (#163). -- Ensured that unless `inline!(true)` was called, plots will always display in - interactive displays, even in Juno (#163). -- Added licenses for fonts shipped with AbstractPlotting (#160). -- Switched from using system `ffmpeg` to using `FFMPEG.jl` (#160). -- Better docstrings for recording functions (#160). -- Let certain attributes passed to mutating plot functions affect the Scene (#160). -- Changed the default theme for `colorlegend` so that it scales with the resolution of the scene. -- Provided a way to generate a lower quality texture atlas via `set_glyph_resolution!(Low)` - to make the WebGL backend more lightweight (#180). -- Fixed `scale_plot` not actually working (#180). -- Cleaned up theme merging & scene attribute composition, so this works (#180): - ```julia - scatter(rand(4), resolution = (200, 200)) - scatter(rand(4), limits = ((0, 0), (200, 200))) - ``` +## Improvements -## Internal changes -- Replaced the `nothing` conversion trait with a new `NoConversion` trait, for clarity (#150). -- A lot of backend API changes to accomodate `WGLMakie` (#160). -- Enabled Gitlab CI! +- Cleaned up LSlider style and implementation, LSlider doesn't react while below other objects anymore +- MakieLayout Mouse event types are now enum instances for less compilation +- MakieLayout Mouse events store position in both data and pixel coordinates +- Aligned colors of LMenu, LSlider, LButton etc better -# v0.9.8 -## User interface changes -- Recipe docstrings are now associated with two functions instead of six (#116). -- New title recipe! (#99) -- Fixed buttons not respecting some attributes, and add a padding option (#114). -- Added `showlibrary(grad::Symbol)` function to show color libraries in `PlotUtils.jl` (#116). -- Added `showgradients` function to show an Array of gradients, indicated by Symbols (#116). -- Fixed incorrect frame duration in `record`. (#132) -- Fixed `save(path, io::VideoStream)` when file type is `.mkv` (#137). -- Improveed camera zooming (#140). +## Bugfixes -## Internal/other changes -- Changeed default alignment of buttons to (:center, :center) (#115) -- Documented a lot of recipes and functions -- Exported `HyperRectangle`, `update_limits!`, `update!`, and more... -- Updated `VideoStream`, `save` docstrings. -- Reworked tests to use `MakieGallery` (pretty big feature on the backend) -- Enabled Travis CI! - -# Additions - -## streamplot - -```julia -using MakieGallery, Makie -run_example("streamplot") -``` - -## timeseries - -```julia -using MakieGallery, Makie -run_example("timeseries") -``` From 5b56195c1250ad15749a1e5e8e4b130e825fa9b6 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:09:10 +0100 Subject: [PATCH 24/30] correct slider fraction --- src/makielayout/lobjects/lslider.jl | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index b3ad39c54..261013f02 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -102,9 +102,9 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) dragging[] = true dif = event.px - event.prev_px fraction = if horizontal[] - dif[1] / width(sliderbox[]) + dif[1] / (endpoints[][2][1] - endpoints[][1][1]) else - dif[2] / height(sliderbox[]) + dif[2] / (endpoints[][2][2] - endpoints[][1][2]) end if fraction != 0.0f0 newfraction = min(max(displayed_sliderfraction[] + fraction, 0f0), 1f0) From 0046646c8e00957196138fbe0bbc74ac861533a0 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:09:37 +0100 Subject: [PATCH 25/30] improve ltoggle look and feel a bit --- src/makielayout/defaultattributes.jl | 14 +++++++------- src/makielayout/lobjects/ltoggle.jl | 13 +++++++++++-- 2 files changed, 18 insertions(+), 9 deletions(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index d296d4e22..7608762c5 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -546,28 +546,28 @@ function default_attributes(::Type{LToggle}, scene) "The width of the toggle." width = 60 "The height of the toggle." - height = 30 + height = 28 "Controls if the parent layout can adjust to this element's width" tellwidth = true "Controls if the parent layout can adjust to this element's height" tellheight = true "The number of poly segments in each rounded corner." - cornersegments = 10 + cornersegments = 15 # strokewidth = 2f0 # strokecolor = :transparent "The color of the border when the toggle is inactive." - framecolor_inactive = RGBf0(0.9, 0.9, 0.9) - "The color of the border when the toggle is active." - framecolor_active = COLOR_ACCENT[] + framecolor_inactive = RGBf0(0.94, 0.94, 0.94) + "The color of the border when the toggle is hovered." + framecolor_active = COLOR_ACCENT_DIMMED[] # buttoncolor = RGBf0(0.2, 0.2, 0.2) "The color of the toggle button." - buttoncolor = RGBf0(1, 1, 1) + buttoncolor = COLOR_ACCENT[] "Indicates if the toggle is active or not." active = false "The duration of the toggle animation." toggleduration = 0.2 "The border width as a fraction of the toggle height " - rimfraction = 0.25 + rimfraction = 0.33 "The align mode of the toggle in its parent GridLayout." alignmode = Inside() end diff --git a/src/makielayout/lobjects/ltoggle.jl b/src/makielayout/lobjects/ltoggle.jl index 541700988..2b7b4c662 100644 --- a/src/makielayout/lobjects/ltoggle.jl +++ b/src/makielayout/lobjects/ltoggle.jl @@ -47,8 +47,9 @@ function LToggle(parent::Scene; bbox = nothing, kwargs...) end end - buttonsize = lift(markersize, rimfraction) do ms, rf - ms * (1 - rf) + buttonfactor = Node(1.0) + buttonsize = lift(markersize, rimfraction, buttonfactor) do ms, rf, bf + ms * (1 - rf) * bf end button = scatter!(parent, buttonpos, markersize = buttonsize, color = buttoncolor, strokewidth = 0, raw = true)[end] @@ -89,5 +90,13 @@ function LToggle(parent::Scene; bbox = nothing, kwargs...) end end + onmouseover(buttonstate) do state + buttonfactor[] = 1.15 + end + + onmouseout(buttonstate) do state + buttonfactor[] = 1.0 + end + LToggle(parent, layoutobservables, attrs, decorations) end From 2237fba02eda4999d5442e1e4ba5cc205a2d084e Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:13:09 +0100 Subject: [PATCH 26/30] align zoomrect style --- src/makielayout/interactions.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index 85a794535..0e4210940 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -124,7 +124,7 @@ function process_interaction(r::RectangleZoom, event::MouseEvent, ax::LAxis) r.from = event.prev_data r.to = event.data r.rectnode[] = _chosen_limits(r, ax) - r.poly = poly!(ax.scene, r.rectnode, color = (:blue, 0.1), strokewidth = 1, strokecolor = (:blue, 0.5))[end] + r.poly = poly!(ax.scene, r.rectnode, color = (COLOR_ACCENT[], 0.1), strokewidth = 2, strokecolor = COLOR_ACCENT[])[end] r.active = true elseif event.type === MouseEventTypes.leftdrag From d4604bebf49ca99b8e70a83a27a9f7bab79be0d3 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:28:18 +0100 Subject: [PATCH 27/30] allow disabling rectzoom directions in ax attributes --- src/makielayout/defaultattributes.jl | 4 ++++ src/makielayout/interactions.jl | 4 ++-- 2 files changed, 6 insertions(+), 2 deletions(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index 7608762c5..d2cee50b4 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -109,6 +109,10 @@ function default_attributes(::Type{LAxis}, scene) xzoomlock = false "Locks interactive zooming in the y direction." yzoomlock = false + "Controls if rectangle zooming affects the x dimension." + rectzoom_affect_x = true + "Controls if rectangle zooming affects the y dimension." + rectzoom_affect_y = true "The width of the axis spines." spinewidth = 1f0 "Controls if the x grid lines are visible." diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index 0e4210940..06e8a2e2a 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -108,11 +108,11 @@ function _chosen_limits(rz, ax) r = positivize(FRect2D(rz.from, rz.to .- rz.from)) lims = ax.limits[] # restrict to y change - if rz.restrict_x + if rz.restrict_x || !ax.rectzoom_affect_x[] r = FRect2D(lims.origin[1], r.origin[2], widths(lims)[1], widths(r)[2]) end # restrict to x change - if rz.restrict_y + if rz.restrict_y || !ax.rectzoom_affect_y[] r = FRect2D(r.origin[1], lims.origin[2], widths(r)[1], widths(lims)[2]) end return r From 4589043751b91768a4db034dd171a13887124798 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:35:40 +0100 Subject: [PATCH 28/30] rename rectzoom attr --- CHANGELOG.md | 1 + src/makielayout/defaultattributes.jl | 4 ++-- src/makielayout/interactions.jl | 4 ++-- 3 files changed, 5 insertions(+), 4 deletions(-) diff --git a/CHANGELOG.md b/CHANGELOG.md index be8aadf93..44ffc5fc7 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,7 @@ - Added interactions model to LAxis. Interactions can be added with `register_interaction!` and removed with `deregister_interaction!`, as well as activated or deactivated temporarily with `activate_interaction!` and `deactivate_interaction!`. - Added `labelslidergrid!`, a function to create an internally aligned grid of sliders with labels and value-labels. +- LAxis has attributes `xrectzoom` and `yrectzoom` that control if the rectangle zoom changes the respective dimension or not. ## Improvements diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index d2cee50b4..61997f5c2 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -110,9 +110,9 @@ function default_attributes(::Type{LAxis}, scene) "Locks interactive zooming in the y direction." yzoomlock = false "Controls if rectangle zooming affects the x dimension." - rectzoom_affect_x = true + xrectzoom = true "Controls if rectangle zooming affects the y dimension." - rectzoom_affect_y = true + yrectzoom = true "The width of the axis spines." spinewidth = 1f0 "Controls if the x grid lines are visible." diff --git a/src/makielayout/interactions.jl b/src/makielayout/interactions.jl index 06e8a2e2a..0a94b722c 100644 --- a/src/makielayout/interactions.jl +++ b/src/makielayout/interactions.jl @@ -108,11 +108,11 @@ function _chosen_limits(rz, ax) r = positivize(FRect2D(rz.from, rz.to .- rz.from)) lims = ax.limits[] # restrict to y change - if rz.restrict_x || !ax.rectzoom_affect_x[] + if rz.restrict_x || !ax.xrectzoom[] r = FRect2D(lims.origin[1], r.origin[2], widths(lims)[1], widths(r)[2]) end # restrict to x change - if rz.restrict_y || !ax.rectzoom_affect_y[] + if rz.restrict_y || !ax.yrectzoom[] r = FRect2D(r.origin[1], lims.origin[2], widths(r)[1], widths(lims)[2]) end return r From 1863b0dc1e3904d03df4c0d8e4ae32669cbdbab9 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Sun, 1 Nov 2020 14:35:56 +0100 Subject: [PATCH 29/30] bit more snappy toggle --- src/makielayout/defaultattributes.jl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index 61997f5c2..09b54982a 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -569,7 +569,7 @@ function default_attributes(::Type{LToggle}, scene) "Indicates if the toggle is active or not." active = false "The duration of the toggle animation." - toggleduration = 0.2 + toggleduration = 0.15 "The border width as a fraction of the toggle height " rimfraction = 0.33 "The align mode of the toggle in its parent GridLayout." From f1e89d3e46ae3b242d4f48879144c15d45151be7 Mon Sep 17 00:00:00 2001 From: Julius Krumbiegel Date: Mon, 2 Nov 2020 08:48:38 +0100 Subject: [PATCH 30/30] slider linewidth controls autosize, correct vertical --- src/makielayout/defaultattributes.jl | 6 ++++-- src/makielayout/lobjects/lslider.jl | 20 ++++++++++++-------- 2 files changed, 16 insertions(+), 10 deletions(-) diff --git a/src/makielayout/defaultattributes.jl b/src/makielayout/defaultattributes.jl index 09b54982a..5d55152d6 100644 --- a/src/makielayout/defaultattributes.jl +++ b/src/makielayout/defaultattributes.jl @@ -504,9 +504,9 @@ function default_attributes(::Type{LSlider}, scene) "The vertical alignment of the slider in its suggested bounding box." valign = :center "The width setting of the slider." - width = nothing + width = Auto() "The height setting of the slider." - height = 15 + height = Auto() "The range of values that the slider can pick from." range = 0:10 "Controls if the parent layout can adjust to this element's width" @@ -517,6 +517,8 @@ function default_attributes(::Type{LSlider}, scene) startvalue = 0 "The current value of the slider." value = 0 + "The width of the slider line" + linewidth = 15 "The color of the slider when the mouse hovers over it." color_active_dimmed = COLOR_ACCENT_DIMMED[] "The color of the slider when the mouse clicks and drags the slider." diff --git a/src/makielayout/lobjects/lslider.jl b/src/makielayout/lobjects/lslider.jl index 261013f02..b9809dba0 100644 --- a/src/makielayout/lobjects/lslider.jl +++ b/src/makielayout/lobjects/lslider.jl @@ -7,7 +7,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) decorations = Dict{Symbol, Any}() @extract attrs ( - halign, valign, horizontal, + halign, valign, horizontal, linewidth, startvalue, value, color_active, color_active_dimmed, color_inactive ) @@ -17,6 +17,14 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) layoutobservables = LayoutObservables{LSlider}(attrs.width, attrs.height, attrs.tellwidth, attrs.tellheight, halign, valign, attrs.alignmode; suggestedbbox = bbox, protrusions = protrusions) + onany(linewidth, horizontal) do lw, horizontal + if horizontal + layoutobservables.autosize[] = (nothing, Float32(lw)) + else + layoutobservables.autosize[] = (Float32(lw), nothing) + end + end + sliderbox = lift(identity, layoutobservables.computedbbox) endpoints = lift(sliderbox, horizontal) do bb, horizontal @@ -31,7 +39,7 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) else x = left(bb) + w / 2 [Point2f0(x, bottom(bb) + w/2), - Point2f0(x, top(bb) + h/2)] + Point2f0(x, top(bb) - w/2)] end end @@ -80,10 +88,6 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) [ca, ci] end - linewidth = lift(horizontal, sliderbox) do hori, sbox - hori ? height(sbox) : width(sbox) - end - endbuttons = scatter!(parent, endpoints, color = linecolors, markersize = linewidth, strokewidth = 0, raw = true)[end] decorations[:endbuttons] = endbuttons @@ -146,8 +150,8 @@ function LSlider(parent::Scene; bbox = nothing, kwargs...) linecolors[] = [color_active_dimmed[], color_inactive[]] end - # trigger bbox - layoutobservables.suggestedbbox[] = layoutobservables.suggestedbbox[] + # trigger autosize through linewidth for first layout + linewidth[] = linewidth[] LSlider(parent, layoutobservables, attrs, decorations) end