At My Fingertips
Rapid Playground
In this activity you practice rotating and composing graphics.
Here is the Taijitu, a famous symbol representing the concept of yin and yang!
At first sight, this graphic looks non-trivial.
The two small circles stand out. But how is the rest built? You may have seen it, it's simpler than it might initially have looked: all you need is circles and semicircles (circular sectors of 180 degrees)!
circle
FunctionIf you haven't already done so, add a circle
function to your toolbox.
Your function should have the following signature:
def circle(diameter: float, color: Color) -> Graphic
If you don't have that function yet, you may want to do the Simple Snowman activity to learn how to put a function in the toolbox.
Once you have a circle
function in your toolbox, import it here:
With this import statement, you told Python that the name circle
refers to the circle function you had added to your toolbox. From now on, in this activity here, you can use that name to call that function.
Try it out! Let's show a black circle with a diameter of 100:
semicircle
FunctionThe second kind of graphic we need is a semi-circle. Write a function that creates one, by calling circular_sector with an angle of 180 degrees:
Now let's create a black semicircle with a diameter of 100 and show it:
Now we are ready to construct the Yin Yang symbol! Look carefully at the symbol and think how you could compose it from circles and semicircles.
Implement the yin_yang
function below.
You may find the imported functions beside
, rotate
, above
, and overlay
,
and the colors black
and white
useful.
You are done! You have a working yin_yang
function!
Or are you?
The code we write is read by two different entities. The computer (or, more precisely, the Python interpreter, the program that reads and executes Python code) is relatively easy to satisfy. You can feed it code that looks horrible. And it happily executes it, as long as it follows the syntax and semantics of the Python language.
But besides the computer, there's also someone else who reads your code: humans! That includes you, and also your teammates! In the future, those people may have to understand your code to fix bugs, to improve its performance, or to add new features to it. And if you wrote correct but horrible code, those people (and that includes you!) will not be amused!
So, whenever we have a working function, we clean it up! We refactor it to make it easier to understand for humans. Programmers who write understandable code are cherished. But nobody wants to work with programmers who write hard-to-read code.
So, let's clean up your yin_yang
function to make it easy to understand.
So, here are two different ways the function could be written. There are many others. But these two differ in their use of names for naming intermediate results.
def yin_yang(diameter: float) -> Graphic:
return overlay(
above(
overlay(
circle(diameter / 6, black),
circle(diameter / 2, white)
),
overlay(
circle(diameter / 6, white),
circle(diameter / 2, black)
)
),
beside(
rotate(90, semicircle(diameter, white)),
rotate(-90, semicircle(diameter, black))
)
)
If you want to understand the above code, you have to pretty much read it all. It is nicely indented, and the entire structure of the yin yang graphic is visible in one place. But the structure is quite big. It probably makes sense to name some of the pieces, so that a reader (you, in the future, or some future teammate) doesn't need to infer what the pieces of the expression correspond to.
def yin_yang(diameter: float) -> Graphic:
white_eye = overlay(
circle(diameter / 6, black),
circle(diameter / 2, white)
)
black_eye = overlay(
circle(diameter / 6, white),
circle(diameter / 2, black)
)
eyes = above(white_eye, black_eye)
base = beside(
rotate(90, semicircle(diameter, white)),
rotate(-90, semicircle(diameter, black))
)
return overlay(eyes, base)
The above code extracts meaningful pieces of the large expression into smaller subexpressions,
so it can introduce names for them.
If we read an expressions like overlay(eyes, base)
,
we can kind of imagine what it means,
and we don't have to read overlay(
10 lines of code,
4 lines of code)
.
That's a benefit of naming things:
we don't need to read much to have a reasonably good idea of what's going on.
Of course, the names we pick need to be accurate.
We wouldn't want to use overlay(honeybee, flower)
to describe a yin yang!
If your code has potential for improvement, refactor it now, to make it more readable.
Then, if you think you might want to reuse your yin_yang
function in other activities
(or in the playground, where you can be creative and compose whatever graphics you like),
feel free to save it in your toolbox.
You also could save your semicircle
function;
it looks like a generally useful function that could come in handy in the future.
Isn't the yin yang symbol beautiful? Not just visually, but also in terms of its composition?
There are multiple ways to compose the symbol. Do you like the way in which you did it? Did you use names for various pieces? Did you inline everything into one "giant" expression? Or did you even create additional functions for creating the various pieces? You can always refactor between the various ways.
If your implementation is easy to understand, for you and for others you show it to, you have a good reason to feel proud!
This activity has been created by LuCE Research Lab and is licensed under CC BY-SA 4.0.
Yin Yang
PyTamaro is a project created by the Lugano Computing Education Research Lab at the Software Institute of USI
Privacy Policy • Platform Version 19a6bb6 (Mon, 27 Jan 2025 12:33:26 GMT)