-
-
Notifications
You must be signed in to change notification settings - Fork 52
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
base: main
Are you sure you want to change the base?
Changes from 12 commits
8f62037
f1e0822
e1f5cd8
de8f54f
7ffc1ba
e908f6d
da0184c
1ac4713
c891118
27620ba
80957f1
6017adc
10c9686
68a452a
36bbcbd
5811f1a
2debc17
545268c
80ca14e
92f36ca
6e91da7
2c18b27
d6331a2
51dde2c
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
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...) | ||
end | ||
scale(scaleargs) | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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... There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 There was a problem hiding this comment. Choose a reason for hiding this commentThe 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... There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 ? There was a problem hiding this comment. Choose a reason for hiding this commentThe 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. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. You can find a BoundingBox for a lot of "shapes" using eg @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. | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. would have |
||
- `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) |
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -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, | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe 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 |
There was a problem hiding this comment.
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.
There was a problem hiding this comment.
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!)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
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
.There was a problem hiding this comment.
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 😂)
There was a problem hiding this comment.
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...There was a problem hiding this comment.
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 toMethod
.eg: If there's a variable
circle = Object(...)
, and user passes circle as an argument, it will throw theObjects of type Javis.Object are not callable
error and maybe use a try catch block to give out a better error message?There was a problem hiding this comment.
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.