Lab 08 - Picture Puzzles & Filters

Due by 11:59pm on February 11, 2025

Starter Files

Download lab08.zip. Inside the archive, you will find starter files for the questions in this lab.

Download the zip file, unzip it, and move it to your cs111/labs folder. You should find within it:

  • test_files/copper.png
  • test_files/iron.png
  • test_files/west.png
  • test_files/cougar.png
  • lab08.py
  • test_lab08.py

Topics

Installing the BYU Image Library

To start coding with images, install BYU's image library by either typing one of the following commands into the terminal. If one does not work try the other option.

pip install byuimage
python3 -m pip install byuimage

To uninstall a library at any time, type into the terminal pip uninstall <library_name> or python3 -m pip uninstall <library_name>

Note: In some cases, instead of pip you may need to try pip3 or use python or py to run your code; use your respective version when installing libraries (e.g. py -m pip install <library_name> instead of the one given above). Then, when running your code, make sure to use the same version that you installed the library with.

To check if you installed the library correctly, open up the python interpreter by typing python3, python, or py (whichever works for you), and type from byuimage import Image into the interpreter. If nothing errors, you installed the BYU image library correctly. If it errors, try the another installation option. If you installed the BYU image library, properly it should look something like this:

Python 3.13.1 (main, Dec  3 2024, 17:59:52) [Clang 16.0.0 (clang-1600.0.26.4)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> from byuimage import Image
>>> quit()

In some cases, if you installed Python multiple times, there will be a problem between the Python version that your IDE (Pycharm, VS Code, etc.) and the Python version that the terminal uses, i.e., they will use different versions. Optionally, if you want to check if your IDE is working properly,, run lab08.py WITH from byuimage Import image at the top of the file using your IDE's run button. If you get an error, you can refer to Pycharm's or VS Code's guide for configuring a Python Interpreter: Pycharm, VS Code (do not worry about the virtual environment section).

Image Review

Digital images consist of pixels. A pixel is a small square, showing a single color where each color can be controlled using a combination of red, green and blue (RGB). Each color ranges from a value of 0 (minimum) to 255 (maximum) and mix to form the color of the pixel.

Coding with Image

To code with images, you need to import or load the BYU image library for your program at the start/top of the file. (Recall that a library is just some pre-written code). From that library you need to import specifically Image like so:

from byuimage import Image

Image takes in a string as an argument. This string should contain an image's filename or the path to the image's filename. Image effectively grabs that image filename that we provided and turns it into an image we can work with in our code.

The following is an example on how you can go through each of the pixels in an image, modify each pixel, and show the result. Demo the following code by copying and pasting it at the bottom of your lab08.py file:

cougar_image = Image("test_files/cougar.png")
# ^ path to `cougar.png` file
for pixel in cougar_image:
pixel.red = 0
pixel.green = 0

cougar_image.show()

Note: There is also a pixel.blue attribute.

Pixel colors can also be edited all at once using the pixel.color attribute and a list or tuple ordered (, , ):

cougar_image = Image("test_files/cougar.png")
# ^ path to `cougar.png` file
for pixel in cougar_image:
pixel.color = (0, 0, pixel.blue) # set red and green to 0, keep blue the same

cougar_image.show()

After creating a new image, you can also access the image's height, width, and individual pixels by using the following:

  • <image>.height - returns image's height
  • <image>.width - returns image's width
  • <image>.get_pixel(x,y) - returns pixel at (x, y). x and y have to be integers.

With these new tools, we are allowed a new way of going through each pixel in the image. For example:

cougar_image = Image("test_files/cougar.png")
for y in range(cougar_image.height):
for x in range(cougar_image.width):
pixel = cougar_image.get_pixel(x,y)
# etc.

You can think of the first for loop as going through each row where y is a individual row index, and you can think of the second for loop as going through each column or slot in the row where x is a individual column or slot index

It is important to note that an image's origin (0, 0) starts in the upper left corner. Moving towards the right is the positive x direction, and moving downwards is the positive y direction.

Discuss:

  • Why would we use double for loops instead of the for pixel in image method?
  • What difference does it make if you switch the first for loop (for y in ...) with the second for loop (for x in ...)? How does it change the traversal of the image?

Creating New Images

To create a white blank image with certain dimensions, you can use Image.blank(<width>, <height>) where width and height are integers. Image.blank returns a completely white image with the specified dimensions where each pixels' RGB values are set to 255.

For example,

image = Image.blank(100, 50) # 100 pixels wide, 50 pixels high
image.show()

Required Questions

Q1: Iron Puzzle

Note: TA's should demonstrate how to solve this problem.

This is the iron.png file in the zip file you downloaded:

iron

This picture contains an image of something famous; however, the image has been distorted. The famous object is all in blue values, but the blue values have all been divided by 10, so they are too small by a factor of 10. The red and green values are all just meaningless random values (“noise”) added to obscure the real image. You must undo these distortions to reveal the real image.

All your code will be in a function called iron_puzzle(filename). It takes one parameter, the filename of the image with the puzzle in it. It should return the edited iron.png image with the distortions undone. Your code should be able to show the modified image with code like this:

def iron_puzzle(filename):
# load the image, solve the puzzle by modifying the image, return the modified image


solution_image = iron_puzzle("test_files/iron.png")
solution_image.show()

Try it!

Test your code:

python3 -m pytest test_lab08.py::test_iron_puzzle
pytest test_lab08.py::test_iron_puzzle

Note: You must install byu_pytest_utils in order to use the tests given above.

Hint: You will need to loop over all pixels in the image. First, set all the red and green values to 0 to get them out of the way. Look at the result. If you look very carefully, you may see the real image, although it is very very dark (way down towards 0). Now multiply each blue value by 10 scaling it back up to approximately its proper value.

Q2: West Puzzle

west

The hidden image is exclusively in the blue values, so set all pixels' red and green values to 0. The hidden image is encoded using only the blue values that are less than 16 (that is, 0 through 15). If a blue value is less than 16, multiply it by 16 to scale it up to its proper value. Alternately if a blue value for any pixel is 16 or more, it is random garbage and should be ignored (interpreted as 0). The west_puzzle function should return the recovered image.

Write a function called west_puzzle that has one parameter filename to recover the hidden image. Try using two for loops when traversing over each of the pixels.

After commenting out or removing the call to the iron_puzzle function, you should be able to run your west_puzzle with these lines of code:

solution = west_puzzle("test_files/west.png")
solution.show()

Try it when you are finished!

Test your code:

python3 -m pytest test_lab08.py::test_west_puzzle
pytest test_lab08.py::test_west_puzzle

Q3: Darken Image

Note: The following functions will be used as part of Project 1 so it is in your interest to get them completed in lab so you don't have to finish them on your own.

For this problem, write a function that can darken an image an arbitrary amount. The function definition should be:

def darken(filename, percent):

The filename parameter contains the image's filename to darken, and the percent parameter tells you how much to darken the image by. percent must and will be a number between 0 and 1. The closer percent is to 1, the darker the image should be. darken should return the modified image.

Notice that we can darken a pixel 80% by leaving it 20% as bright. We can do this by multiplying each pixels' RGB values by 0.2:

pixel.red = pixel.red * 0.2
pixel.green = pixel.green * 0.2
pixel.blue = pixel.blue * 0.2

This effectively leaves a pixel 20% as bright. Doing so with the cougar.png image would result in an image that looks something like:

cougar darken answer

When you are done, try darkening with various parameters:

solution = darken("myImage.jpeg", 0.3)
solution.show()

darken("myImage.jpeg", 0.6).show()
# ^ This works exactly the same as the previous lines above. Why?

Test your code:

python3 -m pytest test_lab08.py::test_darken
pytest test_lab08.py::test_darken

You can use cougar.png or download an image from Unsplash to use for this problem and move it to your lab08 folder. Unsplash provides free pictures that are legal to use as you want. You should always practice following copyright rules, so use a site like Unsplash instead of Google Images. Skip over the premium images that are shown on Unsplash and use the free ones. Download the small version of images so your code will run faster since you will have fewer pixels to loop over.

Q4: Grayscale Filter

For this problem, write a function called grayscale(filename) that takes the filename of an image and returns a monochrome image (an image only in black, gray, and white). Used on the Cougar.png image that would look something like this:

cougar grayscale answer

To make an image gray, we will use this algorithm:

  • Loop through all of the pixels.
  • For each pixel, calculate the average color:
    average = (pixel.red + pixel.green + pixel.blue) / 3
  • Set the pixel's red, green, and blue values equal to the average.

You should be able to call your function like this:

gray = grayscale("myImage.jpeg")
gray.show()

Try using grayscale with an image.

Test your code:

python3 -m pytest test_lab08.py::test_grayscale
pytest test_lab08.py::test_grayscale

Q5: Sepia Filter

You have probably used a sepia filter for a photo before. Write a function called sepia(filename), which takes a filename parameter and returns modified image that uses a sepia filter. Used on the cougar.png image that would look something like this:

cougar sepia answer

To apply the sepia filter, you will need to loop over all pixels, and then for each pixel, compute these values:

true_red = 0.393*pixel.red + 0.769*pixel.green + 0.189*pixel.blue
true_green = 0.349*pixel.red + 0.686*pixel.green + 0.168*pixel.blue
true_blue = 0.272*pixel.red + 0.534*pixel.green + 0.131*pixel.blue

Then, set each pixel color. For example:

pixel.red = true_red

Finally, it may be the case that the color value is greater than 255. If that happens, set that pixel's color value to the maximum value of 255. For example:

if pixel.red > 255:
pixel.red = 255

Run your function like previously:

solution = sepia("myImage.jpeg")
solution.show()

Test your code:

python3 -m pytest test_lab08.py::test_sepia
pytest test_lab08.py::test_sepia

Q6: Left Border

Write a function called create_left_border(filename, weight) that adds a blue border to the left of the image given the weight. For example, with cougar.png and weight = 25:

cougar left border answer

You should increase the image's width by weight pixels where the new area is blue. The image's size should not decrease because of the border.

To do this you will have to

  1. Create a new larger image.
  2. Make the new area for the border blue.
    • Do this by setting the pixel's green and red values to 0 while setting the blue value to 255.
    • You can either color the entire new image blue, then copy the old image onto the new image OR color only the new area blue. (The first approach is easier, but the second approach is significantly more efficient.)
  3. Copy the contents of the old image to the new image.
    • Iterate over each pixel in the old image and find the correct pixel in the new image. Then copy the contents of one pixel to another.

Test your code:

python3 -m pytest test_lab08.py::test_create_left_border
pytest test_lab08.py::test_create_left_border

Hints

  • Use the double for loop strategy of going through the pixels in the image mentioned above with get_pixel(x,y). Doing this will allow us to use both pixels from both images for copying.
  • To copy the contents of one pixel to another, we can do something like new_pixel.red = original_pixel.red for each of the colors where new_pixel would be the pixel for the new larger image and original_pixel would be the pixel in the original image.
  • When finding the correct pixel to copy over the original image to the bordered image, the pixel will just be the original image's pixel's x position plus the weight.

Submit

Test your entire lab:

python3 -m pytest test_lab08.py
pytest test_lab08.py

Submit the lab08.py file on Canvas to Gradescope in the window on the assignment page.

Extra Practice

Q7: Stripes

Let's bring everything together. Write a function called create_stripes(filename) that takes in an image file. You should:

  • Create a new blank image with the same dimensions as the image given in filename, but the width is an extra 50 pixels and the height is an extra 25 pixels.
  • With the following priority:
    1. Make the pixel green if it is on a even row
    2. Make the pixel blue if it on a odd column (or slot)
    3. Otherwise, make the pixel red
  • Return the newly edited image

Your image should look something like this (perhaps after you zoom in several times)

stripes

Keep in mind that we start counting x and y at 0, so despite the fact that to us the first row is green, it is technically the 0th row which is why we made it green. (Same scenario with the columns/slots.)

Q8: Copper Puzzle

The copper puzzle is another image puzzle. Use the copper.png file that is in the zip file you downloaded. It too has been distorted:

copper

Write a function called copper_puzzle(filename) to recover the hidden image.

The hidden image is in the blue and green values; however, all the blue and green values have all been divided by 20, so the values are very small. The red values are all just random numbers, noise added on top to obscure things. Undo these distortions to reveal the true image.

You should be able to run your code with:

solution = copper_puzzle("test_files/copper.png")
solution.show()

What is the hidden image?

What happens if you fix only the green values or only the blue values?

© 2025 Brigham Young University, All Rights Reserved