I have been a Flex developer since 2007, and I finally decided that I needed to start blogging about some of my work. Every once in a while I do something that I think is pretty cool, so I think it is about time I start sharing these experiences.
I have worked on some projects with Flex that involve the user needing to transform objects on the screen. With these projects, the user always had the use of selection handles on a selection box. The user could just select the rotation handle and start dragging the mouse to cause the object on the screen to rotate. However, in a current Flex project, I needed to provide some kind of component to allow the user to rotate an image on the screen without the use of a selection box and selection handles. In the past I have seen something similar using just a horizontal slider or even up and down arrows to increase or decrease the rotation. As I thought about what I should use, I came up with the idea of doing a circular slider. Basically the idea is to have a slider component that rotates onto itself and the user can drag the thumb around the track continuously. So I set about to accomplish this task.
To make this happen, I needed to create a new class that extends from spark.components.supportClasses.SliderBase
. Then I
needed to override two functions: pointToValue(x:Number, y:Number):Number, and updateSkinDisplayList().
The function pointToValue is called when the slider is being moved. The purpose is to take a mouse x,y coordinate and convert that to a value on the slider track between the minimum and maximum set on the slider. To accomplish the circular nature of the the slider, I gather the x,y position of the ellipse at the top, and the center. Then using the mouse x,y position, I use some fancy trigonometry to find the angle of rotation and convert that number according to the minimum and maximum set for the control.
The function updateSkinDisplayList is called continuously for setting the bounds of skin parts, typically the thumb, whose geometry isn't fully specified by the skin's layout. If you look at the HSlider or VSlider, you will see that it mostly is used for positioning the thumb somewhere along the track according the value set by the pointToValue function.
So in the circular slider, I needed to take some value and figure out where to place the thumb along the circular track. This is accomplished by using matrix transformations. I take the current value on the track, and convert to something in the 0 to 359 range and then I find the x,y coordinate of the center of the ellipse. I create a new matrix, and set the rotation to the converted value from the track. I then take the point at the top of the ellipse and use the matrix to convert it. Since the ellipse may not always be a perfect circle, I also check for any scaling and further adjust the position of the thumb.
You can also set the width and height of the slider thumb in the MXML. This is to override the default functionality where the component height would determine the height the thumb.
Another important factor in all of this is the skin file. The original skins for the HSlider and VSlider use Rects to draw the track. In order for all of this to look correctly, you need to create a skin for the track that draws it using Ellipses.
In my demo below, I show three different circular sliders. The one being used is the one selected using the radio buttons below the sliders. I hope this doesn't confuse anybody.