Last active
July 1, 2021 23:15
-
-
Save e200/7c1756994d4d27ee3a7ddfae7d0ff8f7 to your computer and use it in GitHub Desktop.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
#include "helpers.h" | |
#include "math.h" | |
#include "stdlib.h" | |
#include "stdio.h" | |
#include "string.h" | |
#include "cs50.h" | |
int resolveSepiaValue(float value); | |
int getGxValue(int sobelPixelPosition, int colorAmount); | |
int getGyValue(int sobelPixelPosition, int colorAmount); | |
int getGxGyCombination(int gx, int gy); | |
// Convert image to grayscale | |
void grayscale(int height, int width, RGBTRIPLE image[height][width]) { | |
for (int i = 0; i < height; i++) { | |
for (int j = 0; j < width; j++) { | |
RGBTRIPLE pixel = image[i][j]; | |
double colorSum = pixel.rgbtRed + pixel.rgbtBlue + pixel.rgbtGreen; | |
if (colorSum > 0) { | |
double averageGreyColor = colorSum / 3; | |
int greyColor = round(averageGreyColor); | |
image[i][j].rgbtRed = greyColor; | |
image[i][j].rgbtBlue = greyColor; | |
image[i][j].rgbtGreen = greyColor; | |
} | |
} | |
} | |
return; | |
} | |
// Convert image to sepia | |
void sepia(int height, int width, RGBTRIPLE image[height][width]) { | |
for (int i = 0; i < height; i++) { | |
for (int j = 0; j < width; j++) { | |
RGBTRIPLE pixel = image[i][j]; | |
float _sepiaRed = (.393 * pixel.rgbtRed) + (.769 * pixel.rgbtGreen) + (.189 * pixel.rgbtBlue); | |
int sepiaRed = resolveSepiaValue(_sepiaRed); | |
float _sepiaGreen = (.349 * pixel.rgbtRed) + (.686 * pixel.rgbtGreen) + (.168 * pixel.rgbtBlue); | |
int sepiaGreen = resolveSepiaValue(_sepiaGreen); | |
float _sepiaBlue = (.272 * pixel.rgbtRed) + (.534 * pixel.rgbtGreen) + (.131 * pixel.rgbtBlue); | |
int sepiaBlue = resolveSepiaValue(_sepiaBlue); | |
image[i][j].rgbtRed = sepiaRed; | |
image[i][j].rgbtBlue = sepiaBlue; | |
image[i][j].rgbtGreen = sepiaGreen; | |
} | |
} | |
return; | |
} | |
// Reflect image horizontally | |
void reflect(int height, int width, RGBTRIPLE image[height][width]) { | |
RGBTRIPLE copy[height][width]; | |
for (int i = 0; i < height; i++) { | |
for (int j = 0; j < width; j++) { | |
copy[i][j].rgbtRed = image[i][j].rgbtRed; | |
copy[i][j].rgbtBlue = image[i][j].rgbtBlue; | |
copy[i][j].rgbtGreen = image[i][j].rgbtGreen; | |
} | |
} | |
int widthRange = width - 1; | |
for (int i = 0; i < height; i++) { | |
for (int j = 0; j < width; j++) { | |
image[i][j].rgbtRed = copy[i][widthRange - j].rgbtRed; | |
image[i][j].rgbtBlue = copy[i][widthRange - j].rgbtBlue; | |
image[i][j].rgbtGreen = copy[i][widthRange - j].rgbtGreen; | |
} | |
} | |
return; | |
} | |
// Blur image | |
void blur(int height, int width, RGBTRIPLE image[height][width]) { | |
RGBTRIPLE output[height][width]; | |
memcpy(output, image, sizeof(RGBTRIPLE) * height * width); | |
for (int row = 0; row < width; row++) { | |
for (int column = 0; column < height; column++) { | |
int siblingCount = 0; | |
int red = 0; | |
int green = 0; | |
int blue = 0; | |
for (int x = row - 1; x <= row + 1; x++) { | |
if (x >= 0 && x < width) { | |
for (int y = column - 1; y <= column + 1; y++) { | |
if (y >= 0 && y < height) { | |
RGBTRIPLE pixel = output[y][x]; | |
red += pixel.rgbtRed; | |
green += pixel.rgbtGreen; | |
blue += pixel.rgbtBlue; | |
siblingCount++; | |
} | |
} | |
} | |
} | |
image[column][row].rgbtRed = round((float) red / siblingCount); | |
image[column][row].rgbtGreen = round((float) green / siblingCount); | |
image[column][row].rgbtBlue = round((float) blue / siblingCount); | |
} | |
} | |
return; | |
} | |
// Detect edges | |
void edges(int height, int width, RGBTRIPLE image[height][width]) { | |
RGBTRIPLE imageCopy[height][width]; | |
memcpy(imageCopy, image, sizeof(RGBTRIPLE) * height * width); | |
for (int row = 0; row < width; row++) { | |
for (int column = 0; column < height; column++) { | |
int sobelPixelPosition = 0; | |
int gxRed = 0; | |
int gxGreen = 0; | |
int gxBlue = 0; | |
int gyRed = 0; | |
int gyGreen = 0; | |
int gyBlue = 0; | |
for (int x = row - 1; x <= row + 1; x++) { | |
for (int y = column - 1; y <= column + 1; y++) { | |
if (x >= 0 && x < width && y >= 0 && y < height) { | |
RGBTRIPLE pixel = imageCopy[y][x]; | |
gxRed += getGxValue(sobelPixelPosition, pixel.rgbtRed); | |
gxGreen += getGxValue(sobelPixelPosition, pixel.rgbtGreen); | |
gxBlue += getGxValue(sobelPixelPosition, pixel.rgbtBlue); | |
gyRed += getGyValue(sobelPixelPosition, pixel.rgbtRed); | |
gyGreen += getGyValue(sobelPixelPosition, pixel.rgbtGreen); | |
gyBlue += getGyValue(sobelPixelPosition, pixel.rgbtBlue); | |
} else { | |
gxRed += getGxValue(sobelPixelPosition, 0); | |
gxGreen += getGxValue(sobelPixelPosition, 0); | |
gxBlue += getGxValue(sobelPixelPosition, 0); | |
gyRed += getGyValue(sobelPixelPosition, 0); | |
gyGreen += getGyValue(sobelPixelPosition, 0); | |
gyBlue += getGyValue(sobelPixelPosition, 0); | |
} | |
sobelPixelPosition++; | |
} | |
} | |
image[column][row].rgbtRed = getGxGyCombination(gxRed, gyRed); | |
image[column][row].rgbtGreen = getGxGyCombination(gxGreen, gyGreen); | |
image[column][row].rgbtBlue = getGxGyCombination(gxBlue, gyBlue); | |
sobelPixelPosition = 0; | |
} | |
} | |
return; | |
} | |
// Gets the apropriate sepia value | |
int resolveSepiaValue(float value) { | |
int _sepia = round(value); | |
if (_sepia > 255) { | |
_sepia = 255; | |
} | |
return _sepia; | |
} | |
int getGxValue(int sobelPixelPosition, int colorAmount) { | |
switch (sobelPixelPosition) { | |
case 0: | |
return colorAmount * -1; | |
case 1: | |
return colorAmount * 0; | |
case 2: | |
return colorAmount * 1; | |
case 3: | |
return colorAmount * -2; | |
case 4: | |
return colorAmount * 0; | |
case 5: | |
return colorAmount * 2; | |
case 6: | |
return colorAmount * -1; | |
case 7: | |
return colorAmount * 0; | |
case 8: | |
return colorAmount * 1; | |
default: | |
return 0; | |
} | |
} | |
int getGyValue(int sobelPixelPosition, int colorAmount) { | |
switch (sobelPixelPosition) { | |
case 0: | |
return colorAmount * -1; | |
case 1: | |
return colorAmount * -2; | |
case 2: | |
return colorAmount * -1; | |
case 3: | |
return colorAmount * 0; | |
case 4: | |
return colorAmount * 0; | |
case 5: | |
return colorAmount * 0; | |
case 6: | |
return colorAmount * 1; | |
case 7: | |
return colorAmount * 2; | |
case 8: | |
return colorAmount * 1; | |
default: | |
return 0; | |
} | |
} | |
int getGxGyCombination(int gx, int gy) { | |
int value = round(sqrt(pow(gx, 2) + pow(gy, 2))); | |
return value > 255 ? 255 : value; | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment