#include "segmentation.h"

#include "conversion_grayscale.h"
#include "blur_gaussian.h"

#include <queue>

Segmentation::Segmentation(PNM* img) :
    Transformation(img)
{
}

Segmentation::Segmentation(PNM* img, ImageViewer* iv) :
    Transformation(img, iv)
{
}


QPoint* Segmentation::neighbourhood(QPoint p)
{

    return new QPoint();
}

PNM* Segmentation::transform()
{
    int width = image->width();
    int height = image->height();

    PNM* newImage = new PNM(width, height, QImage::Format_Grayscale8);

    ConversionGrayscale* conversionGrayscale = new ConversionGrayscale(image);
    PNM* grayImage = conversionGrayscale->transform();

    // matrix with labels
    math::matrix<int> lab (width, height);
    // matrix wiht distances
    math::matrix<int> dist (width, height);

    int current_label = 0;
    std::queue<QPoint> fifo;

    // Fill matrixes with -1 for lab and 0 for dist
    for (int i = 0; i < width; i++)
    {
        for (int j = 0; j < height; j++)
        {
            lab[i][j] = -1;
            dist[i][j] = 0;
        }
    }

    // Create sorted list of gray pixels
    std::list<int> hSorted;
    for (int i = 0; i < width; i++)
    {
        for (int j = 0; j < height; j++)
        {
            hSorted.push_back(qGray(grayImage->pixel(i, j)));
        }
    }

    // Sort and remove dupliactes of pixels values in queue
    hSorted.sort();
    hSorted.unique();

    // Set current distance on 0
    int current_distance = 0;

    // Iterate over queue of gray pixels
    for (std::list<int>::iterator h = hSorted.begin(); h != hSorted.end(); h++)
    {
        // Iterate over image
        for (int p_i = 0; p_i < width; p_i++)
        {
            for (int p_j = 0; p_j < height; p_j++)
            {
                // Check if the grey pixel from the image is the same as the one from the queue
                if (qGray(image->pixel(p_i, p_j)) == *h)
                {
                    // If true set lab in this place on -2
                    lab[p_i][p_j] = -2;

                    // Check neighbours -1 0 1 left-right and up-down for p_i, p_j
                    for (int x = -1; x < 2; x++)
                    {
                        for (int y = -1; y < 2; y++)
                        {
                            // Check if is not center
                            if (x != 0 || y != 0)
                            {
                                // Create q_i and q_j
                                int q_i = p_i+x;
                                int q_j = p_j+y;

                                // Check that the values(q_i, q_j) are within the image dimensions
                                if (q_i >= 0 && q_j >= 0 && q_i < width && q_j < height)
                                {
                                    // Check point value lab[q_i][q_j] is greater than or equal to zero
                                    if (lab[q_i][q_j] >= 0)
                                    {
                                        // Initialize queue with neighbours at level h
                                        // of current basins or watersheds
                                        dist[p_i][p_j] = 1;
                                        fifo.push(QPoint(p_i, p_j));
                                    } //end if - Check point value lab[q_i][q_j] is greater than or equal to zero
                                } // end if - Check that the values(q_i, q_j) are within the image dimensions
                            } // end if - Check if is not center
                        } //end for
                    } // end for - Check neighbours -1 0 1 left-right and up-down for p_i, p_j
                } // end if - Check if the grey pixel from the image is the same as the one from the queue
            } // end for
        } //end for - Iterate over image

        // Set current distance on 1
        current_distance = 1;
        fifo.push(QPoint(-1, -1));

        // while loop
        while (true)
        {
            // Get first element and remove from queue
            QPoint p = fifo.front();
            fifo.pop();

            // Check if x and y equal -1
            if (p.x() == -1 && p.y() == -1)
            {
                // Check ff queue is empty - break loop
                if (fifo.empty())
                {
                    break;
                }
                else
                {
                    // Push to queue Point(-1, -1)
                    fifo.push(QPoint(-1, -1));

                    // Increment current distanece
                    current_distance += 1;

                    // Get first element and remove from queue
                    p = fifo.front();
                    fifo.pop();
                }
            }

            // Check neighbours -1 0 1 left-right and up-down for p_i, p_j
            for (int x = -1; x < 2; x++)
            {
                for (int y = -1; y < 2; y++)
                {
                    // Check if is not center
                    if (x != 0 || y != 0)
                    {
                        // Get current point x, y values and create q_i, q_j
                        int p_i = p.x();
                        int p_j = p.y();
                        int q_i = p_i + x;
                        int q_j = p_j + y;

                        // Check that the values(q_i, q_j) are within the image dimensions
                        if (q_i >= 0 && q_j >= 0 && q_i < width && q_j < height)
                        {
                            // Check distance value of point q_i, q_j is less than current distance and lab[q_i][q_j] is greater than or equal to zero
                            if (dist[q_i][q_j] < current_distance && lab[q_i][q_j] >= 0)
                            {
                                // Check if lab point for q_i and q_j is grather than 0
                                if (lab[q_i][q_j] > 0)
                                {
                                    if (lab[p_i][p_j] == -2 or lab[p_i][p_j] == 0)
                                    {
                                        lab[p_i][p_j] = lab[q_i][q_j];
                                    }
                                    else if (lab[p_i][p_j] != lab[q_i][q_j])
                                    {
                                        lab[p_i][p_j] = 0;
                                    }
                                }
                                else if (lab[p_i][p_j] == -2)
                                {
                                    lab[p_i][p_j] = 0;
                                } // end else - Check if lab point for q_i and q_j is grather than 0

                            }
                            else if (dist[q_i][q_j] == 0 && lab[q_i][q_j] == -2)
                            {
                                // The q is plateau pixel
                                dist[q_i][q_j] = current_distance + 1;
                                fifo.push(QPoint(q_i, q_j));
                            }
                        } // end if - Check that the values(q_i, q_j) are within the image dimensions
                    } // end if (x != 0 || y != 0)
                } // end for
            } // end for - Check neighbours -1 0 1 left-right and up-down for p_i, p_j
        } // end while

        // Iterate over image - detect and process new minimal at level h
        for (int p_i = 0; p_i < width; p_i++)
        {
            for (int p_j = 0; p_j < height; p_j++)
            {
                // Check if the grey pixel from the image is the same as the one from the queue
                if (qGray(image->pixel(p_i, p_j)) == *h)
                {
                    // Set distance to zero
                    dist[p_i][p_j] = 0;

                    // Check if lab[p_i][p_j] is -2
                    if (lab[p_i][p_j] == -2)
                    {
                        // Increment current lab
                        current_label += 1;
                        // Push point[p_i, p_j] to queue
                        fifo.push(QPoint(p_i, p_j));
                        // Set current lab value to lab[p_i][p_j]
                        lab[p_i][p_j] = current_label;

                        // while loop until the queue isn't empty
                        while (!fifo.empty())
                        {
                            QPoint q = fifo.front();
                            fifo.pop();

                            // Check neighbours -1 0 1 left-right and up-down for q_i, q_j
                            for (int x = -1; x < 2; x++)
                            {
                                for (int y = -1; y < 2; y++)
                                {
                                    if (x != 0 || y != 0)
                                    {
                                        int q_i = q.x();
                                        int q_j = q.y();
                                        int r_i = q_i + x;
                                        int r_j = q_j + y;
                                        if (r_i >= 0 && r_j >= 0 && r_i < width && r_j < height)
                                        {
                                            if (lab[r_i][r_j] == -2)
                                            {
                                                fifo.push(QPoint(r_i, r_j));
                                                lab[r_i][r_j] = current_label;
                                            } // end if set lab[r_i][r_j] == -2

                                        } // end if (r_i >= 0 && r_j >= 0 && r_i < width && r_j < height)
                                    } // end if (x != 0 || y != 0)
                                } // end for inspect neighbours of q
                            } // end for - Check neighbours -1 0 1 left-right and up-down for q_i, q_j
                        } // end while - while loop until the queue isn't empty
                    } // end if - Check if lab[p_i][p_j] is -2
                } // end if - Check if the grey pixel from the image is the same as the one from the queue
            } // end for
        } // end for - Iterate over image - detect and process new minimal at level h
    } // end for - Iterate over queue of gray pixels

    // Set very small value
    int max = -100;

    // Get max for lab points
    for (int i = 0; i < width; i++)
    {
        for (int j = 0; j < height; j++)
        {
            if (lab[i][j] > max)
            {
                max = lab[i][j];
            }
        }
    }
    // Protection against division by 0
    if (max == 0)
    {
        max = 1;
    }

    // Draw image
    for (int i = 0; i < width; i++)
    {
        for (int j = 0; j < height; j++)
        {
            double pixel = lab[i][j] * 255/max;
            double r = pixel * 0.3;
            double g = pixel * 0.6;
            double b = pixel * 0.1;
            QColor newPixel = QColor(r+g+b,r+g+b,r+g+b);
            newImage->setPixel(i, j, QColor(newPixel).rgb());
        }
    }
    return newImage;
}