Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

JImage Implementation for J-Objects #399

Open
wants to merge 24 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 12 commits
Commits
Show all changes
24 commits
Select commit Hold shift + click to select a range
8f62037
Exported the JImage function from the JImage shorthand
TheCedarPrince Aug 14, 2021
f1e0822
Possible implementation of adding an image easily to JObjects
TheCedarPrince Aug 14, 2021
e1f5cd8
Made argument names less confusing
TheCedarPrince Aug 14, 2021
de8f54f
Merge branch 'master' into tcp-image-jobjects
TheCedarPrince Sep 28, 2021
7ffc1ba
Added scaling capabilities
TheCedarPrince Sep 28, 2021
e908f6d
Added docstring, changed default for shape
TheCedarPrince Oct 5, 2021
da0184c
In process of adding tests for JImage
TheCedarPrince Oct 5, 2021
1ac4713
Merge branch 'master' into tcp-image-jobjects
TheCedarPrince Dec 29, 2021
c891118
Added type signature from Cairo
TheCedarPrince Dec 30, 2021
27620ba
Reference image for JImage
TheCedarPrince Dec 30, 2021
80957f1
Fixed docstrings
TheCedarPrince Dec 30, 2021
6017adc
Revised using dispatch image
TheCedarPrince Dec 30, 2021
10c9686
Merge branch 'master' into tcp-image-jobjects
TheCedarPrince Jan 5, 2022
68a452a
Added TODO for scaling logic
TheCedarPrince Jan 5, 2022
36bbcbd
Temporary disabling of tests
TheCedarPrince Jan 5, 2022
5811f1a
Added TODO for tests, fixed test cases, and fixed references
TheCedarPrince Jan 5, 2022
2debc17
Added individual tests; working on scaling
TheCedarPrince Jan 10, 2022
545268c
Begin adding autoscaling support for JImage
TheCedarPrince Jan 10, 2022
80ca14e
Added image support dispatch for JBox
TheCedarPrince Jan 16, 2022
92f36ca
Added image support dispatch for JPoly
TheCedarPrince Jan 16, 2022
6e91da7
Added image support dispatch for JStar
TheCedarPrince Jan 16, 2022
2c18b27
Added image support dispatch for JRect
TheCedarPrince Jan 17, 2022
d6331a2
Added image support dispatch for JCircle
TheCedarPrince Jan 17, 2022
51dde2c
Added image support dispatch for JEllipse
TheCedarPrince Jan 17, 2022
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 3 additions & 2 deletions src/Javis.jl
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
module Javis

using Animations
import Cairo: CairoImageSurface, image
import Cairo: CairoImageSurface, CairoSurfaceBase, image
using FFMPEG
using Gtk
using GtkReactive
Expand Down Expand Up @@ -724,6 +724,7 @@ include("shorthands/JEllipse.jl")
include("shorthands/JStar.jl")
include("shorthands/JPoly.jl")
include("shorthands/JShape.jl")
include("shorthands/JImage.jl")

export render, latex
export Video, Object, Background, Action, RFrames, GFrames
Expand All @@ -738,7 +739,7 @@ export act!
export anim_translate, anim_rotate, anim_rotate_around, anim_scale

export @Frames, prev_start, prev_end, startof, endof
export JBox, JCircle, JEllipse, JLine, JPoly, JRect, JStar, @JShape
export JBox, JCircle, JEllipse, JImage, JLine, JPoly, JRect, JStar, @JShape

# custom override of luxor extensions
export setline, setopacity, fontsize, get_fontsize, scale, text
Expand Down
66 changes: 66 additions & 0 deletions src/shorthands/JImage.jl
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
function _JImage(pos, img, centering, shapeargs, shape, scaleargs)
if !isnothing(shape)
shape(shapeargs...)
Copy link
Member

@Sov-trotter Sov-trotter Aug 15, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

One minor thing:
I think we should mention(in the docstring) that shape should be a valid Luxor shape(a luxor function).
eg: rect, box, poly etc.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"valid Luxor shape" I'm not aware of this concept. :) Do you mean a built-in Luxor function - eg one that constructs paths, or one that generates points for polygons...? (Not trying to be awkward, just I get worried in case I'm not providing the tools you folks need for your cool constructions!)

Copy link
Member

@Sov-trotter Sov-trotter Aug 15, 2021

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Do you mean a built-in Luxor function - eg one that constructs paths

Yeah! We need shape to be a valid luxor function. Since a user may interpret it as entering a full shape name eg: rectangle instead of rect.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We mean any kind of function either their own or more likely a Luxor drawing function. Mostly those which provide the clipping action (so yeah a drawing function 😂)

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

OK, yes. Perhaps any function that has an action parameter/keyword...

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Another thing I want to mention is that shape should by typed to Method.
eg: If there's a variable circle = Object(...), and user passes circle as an argument, it will throw the Objects of type Javis.Object are not callable error and maybe use a try catch block to give out a better error message?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yea - error handling is something we need to reckon with @Sov-trotter . I agree with your thoughts and will see what I can do.

end
scale(scaleargs)
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would really like to have automatic scaling by checking the size of the shape and that of the image.
For the shape it would be awesome to be able to call it with :clip / :path automatically but it seems action isn't always a keyword argument which makes that a bit hard. Any idea on that @cormullion ?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

yes, following on from my question (#399 (comment)) not all graphic construction functions are similar in having an action keyword. We'd have to compile a list and see which ones could be updated...

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Are you interested in standardizing the functions though such that all have an option which provide using the action as a kwarg?

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Perhaps we can make a list of possible functions as an issue at Luxor.jl. It might be easy or not, not too sure how many are affected...

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

since this spawned off another issue, shall we resolve this comment @Wikunia and @cormullion ?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I decided to close this conversation as I do not think automatic scaling is feasible as all shapes can define a bounding box differently within Luxor.
To find that bounding box for a shape depends on the arguments provided by such a shape.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

You can find a BoundingBox for a lot of "shapes" using eg pathtopoly():

@draw begin
    circle(O, 100, :stroke)
    circle(O, 100, :path)
    bbox = BoundingBox(pathtopoly()[1])
    box(bbox, :stroke)
end

If you can convert a shape to a polygon, then you can get a bounding box that way. Are there any other shapes in Luxor that you'd like BoundingBox to work on?

placeimage(img, pos, centered = centering)
TheCedarPrince marked this conversation as resolved.
Show resolved Hide resolved
return pos
end

"""
JImage(pos::Point, img::CairoSurfaceBase{UInt32}, centering = true; shapeargs = (), shape = nothing, scaleargs = 1)

Place a given image at a given location as a `Javis` object.
Images can be cropped to different shapes and scaled to different sizes while being placed.
TheCedarPrince marked this conversation as resolved.
Show resolved Hide resolved

# Arguments
- `pos::Point`: Where to place the image inside a shape.
- `img::CairoSurfaceBase{UInt32}`: Expects a CairoSurfaceBase object via `readpng("your_image.png")`
- `centering::Bool`: Centers the object at `pos`
- `shapeargs`: Arguments to be passed to a given shape type
- `shape`: A Luxor shape function such as `circle`, `box`, etc.
- `scaleargs`: The arguments used for scaling the image used on the shape

# Return

Returns the position of the image location, `pos`.
"""
JImage(pos::Point, img::CairoSurfaceBase{UInt32}, centering::Bool = true; shapeargs = (), shape = nothing, scaleargs = 1) =
(
args...;
pos = pos,
img = img,
centering = centering,
shapeargs = shapeargs,
shape = shape,
scaleargs = scaleargs,
) -> _JImage(pos, img, centering, shapeargs, shape, scaleargs)

"""
JImage(pos::Point, img::CairoSurfaceBase{UInt32}, centering = true; shapeargs = (), shape = nothing, scaleargs = 1)

Place a given image at a given location as a `Javis` object.
Images can be cropped to different shapes and scaled to different sizes while being placed.

# Arguments
- `pos::Point`: Where to place the image inside a shape
- `img::String`: Expects the path to an image
- `centering::Bool`: Centers the object at `pos`
- `shapeargs`: Arguments to be passed to a given shape type
- `shape`: A Luxor shape function such as `circle`, `box`, etc.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

would have shape before shapeargs and maybe add an example which shows what shapeargs and scaleargs do?

- `scaleargs`: The arguments used for scaling the image used on the shape

# Return

Returns the position of the image location, `pos`.
"""
JImage(pos::Point, img::String, centering::Bool = true; shapeargs = (), shape = nothing, scaleargs = 1) =
(
args...;
pos = pos,
img = readpng(img),
centering = centering,
shapeargs = shapeargs,
shape = shape,
scaleargs = scaleargs,
) -> _JImage(pos, img, centering, shapeargs, shape, scaleargs)
Binary file added test/refs/dispatch.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
92 changes: 92 additions & 0 deletions test/shorthands.jl
Original file line number Diff line number Diff line change
Expand Up @@ -147,3 +147,95 @@ video = Video(800, 800)
rm("images/palette.png")
rm("shorthands.gif")
end

video = Video(800, 800)
@testset "Image Placement" begin
function ground(args...)
background("white")
sethue("black")
end

Background(1:90, ground)

circle_img = Object(
1:5,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = circle,
shapeargs = nothing,
scaleargs = 1,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe add another test case with different shapeargs and different scale args.

),
)
poly_img = Object(
1:10,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = poly,
shapeargs = nothing,
scaleargs = 1,
),
)
rect_img = Object(
1:15,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = rect,
shapeargs = nothing,
scaleargs = 1,
),
)
star_img = Object(
1:20,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = star,
shapeargs = nothing,
scaleargs = 1,
),
)
ellipse_img = Object(
1:25,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = ellipse,
shapeargs = nothing,
scaleargs = 1,
),
)
box_img = Object(
1:30,
JImage(
O,
readpng("refs/dispatch.png"),
true;
shape = box,
shapeargs = nothing,
scaleargs = 1,
),
)

render(video; tempdirectory = "images", pathname = "shorthands.gif")

for i in ["01", 20, 21, 39, 40, 59, 65, 85]
@test_reference "refs/shorthands$i.png" load("images/00000000$i.png")
@test isfile("shorthands.gif")
end

for i in 1:90
rm("images/$(lpad(i, 10, "0")).png")
end

rm("images/palette.png")
rm("shorthands.gif")

end