Skip to content

Instantly share code, notes, and snippets.

@e200
Last active July 1, 2021 23:15
Show Gist options
  • Save e200/7c1756994d4d27ee3a7ddfae7d0ff8f7 to your computer and use it in GitHub Desktop.
Save e200/7c1756994d4d27ee3a7ddfae7d0ff8f7 to your computer and use it in GitHub Desktop.
#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