Attract-Mode Support Forum
Attract-Mode Support => Scripting => Topic started by: Bgoulette on November 12, 2015, 08:55:54 PM
-
Hi all,
I'm trying to get an image added to a surface to fill the surface but maintain aspect ratio. What I mean is I have the preserve_aspect_ratio flag set to true, but that keeps the asset "inside" the container, meaning you'll get letter- or pillar-boxing if the image isn't exactly the size of the container. What I want to do is treat the smallest dimension of the image to be loaded as the maximum size of the corresponding container dimension and scale the other dimension accordingly, meaning that in such cases, parts of the image would be "clipped."
An example: if I have a container (a surface we'll say) that's 100x100 and I have a snap that's 100x50 (for easy math!), I'd want AM to be able to see it as 200x100, thus ensuring that the container is filled even if it means I'm missing the "sides" of my image.
I tried an unholy combination of width, height, and subimg properties and positioning, and while it worked most of the time (or appeared to), on rare occasions, something would cause it to flip out. Here's what I have (that doesn't completely work):
function sizeSnap (str_sig)
{
/**/
// Determine snap width/height (subimg_) and scale accordingly:
// Take the smaller value and scale to SNAP_SIZE
local r=1.0; // ratio
local dims={w=snap.subimg_width, h=snap.subimg_height};
// If we haven't already, let's adjust the snap size:
if (dims.w > dims.h) {
// Wider
r=(dims.w+0.0)/dims.h;
snap.height=SNAP_SIZE;
snap.width=SNAP_SIZE*r.tofloat();
}
else if (dims.w < dims.h) {
// Taller
r=(dims.h+0.0/dims.w);
snap.width=SNAP_SIZE;
snap.height=SNAP_SIZE*r.tofloat();
}
else {
// Equal
snap.width=SNAP_SIZE;
snap.height=SNAP_SIZE;
}
// Position in frame:
snap.x=(SNAP_SIZE-snap.width)/2.0;
snap.y=(SNAP_SIZE-snap.height)/2.0;
/**/
}
(This assumes the constant SNAP_SIZE is defined and "snap" is an artwork image added earlier in the code.)
It's mostly there, just on occasion, moving up or down the list will cause a freak out that, typically, results in an image appearing as a bunch of vertical bars (even on images that are wider than tall). Not sure where that's happening...
Any thoughts? Or is there a simpler way to do this?! Thanks!
-
I am actually working on exactly this, trying to make a custom object to handle the adjustment. Just to clarify, what you are referring to would be a FILL ( cropped and centered ) in the specified dimensions?
(http://i.imgur.com/UnBOENs.jpg)
A few different concerns/things to consider:
If the image is on its own surface that is already adjusted to the size you want, you could enable preserve_aspect_ratio (allow it to worry about the aspect) and then just scale and center to fill the surface.
If the image will not fill the parent surface (such as if fe is the parent surface ), you could modify subimg values. In this case though, you'd probably want to disable preserve_aspect_ratio and handle that yourself. You would also need to use a transition callback to make sure to get the correct texture_width/texture_height values of the image, then do the subimg x/y/w/h adjustments.
Ideally in my custom object, I want options to say how to align the fill - like FILL ( centered X/Y ), FILL_TOP ( centered X ), FILL_LEFT ( centered on Y ), FILL_BOTTOM ( centered on X ).
My main concern in doing this is that regardless, some images don't look good when cropped/filled. What images are you planning to use this with?
I'll be doing some work on AM stuff today, so I'll try to get mine working and add some code on this thread.
-
Thanks for the response!
Yeah, I'm trying to "fill" in the sense you described above. Like how Windows' desktop wallpaper can be set to fill which crops either the top or bottom depending on the image's dimensions, as you show in your Metroid box art example.
I had tried creating a surface with the dimensions I wanted (let's say 720x720), but I was trying to adjust that object's size, rather than creating an intermediary. I think I'll give that a shot and see what happens.
I had tried manipulating the subimg_ properties, but once adjusted, it seems like sometimes the next image would maintain the (wrong) subimg_ values, though that might be a completely wrong assumption!
You're right: some images look like crap when cropped (filled), but I'm setting up a square layout (centered on the actual screen size) and has a square area for snaps (mostly title screens, though that could change). It's more for visual interest than anything more meaningful (in my case), but having the ability would be helpful, I think, so I'm definitely interested in seeing your custom object!
Here's the file I'm using as my background. The snap would sit inside the larger transparent area:
(http://i.imgur.com/XFbFuyGm.png) (http://imgur.com/XFbFuyG)
(though the transparent area appears dark gray here!)
Here's a screengrab of the layout "in action":
(http://i.imgur.com/Ii4PaWJm.png) (http://imgur.com/Ii4PaWJ)
I need to add another image to sit behind everything to hide the pillar-boxing effect. The goal is to ultimately put this in a cab that uses a square portion of a vertically-oriented 42" lcd to get the best of both vertical and horizontal games without having to actually rotate a monitor, but just in case I'm trying to set it up to look decent on a "standard" monitor, too. Sort of. I can't imagine anyone else would want to use this particular layout! ;)
-
Okay, here's a revisited attempt, placing the code in the transition callback, but it seems to be "one behind," if that makes any sense. Meaning, let's say I'm seeing DoDonPachi on the screen; the print statement for the rom name shows me Dimahoo (the previous entry in my list), etc. Am I missing something, or executing this branch at the wrong time? Just trying to wrap my head around when the callback should actually execute, or if I'm using the wrong ttype here:
function doScreenUpdates (ttype, var, ttime)
{
switch (ttype) {
/* other ttypes handled here */
case Transition.ToNewSelection:
local r=1.0; // ratio
local dims={w=snapContainer.subimg_width, h=snapContainer.subimg_height};
print ("rom: "+fe.game_info(Info.Name, 0)+"\n");
print ("dims: "+dims.w+", "+dims.h+"\n");
if (dims.w > dims.h) {
// Wider
r=dims.w.tofloat()/dims.h.tofloat();
snapContainer.height=SNAP_SIZE*r.tofloat();
}
else if (dims.h > dims.w) {
// Taller
r=dims.h.tofloat()/dims.w.tofloat();
snapContainer.width=SNAP_SIZE*r.tofloat();
}
else {
// Equal
snapContainer.width=SNAP_SIZE;
}
// Center snapContainer:
snapContainer.x=(SNAP_SIZE-snapContainer.width)/2.0;
snapContainer.y=(SNAP_SIZE-snapContainer.height)/2.0;
print ("snapContainer coords: "+snapContainer.x+", "+snapContainer.y+"\n-----\n");
break;
}
}
Thanks for any insight you're able to provide!
Actually, I noticed if I replace the "0" index in the fe.game_info() call with "var," it returns the proper rom name; however, the snapContainer object still holds a reference (image) to the prior selection. Apparently. Is there a way to "update" which snap we're dealing with through var, or am I (again!) thinking about things the wrong way? Again, thanks!
-
ideally you probably want to do what I'm doing - create your own object with its own transition, like:
class FillArt
{
ref = null;
constructor( art, x, y, w, h, surface = ::fe )
{
ref = surface.add_artwork( x, y, w, h );
fe.add_transition_callback( this, "on_transition" );
}
function on_transition( ttype, var, ttime )
{
switch( ttype )
{
case Transition.ToNewSelection:
case Transition.ToNewList:
case Transtion.StartLayout:
//local texture_width = ref.texture_width;
//local texture_height = ref.texture_height;
//do scale/centering to ref
//ref.subimg_width = x;
//ref.subimg_height = y;
local name = fe.game_info(Info.Name, var + ref.index_offset);
break;
}
return false;
}
}
FillArt("snap", 0, 0, 320, 240);
Just a quick response but hopefully this gets you in the right direction. ref here will be the reference to your actual artwork object.
Don't worry about your incorrect index info unless you are modifying something based on it, since you are just modifying the image properties. But I think it would be something like ( var + obj.index_offset ) as shown here. I tend to get confused since the var changes depending on which transition is occurring :)
-
I'm about to demonstrate my lack of knowledge, but hey... :)
I looked through that example, and I can understand what's going on (I think), except for how you'd call/integrate an instance of that new class in the overall layout. (This might be a case where I display my confusion with Squirrel, too.)
Where I used something like this:
local snapcont=fe.add_artwork("snap", 0, 0, 720, 720);
would I now use something like:
local snapcont=FillArt("snap", 0, 0, 720, 720);
?
Wait, that can't be right, because I haven't specified what "ref" is (or is it understood, or is it "surface" passed directly in the constructor?). Or would there be a different way to instantiate it?
Wait (part 2: electric boogaloo)! I'd instantiate it on, say, snapCanvas like this, right?
local snapcont=FillArt("snap", 0, 0, 720, 720, snapCanvas)
?
Except in the constructor, you've got ref=surface.add_artwork(x, y, w, h) but no reference to the art itself?
I apologize for putting my ignorance on display, but I keep thinking about how I'd do things in ActionScript, and, well, Flash is pretty much dead (and I don't want to write yet another fe from the ground up anyway! Wouldn't even know where to begin!). Thanks for your patience!
-
No problem.. when you create the "instance" of the FillArt class, it will assign the artwork instance to ref:
local snapcont = FillArt("snap", 0, 0, 0, 0);
That gives you an instance of the FillArt class. The constructor is run with the parameters you passed above, and the constructor is what will set ref to an instance of the artwork returned by the add_artwork function.
The caveat of doing it like this is you can't use snapcont.width/.height/etc for your artwork properties... but you can access 'ref' using snapcont.ref.width/.height/etc
Make sense?
-
I think I understand... here's where I am:
// Some other classes:
class FillArt
{
art=null;
constructor(ref, x, y, w, h, surface=::fe)
{
art=surface.add_artwork(ref, x, y, w, h);
art.preserve_aspect_ratio=true;
fe.add_transition_callback("onTransition");
}
function onTransition (ttype, var, ttime)
{
print ("onTransition\n");
switch (ttype)
{
case Transition.ToNewSelection:
case Transition.ToNewList:
case Transition.StartLayout:
local r=1.0; // Ratio
local dims={w=art.texture_width, h=art.texture_height};
if (dims.w > dims.h)
{
// Wider
r=dims.w.tofloat()/dims.h.tofloat();
art.width=::SNAP_SIZE*r;
art.height=::SNAP_SIZE;
}
else if (dims.w < dims.h)
{
// Taller
r=dims.h.tofloat()/dims.w.tofloat();
art.height=::SNAP_SIZE*r;
art.width=::SNAP_SIZE;
}
else
{
// Must be equal!
art.width=art.height=::SNAP_SIZE;
}
return false;
break;
}
}
};
That appears after my constant declarations (e.g., SNAP_SIZE). Later on, I have:
// Start with the snap: it needs to go beneath everything:
local snapCanvasCoords={x=72, y=58};
local snapCanvas=canvas.add_surface(SNAP_SIZE, SNAP_SIZE); // "canvas" is a surface created previously and added as fe.add_surface()
snapCanvas.x=snapCanvasCoords.x;
snapCanvas.y=snapCanvasCoords.y;
local snapContainer=FillArt("snap", 0, 0, SNAP_SIZE, SNAP_SIZE, snapCanvas);
The image (the snap) shows up where it should, but it appears as though the FillArt "onTransition" handler never fires. I tossed in a print statement just to see what was going on, but it never seems to output anything. Above, I haven't bothered attempting to position the art: just trying to ensure it scales appropriately, but it appears it doesn't scale at all... :(
What am I missing?! Thanks!
-
you took out the 'this' :)
It should be:
fe.add_transition_callback( this, "onTransition");
'this' tells it to look in the class that the 'onTransition' function is in
-
Gagh! Thanks! :)
-
one other thing you'll want to correct - put the break after your code under the 'case' options in the switch, and THEN return false. Otherwise, if you try to add other ttype cases, it will run through all your code until the break is reached, which would be never it.. like so:
function transition( ttype, var, ttime )
{
switch ( ttype )
{
case 1:
case 2:
//code for case 1 + 2
break;
case 3:
//code for case 3
break;
}
return false;
}
-
Success! I think! :) The below snippet seems to work as intended:
// Some other classes:
class FillArt
{
art=null;
constructor(ref, x, y, w, h, surface=::fe)
{
art=surface.add_artwork(ref, x, y, w, h);
art.preserve_aspect_ratio=true;
fe.add_transition_callback(this, "onTransition");
}
function onTransition (ttype, var, ttime)
{
switch (ttype)
{
case Transition.FromOldSelection:
case Transition.ToNewSelection:
case Transition.ToNewList:
case Transition.StartLayout:
local r=1.0; // Ratio
local dims={w=art.texture_width, h=art.texture_height};
if (dims.w > dims.h)
{
// Wider
r=dims.w.tofloat()/dims.h.tofloat();
art.width=SNAP_SIZE*r;
art.height=SNAP_SIZE;
}
else if (dims.w < dims.h)
{
// Taller
r=dims.h.tofloat()/dims.w.tofloat();
art.height=SNAP_SIZE*r;
art.width=SNAP_SIZE;
}
else
{
// Must be equal!
art.width=art.height=SNAP_SIZE;
}
art.x=(SNAP_SIZE-art.width)/2.0;
art.y=(SNAP_SIZE-art.height)/2.0;
// print ("r: "+r+"\ndims: "+dims.w+", "+dims.h+"\nw/h: "+art.width+", "+art.height+"\n");
break;
}
return false;
}
};
SNAP_SIZE is defined prior to instantiating the class, but having added the Transition.FromOldSelection to the list of ttypes to handle, things seem to be sizing in a correct enough way as of now! Woo!
(I reserve the right to attenuate my enthusiasm if I discover I'm wrong! ;) )
[Also, corrected "break" error as noted above!]
-
Great to hear! I didn't get to do much of anything today, so I'll try to check it out tomorrow. We can do some optimizations and add a few features to the class, then convert it to a module :)
-
Great to hear! I didn't get to do much of anything today, so I'll try to check it out tomorrow. We can do some optimizations and add a few features to the class, then convert it to a module :)
Sounds good! Would love to be able to contribute in some small way!
Here's a slightly revised version of the class. Now it's not so tightly coupled to my particular implementation: I added _w and _h members to the class for assignment from the original instantiation call (versus using my SNAP_SIZE constant).
I also added a method for choosing an alignment option. It worked when called from within the class, but that's not how it should be used: it needs to be accessible from outside the class so you could call it from the primary guts of the layout, but I'm not sure how to do that in Squirrel just yet. I'll keep poking around. You'll see I'm referencing some static strings that don't actually exist: wasn't sure where to declare those (something akin to Transition.* is what I was trying to do: see below for the intended usage).
Anyway, here's the revised class with a lot of non-functional stuff in it (commented out):
[redacted]
And here's an example use case:
local myFilledArt=FillArt("snap", 0, 0, 720, 720, container); // container is a previously created surface or other image object
myFilledArt.set_anchor(FillArt.TopLeft);
But again, currently, that second line doesn't work...
Edit: I created an enum object to hold the anchor point strings and am in fact using that to set the initial position of the FillArt instance:
enum Anchor
{
TopLeft="TopLeft"
TopCentre="TopCentre"
TopRight="TopRight"
CentreLeft="CentreLeft"
Centre="Centre"
CentreRight="CentreRight"
BottomLeft="BottomLeft"
BottomCentre="BottomCentre"
BottomRight="BottomRight"
}
class FillArt
{
art=null;
_w=0;
_h=0;
constructor(ref, x, y, w, h, surface=::fe)
{
art=surface.add_artwork(ref, x, y, w, h);
art.preserve_aspect_ratio=true;
// Set internal properties for width/height:
_w=w;
_h=h;
fe.add_transition_callback(this, "onTransition");
}
function onTransition (ttype, var, ttime)
{
switch (ttype)
{
case Transition.FromOldSelection:
case Transition.ToNewSelection:
case Transition.ToNewList:
case Transition.StartLayout:
local r=1.0; // Ratio
local dims={w=art.texture_width, h=art.texture_height};
if (dims.w > dims.h)
{
// Wider
r=dims.w.tofloat()/dims.h.tofloat();
art.width=_w*r;
art.height=_h;
}
else if (dims.w < dims.h)
{
// Taller
r=dims.h.tofloat()/dims.w.tofloat();
art.height=_h*r;
art.width=_w;
}
else
{
// Equal
art.width=art.height=_w;
}
// Image is centered by default:
set_anchor(Anchor.Centre);
break;
}
return false;
}
/**/
// Function works when called from within FillArt, but not sure how to
// make it accessible from outside (incl. static members)...
function set_anchor (str_anchor)
{
::print ("anchor: "+str_anchor+"\n");
switch (str_anchor)
{
case Anchor.TopLeft:
art.x=0;
art.y=0;
break;
case Anchor.TopCentre:
art.x=(_w-art.width)/2.0;
art.y=0;
break;
case Anchor.TopRight:
art.x=_w-art.width;
art.y=0;
break;
case Anchor.CentreLeft:
art.x=0;
art.y=(_h-art.height)/2.0;
break;
case Anchor.CentreRight:
art.x=_w-art.width;
art.y=(_h-art.height)/2.0;
break;
case Anchor.BottomLeft:
art.x=0;
art.y=_h-art.height;
break;
case Anchor.BottomCentre:
art.x=(_w-art.width)/2.0;
art.y=_h-art.height;
break;
case Anchor.BottomRight:
art.x=_w-art.width;
art.y=_h-art.height;
break;
case Anchor.Centre:
default:
art.x=(_w-art.width)/2.0;
art.y=(_h-art.height)/2.0;
break;
}
return;
}
/**/
};
However, it still doesn't work when attempting to set the anchor position externally :(
-
alright got to check it out for a bit. I'm working on some updates, but still a bit away.
For learning purposes - functions within a class can work as either a static function - FillArt.set_anchor() or as a function of the class instance - my_art.set_anchor(). Since you will have multiple FillArt instances and you (might?) want each to have its own anchor setting - you want the latter. Within the class itself, you can just call set_anchor() like you did in the transition function and it will be setting the anchor for that instance.
Some changes I'm looking at:
- Store the anchor position in a variable. Then you can do a get ( local current = my_fillart.anchor; ) as well as pass the current setting to the set_anchor function.
- You had the anchor being reset to Centre each time the transitions occurs. Instead, you can now pass the current anchor setting to the set_anchor function.
- Instead of enum (which is apparently local to the script), I switched it to a table stored in the root table (::). Now we can make this a module, and Anchor will be available to the layout script.
- I gave the fill art its own surface (_parent) which the art is drawn on. Now you can properly move the object and the art fills the surface according to the anchor.
- I added aliases to the Anchor table for 'center' - because, well :)
I'll try to finish it up tonight, not sure if I'll get the time to yet or not.
-
Looking forward to seeing what you come up with! Just for kicks, I tried going back and implementing the changes you describe. With spectacular failures all around! :( Seeing what you've modified I'm hoping will also help me get a better grasp on how you do things in Squirrel.
And the only reason I spelled center "the wrong way" ;) was because somewhere, I think for aligning text, the centering version is Align.Centre, which trips me up every time! I was trying to stay consistent, but yeah, I much prefer spelling it "the right way!" ;) hehe
-
okay, here's what I have - probably still some outstanding issues:
http://imgur.com/a/MQbUQ
As I mentioned, here's some changes which you'll notice:
* this code can now be put in modules/fill-art.nut and used with fe.load_module("fill-art");
* modified anchor to squirrel root table, and modified values to be less confusing
* each FillArt object has its own surface (parent) - the art objects x/y/w/h is adjusted based on aspect of parent and texture
* moved aspect/anchor code to update function ( because we will need to update at other times, apart from the transition which is not yet implemented )
* added scale property - 1.0 will fit to parent with cropping, > 1.0 will scale texture up, < 1.0 would not fill parent
* added _get/_set functions - this will redirect properties to the parent surface so you can use things like my_fillart.x/.y/.width/.height/.alpha/etc.. ( this still needs some modifications )
* added a neat scale animation ( my_fill_art.animate = true ) with some options (described below)
/////////////////////////////////////////////////////////
//
// Attract-Mode Frontend - FillArt Module
//
/////////////////////////////////////////////////////////
//
// FillArt - creates an artwork that will fill the specified dimensions,
// cropping the art as necessary - but keeping aspect ratio.
//
// Example:
// local my_art = FillArt( 50, 50, 400, 200 );
// my_art.anchor = Anchor.Bottom; //set where texture is anchored to ( if necessary based on object size / texture size / scale )
// my_art.scale = 1.0; //set texture scale
// my_art.animate = true; //whether to use a scale animation
// my_art.animate_to = 1.5; //scale to animate to
// my_art.animate_ms = 10000; //animation time in ms
//
::Anchor <-
{
TopLeft = "TopLeft",
Top = "Top",
TopRight = "TopRight",
Left = "Left",
Centre = "Center",
Center = "Center",
Right = "Right",
BottomLeft = "BottomLeft",
Bottom = "Bottom",
BottomRight = "BottomRight"
}
class FillArt
{
VERSION = 1.0; //first version
art = null; // the art object
anchor = null; // what side of the surface to anchor to
scale = 1.0; // 1.0 will fill parent - cropping edges
// > 1.0 will fill parent and scale larger
// < 1.0 will not fill but still center / account for anchor
animate = false; // whether to do a scale animation on transitions
animate_ms = 5000; // length of animate in ms
animate_to = 1.5; //scale to animate to
_animate_start = 0; //time animation started
_animate_running = false; //true if animation is currently running
_animate_scale = 1.0; //current animation scale
_parent = null; // the parent surface we will fill
_texture_width = 0; // store art texture width, once we get it
_texture_height = 0; // store art texture height, once we get it
constructor(ref, x, y, w, h, surface=::fe)
{
//defaults
anchor = ::Anchor.Center;
//create the parent surface - which we will fill with the art
_parent = surface.add_surface( w, h );
_parent.x = x;
_parent.y = y;
//debug purpose - helps us see the parent surface size
//_parent.add_image(::fe.script_dir + "pixel.png", 0, 0, w, h);
//add the artwork to the parent
art = _parent.add_artwork(ref, x, y, 0, 0);
//add callbacks
::fe.add_transition_callback( this, "onTransition" );
::fe.add_ticks_callback( this, "onTick" );
}
//redirect undefined properties to the parent surface object
function _get( idx ) { return _parent[idx]; }
function _set( idx, val ) {
_parent[idx] = val;
return val;
}
function onTransition (ttype, var, ttime)
{
if ( ttype == Transition.FromOldSelection || ttype == Transition.ToNewList )
{
_texture_width = art.texture_width;
_texture_height = art.texture_height;
update();
}
//do animation
if ( animate && ( ttype == Transition.ToNewSelection || ttype == Transition.StartLayout || ttype == Transition.ToNewList ) )
{
_animate_running = true;
_animate_start = get_time();
}
return false;
}
function get_time() { return ::clock() * 1000; }
function do_animation()
{
if ( _animate_running )
{
local elapsed = get_time() - _animate_start;
local animate_progress = ( elapsed / animate_ms.tofloat() );
_animate_scale = scale + ( animate_to - scale ) * animate_progress;
//::print( "animate progress: " + animate_progress + " : " + _animate_scale + "\n" );
//end animate
if ( animate_progress >= 1.0 ) _animate_running = false;
update();
}
}
function onTick( ttime )
{
if ( animate && _animate_running ) do_animation();
}
function update()
{
local current_scale = ( animate ) ? _animate_scale : scale;
local parent_aspect = _parent.width.tofloat() / _parent.height.tofloat();
local texture_aspect = _texture_width.tofloat() / _texture_height.tofloat();
if ( parent_aspect >= 1.0 )
{
//fill to parent width
if ( texture_aspect >= 1.0 )
{
//wide texture
//::print("wide parent, wide tex\n");
art.width = ( _texture_width * ( _texture_width / _parent.width.tofloat() ) ) * current_scale;
} else
{
//tall texture
//::print("wide parent, tall tex\n");
art.width = ( _texture_width * ( _parent.width / _texture_width.tofloat() ) ) * current_scale;
}
art.height = art.width / texture_aspect;
} else
{
//fill to parent height
if ( texture_aspect >= 1.0 )
{
//::print("tall parent, wide tex\n");
//wide texture
art.height = ( _texture_width * ( _parent.height / _texture_height.tofloat() ) ) * current_scale;
} else
{
//tall texture
//::print("tall parent, tall tex\n");
art.height = ( _texture_width * ( _parent.height / _texture_width.tofloat() ) ) * current_scale;
}
art.width = art.height * texture_aspect;
}
switch ( anchor )
{
case ::Anchor.TopLeft:
art.x = 0;
art.y = 0;
break;
case ::Anchor.Top:
art.x = ( _parent.width - art.width ) / 2.0;
art.y = 0;
break;
case ::Anchor.TopRight:
art.x = _parent.width - art.width;
art.y = 0;
break;
case ::Anchor.Left:
art.x = 0;
art.y = ( _parent.height-art.height ) / 2.0;
break;
case ::Anchor.Right:
art.x = _parent.width - art.width;
art.y = ( _parent.height - art.height ) / 2.0;
break;
case ::Anchor.BottomLeft:
art.x = 0;
art.y = _parent.height - art.height;
break;
case ::Anchor.Bottom:
art.x= ( _parent.width - art.width ) / 2.0;
art.y= _parent.height - art.height;
break;
case ::Anchor.BottomRight:
art.x = _parent.width - art.width;
art.y = _parent.height - art.height;
break;
case ::Anchor.Center:
default:
art.x = ( _parent.width-art.width ) / 2.0;
art.y = ( _parent.height-art.height ) / 2.0;
break;
}
//::print("anchor: " + anchor + " texture size: " + " - " + current_scale + "\n" );
}
}
-
This is great! Does it work with videos too?
-
Wow, that's a lot of changes! :)
I did notice that, for some reason, certain snaps don't resize appropriately. Not sure why that might be, unless it's a math thing under the hood for those elements.
I revised my code (stealing liberally from parts of yours) for a "no-frills" version that works as it should (apparently) as a module, but without the animation or convenience features you included. I'll revisit it later, perhaps, though there's probably no need!
Edit: code! Save as fill-art-bg.nut in the modules folder, load with fe.load_module("fill-art-bg"):
::Anchor <-
{
TopLeft=7,
Top=8,
TopRight=9,
Left=4,
Center=5,
Right=6,
BottomLeft=1,
Bottom=2,
BottomRight=3
};
class FillArt
{
anchor=null;
art=null;
_w=0;
_h=0;
constructor(ref, x, y, w, h, surface=::fe)
{
anchor=::Anchor.Center;
art=surface.add_artwork(ref, x, y, w, h);
art.preserve_aspect_ratio=true;
// Set internal properties for width/height:
_w=w;
_h=h;
::fe.add_transition_callback(this, "onTransition");
}
function onTransition (ttype, var, ttime)
{
switch (ttype)
{
case Transition.FromOldSelection:
case Transition.ToNewSelection:
case Transition.ToNewList:
case Transition.StartLayout:
local r=1.0; // Ratio
local dims={"w":art.texture_width, "h":art.texture_height};
if (dims.w > dims.h)
{
// Wider
r=dims.w.tofloat()/dims.h.tofloat();
art.width=_w*r;
art.height=_h;
}
else if (dims.w < dims.h)
{
// Taller
r=dims.h.tofloat()/dims.w.tofloat();
art.height=_h*r;
art.width=_w;
}
else
{
// Equal
art.width=art.height=_w;
}
// Image is centered by default:
this.set_anchor(anchor);
break;
}
return false;
}
function set_anchor (int_anchor)
{
switch (int_anchor)
{
case ::Anchor.TopLeft:
art.x=0;
art.y=0;
break;
case ::Anchor.Top:
art.x=(_w-art.width)/2.0;
art.y=0;
break;
case ::Anchor.TopRight:
art.x=_w-art.width;
art.y=0;
break;
case ::Anchor.Left:
art.x=0;
art.y=(_h-art.height)/2.0;
break;
case ::Anchor.Right:
art.x=_w-art.width;
art.y=(_h-art.height)/2.0;
break;
case ::Anchor.BottomLeft:
art.x=0;
art.y=_h-art.height;
break;
case ::Anchor.Bottom:
art.x=(_w-art.width)/2.0;
art.y=_h-art.height;
break;
case ::Anchor.BottomRight:
art.x=_w-art.width;
art.y=_h-art.height;
break;
case ::Anchor.Center:
default:
art.x=(_w-art.width)/2.0;
art.y=(_h-art.height)/2.0;
break;
}
return;
function get_anchor ()
{
return anchor;
}
}
/**/
};
-
This is great! Does it work with videos too?
Artwork only right now, so that does include artwork video.. but I could easily modify it to work with standard images/videos as well.
I know I made quite a few changes - don't want to ruin your learning experience with squirrel, but I was looking for something very similar as well :) It should be clear enough what's what and if you have any questions, just let me know.
I've been playing around and you can do some pretty cool setups :) It would be really nice to work with "fan art" but looks pretty good with other art too.
Pretty sure there is still some incorrect things for wider textures, but hopefully get those ironed out.
-
Yeah, no worries: I was missing the concept of creating new tables on the root (e.g., ::Anchor <- {..}), so that was definitely helpful.
I still want to investigate animation. Nothing that hasn't already been done, but knowing how it's been done should help should I want to venture into deeper waters in the future! :)
-
I'm sure I have some of the math wrong, probably for wide textures but working to fix that.. here's a couple mockups layouts I did using FillArt:
https://streamable.com/ov10
https://streamable.com/p14r
https://streamable.com/gvmi
Ignore the choppy animation, as that has to do with AM running in software mode on this machine. Notice in the last one you don't actually have to fill it and can set the scale smaller and it still applies the anchor - in that one it is anchored to Top.
-
That is extremely cool!
I reworked some of what I'd been working on, just because, to get the fill working across different widths/height. I ended up running two conditionals: one to initially apply the resize, and a second to "re" resize if there was still some underlap. I had a few snaps that were still not filling, but they are now. I'm sure there's better math, but it's working. Nothing as cool as the animation though! :)
-
All right, another conundrum!
I'm using the conveyor module because I want to animate a SimpleArtStrip moving left to right. While it works as it should when used as-is, I'm trying to incorporate the FillArt module rather than generic, uncorrected aspect ratio add_artwork objects.
I'd be happy with rewriting my simplified version of the FillArt module, but I'm not sure how. Here's where I'm stuck. If I take something like the following:
local my_surface=fe.add_surface(100, 100);
local my_image=my_surface.add_image("imagepath.png");
would my_image hold any reference to my_surface? Is there any kind of "parent" attribute or similar? In my current implementation, the FillArt class allows you to pass a reference to the surface to which you'd like to apply it (defaults to ::fe). Now, I'd like to fix it so I could do something like the following:
local my_surface=fe.add_surface(100, 100);
local my_filled_art=my_surface.add_image(FillArt("snap", 0, 0, 100, 100));
and have the FillArt class obtain "parent" info from its instantiation (in this case, "my_surface"). Can this be done? Even better would be to enable something that let me do this:
// Assume my_surface as above
local my_filled_art=my_surface.add_filled_art("snap", 0, 0, 100, 100);
but I'm not sure if the overarching fe object could be appended as such.
Any thoughts? Thanks!
-
That's exactly what I did with my FillArt implementation. There is no reference to the parent surface from surface.add_xxx functions so you can do one of two things... use the my_surface reference you already created or do what I did - write a class which keeps a variable called 'parent' that references the surface like so:
class MyObject
{
parent = null;
art = null;
constructor( name, x, y, w, h )
{
parent = ::fe.add_surface(w,h);
parent.x = x;
parent.y = y;
art = parent.add_artwork( name, 0, 0, parent.width, parent.height );
}
}
Now this class has reference to both the parent object (the surface) and the artwork:
local obj = MyObject( "snap", 50, 50, 320, 240 );
local parent = obj.parent;
local art = obj.art;
Without creating 'redirection' _get and _set functions though like I did in FillArt, you will have to use those references to modifying the settings - in other words, obj.x/.y/.width/.height don't exist in the class - you'd have to use obj.parent.x/.y/.w/.h to change the surface properties, or obj.art.x/.y/.width/.height to change the artwork (on the surface) properties.
-
Okay, I think I can wrap my head around that... :) Is the posted version of your FillArt module up-to-date? Or is there a more complete version kicking around on your hard drive? And if so, might I take a peek at it? Thanks!
-
Okay, I can't wrap my head around that. I'm not sure why Squirrel is proving so difficult for me to comprehend! :(
Have you given any thought to integrating your FillArt module with the Conveyor/SimpleArtStrip? All of my attempts end in tears and ashes :(
-
Classes act like most languages - I'd recommend learning a bit about how they work but let me try to give a little help:
You define classes to describe how an object will work - what it's properties are and what "functions" it can perform. Then, when you use the class you "instantiate" an object - make an instance of it. This is just what fe.add_artwork, fe.add_image, etc do - when you add those lines in your code, you are creating an instance of the artwork class, or image class, etc..
local my_image = fe.add_image( "snap", 0, 0, 320, 240 );
Since my_image is now an instance of the image class - you have access to all its defined properties and functions - .x, .y, .set_rgb(), etc.
What I'm describing here is creating your own class but it will have references to other AM objects. We're basically wrapping multiple objects into our own class. The "constructor" is essentially a function that runs when you create the instance. So look a this example:
class RedImage
{
image = null;
constructor( name, x, y, w, h)
{
image = fe.add_image( name, x, y, w, h );
red();
}
function red()
{
image.set_rgb( 255, 0, 0 );
}
}
local red_image = RedImage( "test.png", 0, 0, 320, 240 );
So, we created an instance of 'RedImage', the constructor runs - which sets the 'image' variable to an instance of fe.add_image, then the 'red' function is called which sets the rgb of the image to red.
In your case, you want to work with a surface and objects on the surface, so the class might look like this:
class MySurface
{
parent = null;
art = null;
constructor(x, y, w, h)
{
parent = fe.add_surface(w, h);
art = parent.add_artwork("snap", 0, 0, w, h );
}
function do_something()
{
print("I did something\n");
}
}
local obj = MySurface(0, 0, 320, 240 );
Now the constructor runs - creates a surface (parent) and adds a snap artwork (art) to the instance. We can now use our defined instance 'obj' to access those, or run any functions we define, or variables that are defined within the class.
It's pretty easy to modify the SimpleArtStrip/SimpleArtStripSlot to use something other than the artwork, but I think it'd be better for you to learn that :)
-
Thanks for the lesson :) I know how classes work, favoring composition over inheritance in most cases, etc., but I know it from an ECMA perspective (specifically ActionScript), so the syntactical differences are what are tripping me up.
Do you know if there's a good resource learning Squirrel with some actually useful examples? I mean I've looked through the reference docs, but they're extremely sparse, to the point of being of no use to someone who doesn't understand the lexical peculiarities. Thanks!
Edit: One example: if I were writing an accessor or mutator function in ActionScript, I'd have something like this:
// pseudocode
class ClassName
{
_parent=new Sprite();
_art=new Sprite();
// constructor
ClassName () {}
function get art () { return _art; }
function set art (val) { _art=val; }
}
I see there are metamethods (_get and _set) in Squirrel, but they don't work that way, and blah blah blah... :(
Again, thanks for any insight!
-
Yes the reference docs are very... literal :) Definitely check out this:
https://electricimp.com/docs/squirrel/squirrelcrib/
Also, I started a brief intro to squirrel and layouts here:
https://github.com/mickelson/attract/wiki/Introduction-to-Squirrel-Programming
I haven't gotten around to things like classes in that though.
class ClassName
{
_parent = null;
_art = null;
constructor()
{
_parent = Sprite();
_art = Sprite();
}
}
Two main differences here.
Notice I set to null, then initialized in the constructor - this is because variables that are not integer, bool or string are shared between class instances if they are initialized in the class (as in your example). Doing this forces the variable to be unique to each instance.
get and sets are not necessary. As far as I know there is no private variables. You could still do functions for getting and setting if it makes sense for the class. There is however a backend, hidden get and set for variables of the class which is the _get and _set I use in FillArt. Those are run whever some attempts to get or set a variables for the class. If you did my_class.something = 10 - the _set function is called and you could just allow it to be set, or override the behavior.
Hope this helps! I can probably put together a SimpleFillArtStrip / SimpleFillArtStripSlot sometime tomorrow. I'd recommend trying to look at the example in conveyor and trying to figure out how to do it for yourself though.
-
Okay, I've worked on this quite a bit since I need this for a few different things and it seems quite useful:
(http://i.imgur.com/fuELhG2.png)
Does that all look like what we'd want in relation to the art's container size and the art aspect ratio? I think that is all correct, but still doing some tests. I've reworked the fill-art module so it does fit or fill, depending on preference - hence PreserveArt - this also works as a PreserveImage.
Once I have a few more kinks worked out, I'll post the code here, upload it to my github and possibly work on a conveyor mod. I'm doing something called a 'Shelf' right now which is similar:
(http://i.imgur.com/1khljl8.png)
And finally, this is Shelf using FILL with PSP (tall textures) anchored to the texture top:
(http://i.imgur.com/WJfbka2.png)
;D
-
Liquid8d-
WOW! This is great work. Are these value or reference based containers, I assume reference based? And, how does scaling affect the objects in the container. Would it be a stretch to call these containers flexible surfaces.
-
Each object has its own surface with the art added to it. The art is then scaled and anchored on the surface appropriately based on the surface dimensions and the art texture dimensions. I still can't for the life of me quite get the math (i.e. aspects ) right. It works shown in the pics above, but some texture dimensions still aren't scaling/anchoring correctly.
Ideally this could be likened to setting the aspect on your tv or video player - crop, fit, fill, etc.. if I can get the math right that's how it should work.
FYI - I haven't forgotten about the animate module, although this has taken some time away from that. I still have some fairly major fixes for it to work properly to take over for v1 :)
-
This probably isn't the math you're talking about, but I took the previous sizing branches you'd posted and discovered some tweaks that needed to be made. I think these are all accurate:
function update ()
{
local _pAspect=pdims.w.tofloat()/pdims.h.tofloat(); // parent dimensions
local _tAspect=tdims.w.tofloat()/tdims.h.tofloat(); // texture dimensions
/*
::print ("_pAspect: "+_pAspect+"; _tAspect: "+_tAspect+"\n");
::print ("pdims: "+pdims.w+", "+pdims.h+"\n");
::print ("tdims: "+tdims.w+", "+tdims.h+"\n");
/**/
if (_pAspect >= 1.0)
{
// Fill to parent width
if (_tAspect >= 1.0)
{
// Wider or equal
art.width=(tdims.w*(pdims.w/tdims.h.tofloat()));
}
else
{
// Taller
art.width=(tdims.w*(pdims.w/tdims.w.tofloat()));
}
art.height=art.width/_tAspect;
}
else
{
// Fill to parent height
if (_tAspect >=1 )
{
art.height=(tdims.w*(pdims.h/tdims.h.tofloat()));
}
else
{
art.height=(tdims.w*(pdims.h/tdims.w.tofloat()))
}
art.width=art.width*_tAspect;
}
switch (anchor)
{
case ::Anchor.TopLeft:
art.x=0;
art.y=0;
break;
case ::Anchor.Top:
art.x=(pdims.w-art.width)/2.0;
art.y=0;
break;
case ::Anchor.TopRight:
art.x=pdims.w-art.width;
art.y=0;
break;
case ::Anchor.Left:
art.x=0;
art.y=(pdims.h-art.height)/2.0;
break;
case ::Anchor.Right:
art.x=pdims.w-art.width;
art.y=(pdims.h-art.height)/2.0;
break;
case ::Anchor.BottomLeft:
art.x=0;
art.y=pdims.h-art.height;
break;
case ::Anchor.Bottom:
art.x=(pdims.w-art.width)/2.0;
art.y=pdims.h-art.height;
break;
case ::Anchor.BottomRight:
art.x=pdims.w-art.width;
art.y=pdims.h-art.height;
break;
case ::Anchor.Center:
default:
art.x=(pdims.w-art.width)/2.0;
art.y=(pdims.h-art.height)/2.0;
break;
}
// ::print ("art.width: "+art.width+"; art.height: "+art.height+"\n");
}
You've added tons more bells and whistles than what I'd been thinking about, but it's been great seeing how to extend the module's usefulness!
-
Liquid8d-
No worries on the animate module, mate. I was just trying to give you a little nudge. Out of all the projects you are working on this one probably should take priority for its relevance.
For what it's worth, when I was looking for a way to scale my layouts I found this simple code for my cheat sheets. To proportionally scale an image to the dimensions of its parent container you could use something like this. Setting the dimensions of your container as a variable. But, I still learning and have a very basic understanding of this stuff. :)
var scale:Number = ((container.height/container.width) > (image.height/image.width)) ? (container.width/image.width) : (container.height/image.height);
image.scaleX = scale;
image.scaleY = scale;
-
I revisited my width/height logic and revised it some. There might be a more efficient way to do this (I was right on the cusp of something while I was in the shower, but without anything to write on, I promptly forgot!) Maybe this will be additional food for thought:
function update ()
{
local _target_aspect=null;
local _source_aspect=null;
if (_target.width>=_target.height)
{
// Wider (or equal)
_target_aspect=_target.width/_target.height.tofloat();
if (_source.width>=_source.height)
{
// Art is wider than tall; favor height for sizing
_source_aspect=_source.width/_source.height.tofloat();
art.width=_target.height*_source_aspect.tofloat();
art.height=_target.height;
// One last check in case sizing leaves us short
if (art.width<_target.width)
{
art.width=_target.width;
art.height=_target.width*(1.0/_source_aspect);
}
}
else
{
// Art is taller than wide; favor width for sizing
_source_aspect=_source.height/_source.width.tofloat();
art.width=_target.width;
art.height=_target.width*_source_aspect.tofloat();
// One last check
if (art.height<_target.height)
{
art.width=_target.width*(1.0/_source_aspect);
art.height=_target.height;
}
}
}
else
{
// Taller
_target_aspect=_target.height/_target.width.tofloat();
if (_source.width>=_source.height)
{
// Art is wider than tall; favor height
_source_aspect=_source.width/_source.height.tofloat();
art.width=_target.height*_source_aspect;
art.height=_target.height;
// One last check
if (art.height<_target.height)
{
art.width=_target.height*(1.0/_source_aspect);
art.height=_target.height;
}
}
else
{
// Art is taller than wide; favor width
_source_aspect=_source.height/_source.width.tofloat();
art.width=_target.width;
art.height=_target.width*_source_aspect;
// One last check
if (art.height<_target.height)
{
art.width=_target.height*(1.0/_source_aspect);
art.height=_target.height;
}
}
}
// Set anchor position
switch (anchor)
{
case ::Anchor.TopLeft:
art.x=0;
art.y=0;
break;
case ::Anchor.Top:
art.x=(_target.width-art.width)/2.0;
art.y=0;
break;
case ::Anchor.TopRight:
art.x=_target.width-art.width;
art.y=0;
break;
case ::Anchor.Left:
art.x=0;
art.y=(_target.height-art.height)/2.0;
break;
case ::Anchor.BottomLeft:
art.x=0;
art.y=_target.height-art.height;
break;
case ::Anchor.Bottom:
art.x=(_target.width-art.width)/2.0;
art.y=_target.height-art.height;
break;
case ::Anchor.BottomRight:
art.x=_target.width-art.width;
art.y=_target.height-art.height;
break;
case ::Anchor.Center:
default:
art.x=(_target.width-art.width)/2.0;
art.y=(_target.height-art.height)/2.0;
break;
}
}
-
OKAY. I think this is accurate, this is what I finally came up with :) Going to do a bit more testing and I will add this to my github..
(http://i.imgur.com/7r1POja.png)
/////////////////////////////////////////////////////////
//
// Attract-Mode Frontend - PreserveArt/PreserveImage Module
//
/////////////////////////////////////////////////////////
// PreserveArt - creates a surface with artwork that will
// fit or fill the specified dimensions. This preserves
// art aspect ratio, but gives option to anchor the art
// to one side of its dimensions.
//
::Anchor <-
{
Top = "Top",
Left = "Left",
Centre = "Center",
Center = "Center",
Right = "Right",
Bottom = "Bottom",
}
class PreserveArt
{
VERSION = 1.0;
debug = false;
surface = null;
art = null;
isArt = true;
anchor = ::Anchor.Center;
fit_or_fill = "fit";
request_size = true;
constructor( name, x, y, w, h, parent = ::fe )
{
surface = parent.add_surface( w, h );
surface.x = x;
surface.y = y;
if ( debug ) surface.add_image( ::fe.script_dir + "pixel.png", 0, 0, surface.width, surface.height );
art = ( isArt ) ? surface.add_artwork( name, 0, 0, w, h ) : surface.add_image( name, 0, 0, w, h );
::fe.add_transition_callback( this, "onTransition" );
::fe.add_ticks_callback( this, "onTick" );
}
function set_anchor( a )
{
anchor = a;
request_size = true;
}
function set_fit_or_fill( f )
{
fit_or_fill = f;
request_size = true;
}
function update()
{
if ( art.texture_width != 0 && art.texture_height != 0 )
{
request_size = false;
local aspect = surface.width / surface.height.tofloat();
local texture_aspect = art.texture_width / art.texture_height.tofloat();
if ( fit_or_fill == "fit" )
{
if ( aspect > 1.0 )
{
//wide
if ( texture_aspect > 1.0 )
{
print("wide, wide, fit");
art.width = surface.width;
art.height = art.width / texture_aspect;
if ( art.height > surface.height )
{
//reverse
art.height = surface.height;
art.width = art.height * texture_aspect;
}
} else
{
print("wide, tall, fit");
art.height = surface.height;
art.width = art.height * texture_aspect;
if ( art.width > surface.width )
{
//reverse
art.width = surface.width;
art.height = art.width / texture_aspect;
}
}
} else
{
//tall
if ( texture_aspect > 1.0 )
{
print("tall, wide, fit");
art.width = surface.width;
art.height = art.width / texture_aspect;
if ( art.height > surface.height )
{
//reverse
art.height = surface.height;
art.width = art.height * texture_aspect;
}
} else
{
print("tall, tall, fit");
art.height = surface.height;
art.width = art.height * texture_aspect;
if ( art.width > surface.width )
{
//reverse
art.width = surface.width;
art.height = art.width / texture_aspect;
}
}
}
} else if ( fit_or_fill == "fill" )
{
if ( aspect > 1.0 )
{
//wide
if ( texture_aspect > 1.0 )
{
print("wide, wide, fill");
art.height = surface.height;
art.width = art.height * texture_aspect;
if ( art.width < surface.width )
{
//reverse
art.width = surface.width;
art.height = art.width / texture_aspect;
}
} else
{
print("wide, tall, fill");
art.width = surface.width;
art.height = art.width / texture_aspect;
if ( art.height < surface.height )
{
//reverse
art.height = surface.height;
art.width = art.height * texture_aspect;
}
}
} else
{
//tall
if ( texture_aspect > 1.0 )
{
print("tall, wide, fill");
art.width = surface.width;
art.height = art.width / texture_aspect;
if ( art.height < surface.height )
{
art.height = surface.height;
art.width = art.height * texture_aspect;
}
} else
{
print("tall, tall, fill");
art.height = surface.height;
art.width = art.height * texture_aspect;
if ( art.width < surface.width )
{
art.width = surface.width;
art.height = art.width / texture_aspect;
}
}
}
} else
{
//stretch
art.preserve_aspect_ratio = false;
art.width = surface.width;
art.height = surface.height;
}
switch ( anchor )
{
case ::Anchor.Left:
art.x = 0;
art.y = ( surface.height - art.height ) / 2;
break;
case ::Anchor.Right:
art.x = surface.width - art.width;
art.y = ( surface.height - art.height ) / 2;
break;
case ::Anchor.Top:
art.x = ( surface.width - art.width ) / 2;
art.y = 0;
break;
case ::Anchor.Bottom:
art.x = ( surface.width - art.width ) / 2;
art.y = surface.height - art.height;
break;
case ::Anchor.Center:
default:
art.x = ( surface.width - art.width ) / 2;
art.y = ( surface.height - art.height ) / 2;
break;
}
} else
{
//fallback to preserve_aspect_ratio
print("WARNING: FALLBACK TO PRESERVE_ASPECT_RATIO" );
art.x = 0;
art.y = 0;
art.width = surface.width;
art.height = surface.height;
art.preserve_aspect_ratio = true;
}
}
function _get( idx ) { return surface[idx]; }
function _set( idx, val )
{
if ( idx == "index_offset" ) art.index_offset = val; else surface[idx] = val;
}
function print( msg )
{
if ( debug ) ::print("PreserveArt: " + msg + "\n" );
}
function set_rgb( r, g, b ) { surface.set_rgb( r, g, b ); }
function onTransition( ttype, var, ttime )
{
if ( ttype == Transition.StartLayout || ttype == Transition.ToNewList || ttype == Transition.FromOldSelection ) request_size = true;
}
function onTick( ttime )
{
if ( request_size && art.texture_width != 0 && art.texture_height != 0 ) update();
}
}
class PreserveImage extends PreserveArt
{
isArt = false;
}
-
I'm trying to wrap my head around it.
What option will give me:
- anchor bottom-center
- fit-to-width
- preserve aspect
I'm thinking about displaying snaps - both vertical and horizontal - in 4:3 window, so horizontal snaps will be 100% visible (fill the window completely) and vertical snaps will be cut-off from the top (but preserve aspect and fill window completely)
-
I think just setting the anchor to Anchor.Center will do what you want: 4:3 flyers will show full size and "3:4" flyers will show the central portion while filling the width of the screen. If I'm understanding your question...
-
Here's examples using it:
local art = PreserveArt( "snap", 0, 0, 320, 240 );
art.set_anchor( Anchor.Bottom );
art.set_fit_or_fill( "fit" );
local image = PreserveImage( "image.jpg", 0, 0, 320, 240 );
image.set_anchor( Anchor.Left );
image.set_fit_or_fill( "fill" );
"fit" or "fill" will both preserve the texture aspect. Fill in this sense fills the size of the object but keeps the texture aspect - "stretch" will be just like preserve_aspect = false and stretch the texture to the width and height of the object.
If you are wanting to change the setting between fit or fill, you can use the set_fit_or_fill function, but will need to write some code to determine when to switch from one to the other (check texture size on transition)
Let me know if you have other questions or suggestions!
-
Great stuff - especially for responsive layouts.
Thanks.
-
looks great liquid8d. Could you do a pull request against http://github.com/mickelson/attract/tree/master/config/modules when you are happy with this? It would be nice to have it included in the base distribution of the frontend.
-
I've added it to my attract-extra fork. Still a little green on git - not sure how I submit a pull for the specific file across forks (liquid8d/attract-extra) to (mickelson/attract)
:)
-
I've added it to my attract-extra fork. Still a little green on git - not sure how I submit a pull for the specific file across forks (liquid8d/attract-extra) to (mickelson/attract)
:)
Hehe me too (on being green on git). I think the steps go like this if I recall correctly:
- clone http://github.com/mickelson/attract locally
- make your updates and additions to the module files in config/modules
- commit the changes
- push your local repository to your github account
once you do that, there should be a "Create Pull Request" button that shows up in the github web ui, which you can just click to send your changes as a pull request.
It's probably a lot like how you are already working with the attract-extra repository, just with attract instead (and the added step of making the pull request in the github ui).
-
that makes sense - i do have a fork of attract, just playing with some customizations locally until maybe I can contribute once I am more familiar with things :) Didn't want to have multiple versions floating around in both forks. I guess as long as I create any modules/layouts/plugins in extra first, I can just push them to attract and submit from there.
Just looking to add one more thing (ability to scale) and verifying the update tick works properly before I submit it.