Friday, 30 January 2015

IMAGE INPAINTING

ImageSelection package

Entry.java

package imageselection;

import java.awt.event.*;
import javax.swing.*;
import java.awt.*;
import java.util.*;

/**
 * Class to enable selection of region to be inpainted
 */
public class Entry extends JPanel {

  protected Image entryImage;                       // Image to be displayed
  protected Graphics entryGraphics;                 // graphics object to the entry image
  protected int lastX = -1;                         // X co-ordinate of last pressed co-ordinate
  protected int lastY = -1;                         // Y co-ordinate of last pressed co-ordinate
  protected int polySides = 0;                      // number of sides of the polygon to be inpainted
  private Vector PolygonCoordinatesX;               // vector to store X co-ordinate of all vertices of a polygon
  private Vector PolygonCoordinatesY;               // vector to store Y co-ordinate of all vertices of a polygon
  private Image img;                                // instance of Image class
  public Stack SavedImages;                         // stack used to implement undo option
  public Stack RedoImages;                          // stack used to implement redo option
  private int firstX = -1;                          // variable to store first X co-ordinate of a polygon
  private int firstY = -1;                          // variable to store first Y co-ordinate of a polygon
  private int currX = -1;                           // variable to store current X co-ordinate of a polygon
  private int currY = -1;                           // variable to store current Y co-ordinate of a polygon
  public Boolean isDisabled = false;                // flag to determine status of the inpainting module
  public int maxX = -1;                             // stores maximum of all X co-ordinates of all polygon
  public int maxY = -1;                             // stores maximum of all Y co-ordinates of all polygon
  public int minX = -1;                             // stores minimum of all X co-ordinates of all polygon
  public int minY = -1;                             // stores minimum of all Y co-ordinates of all polygon

  Boolean pressed;                                  // Defines whether the polygon is started or not

  /**
   * constructor of Entry class
   * @param img default image to display
   */
  Entry(Image img)
  {
    entryImage = img;
    enableEvents(AWTEvent.MOUSE_MOTION_EVENT_MASK|
                 AWTEvent.MOUSE_EVENT_MASK|
                 AWTEvent.COMPONENT_EVENT_MASK);
    pressed = false;
    SavedImages = new Stack();
    RedoImages = new Stack();
    PolygonCoordinatesX = new Vector();
    PolygonCoordinatesY = new Vector();
  }
  /**
   * method to show updated image
   * @param img image to display
   */
  public void showImage(Image img)
  {
      entryImage = img;
      entryGraphics = entryImage.getGraphics();
      repaint();       
  }

  /**
   * method to get entryimage
   * @return entryimage
   */
  public Image getImage()
  {
      return entryImage;
  }
 
    @SuppressWarnings("unchecked")
  protected void initImage()
  {
        img = entryImage;
        entryGraphics = entryImage.getGraphics();
        Image tmg = createImage(entryImage.getWidth(this),entryImage.getHeight(this));
        Graphics tg = tmg.getGraphics();
    tg.drawImage(entryImage,0,0,null);
        SavedImages.push(tmg);
        repaint();
  }
  
    @Override
  public void paint(Graphics g)
  {
    if ( entryImage==null )
      initImage();
    g.drawImage(entryImage,0,0,this);

    /**
     * Draw red rectangle on first selected co-ordinate of a polygon
     */
    if (firstX != -1) {
        g.setColor(Color.red);
        g.drawRect(firstX-5, firstY-5, 10, 10);
        g.drawRect(firstX-4, firstY-4, 8, 8);
    }

    g.setColor(Color.black);
    g.drawRect(0,0,getWidth(),getHeight());
    g.setColor(Color.green);
   
    if (PolygonCoordinatesX == null || PolygonCoordinatesY == null) {
        return;
    }

    /**
     * Draw black rectangle on every other co-ordinate of a polygon except the first co-ordinate
     */
    g.setColor(Color.black);
    for (int i = 1; i < PolygonCoordinatesX.size(); ++i) {
        g.drawRect((Integer)PolygonCoordinatesX.get(i)-5, (Integer)PolygonCoordinatesY.get(i)-5, 10, 10);
        g.drawRect((Integer)PolygonCoordinatesX.get(i)-4, (Integer)PolygonCoordinatesY.get(i)-4, 8, 8);
    }
    g.setColor(Color.green);

    if (lastX != -1) {
        g.drawLine(lastX, lastY, currX, currY);             // draw line between successive co-ordinates
    }
  }

  @Override
    @SuppressWarnings("unchecked")
  protected void processMouseEvent(MouseEvent e)
  {
      if (!isDisabled) {
        if ( e.getID()==MouseEvent.MOUSE_PRESSED && !pressed) {
            lastX = e.getX();
            lastY = e.getY();
            firstX = lastX;
            firstY = lastY;
            pressed = true;
            PolygonCoordinatesX.add(lastX);
            PolygonCoordinatesY.add(lastY);
            repaint();
        } else if (e.getID() == MouseEvent.MOUSE_PRESSED && pressed) {
            entryGraphics.setColor(Color.green);
            entryGraphics.drawLine(lastX,lastY,e.getX(),e.getY());
            getGraphics().drawImage(entryImage,0,0,this);
            repaint();
            lastX = e.getX();
            lastY = e.getY();
            maxX = maxY = -1;
            minX = lastX;
            minY = lastY;
            PolygonCoordinatesX.add(lastX);
            PolygonCoordinatesY.add(lastY);
            if (Math.abs(lastX - (Integer)PolygonCoordinatesX.get(0)) < 10 && Math.abs(lastY - (Integer)PolygonCoordinatesY.get(0)) < 10) {
                int[] PolyX = new int[PolygonCoordinatesX.size()];
                int[] PolyY = new int[PolygonCoordinatesY.size()];
                for (int i = 0; i < PolygonCoordinatesX.size(); ++i) {
                    PolyX[i] = (Integer)PolygonCoordinatesX.get(i);
                    PolyY[i] = (Integer)PolygonCoordinatesY.get(i);

                    if (minX > PolyX[i]) {
                        minX = PolyX[i];
                    }
                    if (minY > PolyY[i]) {
                        minY = PolyY[i];
                    }
                    if (maxX < PolyX[i]) {
                        maxX = PolyX[i];
                    }
                    if (maxY < PolyY[i]) {
                        maxY = PolyY[i];
                    }
                    System.out.println(PolygonCoordinatesX.get(i) + ", " + PolygonCoordinatesY.get(i));
                }
                polySides = PolygonCoordinatesX.size();
                entryGraphics.fillPolygon(PolyX, PolyY, polySides);
                PolygonCoordinatesX.clear();
                PolygonCoordinatesY.clear();
                currX = currY = -1;
                lastX = lastY = -1;
                firstX = firstY = -1;
                pressed = false;
                Image tmg = createImage(entryImage.getWidth(this),entryImage.getHeight(this));
                Graphics tg = tmg.getGraphics();
                tg.drawImage(entryImage,0,0,null);
                //UndoImages.add(tmg);
                SavedImages.push(tmg);
                RedoImages.clear();
                repaint();
            }
        } else {
            return;
        }
      }
  }
 
    @Override
  protected void processMouseMotionEvent(MouseEvent e)
  {
        if (!isDisabled) {
            if ( e.getID()!=MouseEvent.MOUSE_MOVED )
              return;

            currX = e.getX();
            currY = e.getY();
            repaint();
        }
  }

    void setDisabled()
    {
        isDisabled = true;
    }

    void setEnabled()
    {
        isDisabled = false;
    }

    public Boolean getPressed() {
        return pressed;
    }

    void entryReset() {
        PolygonCoordinatesX.clear();
        PolygonCoordinatesY.clear();
        currX = currY = -1;
        lastX = lastY = -1;
        firstX = firstY = -1;
        pressed = false;
    }
   
    Object entryImage() {
        throw new UnsupportedOperationException("Not yet implemented");
    }
   
}



GradientCalculator.java

package imageselection;

import java.util.*;

/**
 *
 */
public class GradientCalculator {

    public double[][] gradientX;        // Variable to communicate the gradient in X direction with calling class
    public double[][] gradientY;        // Variable to communicate the gradient in X direction with calling class
    int r, g, b;                        // temporary variables to store the r, g and b values of the pixels.

    /**
     * Function that calculates the gradient from the image given in argument using the central difference method.
     * @param pixelmap  Pixelmap of the image
     * @param ih    Height of the image
     * @param iw    Width of the image
     */
    public void calculateGradientFromImage (int pixelmap[][], int ih, int iw)
    {
        gradientX = new double[ih][iw];
        gradientY = new double[ih][iw];

        int[] row = new int[ih];
        int[] column = new int[iw];

        for (int i = 0; i < ih; ++i) {
            row[i] = i+1;
        }

        for (int i = 0; i < iw; ++i) {
            column[i] = i+1;
        }

        double t1, t2, t3;
        if (ih > 1) {
            /**
             * Calculate gradient for boundary pixels.
             */
            for (int j = 0; j < iw; ++j) {
                extractRGB(pixelmap, 1, j);
                int r2 = r; int g2 = g; int b2 = b;

                extractRGB(pixelmap, 0, j);
                int r1 = r; int g1 = g; int b1 = b;

                t1 = (double)(r2-r1)/(row[1]-row[0]);
                t2 = (double)(g2-g1)/(row[1]-row[0]);
                t3 = (double)(b2-b1)/(row[1]-row[0]);
                gradientX[0][j] = -((double)t1+t2+t3)/(3.0*255.0);

                extractRGB(pixelmap, ih-1, j);
                int rn = r; int gn = g; int bn = b;

                extractRGB(pixelmap, ih-2, j);
                int rn1 = r; int gn1 = g; int bn1 = b;

                t1 = (double)(rn-rn1)/(row[ih-1]-row[ih-2]);
                t2 = (double)(gn-gn1)/(row[ih-1]-row[ih-2]);
                t3 = (double)(bn-bn1)/(row[ih-1]-row[ih-2]);
                gradientX[ih-1][j] = -((double)t1+t2+t3)/(3.0*255.0);
            }
        }

        if (ih > 2) {
            /**
             * Calculate the gradient for central pixels.
             */
            for (int i = 1; i < ih-1; ++i) {
                for (int j = 0; j < iw; ++j) {
                    extractRGB(pixelmap, i+1, j);
                    int r2 = r; int g2 = g; int b2 = b;

                    extractRGB(pixelmap, i-1, j);
                    int r1 = r; int g1 = g; int b1 = b;

                    t1 = (double)(r2-r1)/(row[i+1]-row[i-1]);
                    t2 = (double)(g2-g1)/(row[i+1]-row[i-1]);
                    t3 = (double)(b2-b1)/(row[i+1]-row[i-1]);
                    gradientX[i][j] = -((double)t1+t2+t3)/(3.0*255.0);
                }
            }
        }

        if (iw > 1) {
            /**
             * Calculate gradient for boundary pixels.
             */
            for (int j = 0; j < ih; ++j) {
                extractRGB(pixelmap, j, 1);
                int r2 = r; int g2 = g; int b2 = b;

                extractRGB(pixelmap, j, 0);
                int r1 = r; int g1 = g; int b1 = b;

                t1 = (double)(r2-r1)/(column[1]-column[0]);
                t2 = (double)(g2-g1)/(column[1]-column[0]);
                t3 = (double)(b2-b1)/(column[1]-column[0]);
                gradientY[j][0] = (double)(t1+t2+t3)/(3.0*255.0);

                extractRGB(pixelmap, j, iw-1);
                int rn = r; int gn = g; int bn = b;

                extractRGB(pixelmap, j, iw-2);
                int rn1 = r; int gn1 = g; int bn1 = b;
               
                t1 = (double)(rn-rn1)/(column[iw-1]-column[iw-2]);
                t2 = (double)(gn-gn1)/(column[iw-1]-column[iw-2]);
                t3 = (double)(bn-bn1)/(column[iw-1]-column[iw-2]);
                gradientY[j][iw-1] = (double)(t1+t2+t3)/(3.0*255.0);
            }
        }

        if (iw > 2) {
            /**
             * Calculate gradient for central pixels.
             */
            for (int i = 0; i < ih; ++i) {
                for (int j = 1; j < iw-1; ++j) {

                    extractRGB(pixelmap, i, j+1);
                    int r2 = r; int g2 = g; int b2 = b;

                    extractRGB(pixelmap, i, j-1);
                    int r1 = r; int g1 = g; int b1 = b;

                    t1 = (double)(r2-r1)/(column[j+1]-column[j-1]);//g(2:n-1,:) = (f(3:n,:)-f(1:n-2,:))./h(:,ones(p,1));
                    t2 = (double)(g2-g1)/(column[j+1]-column[j-1]);
                    t3 = (double)(b2-b1)/(column[j+1]-column[j-1]);
                    gradientY[i][j] = (double)(t1+t2+t3)/(3.0*255.0);
                }
            }
        }
    }

    /**
     * Function to calculate gradient for other matrices
     * @param a The matrix for which gradient is to be calculated
     * @param ih    Height of the matrix
     * @param iw    Width of the matrix
     */
    public void calculateGradient (int a[][], int ih, int iw)
    {
        gradientX = new double[ih][iw];
        gradientY = new double[ih][iw];
        Vector h = new Vector();
        int[] row = new int[ih];
        int[] column = new int[iw];

        for (int i = 0; i < ih; ++i) {
            row[i] = i+1;
        }

        for (int i = 0; i < iw; ++i) {
            column[i] = i+1;
        }

        if (ih > 1) {
            for (int j = 0; j < iw; ++j) {
                gradientY[0][j] = (double)(a[1][j] - a[0][j])/(row[1]-row[0]);
                gradientY[ih-1][j] = (double)(a[ih-1][j] - a[ih-2][j])/(row[ih-1]-row[ih-2]);
            }
        }

        if (ih > 2) {
            for (int i = 1; i < ih-1; ++i) {
                for (int j = 0; j < iw; ++j) {
                    gradientY[i][j] = (double)(a[i+1][j] - a[i-1][j])/(row[i+1]-row[i-1]);
                }
            }
        }

         if (iw > 1) {
            for (int j = 0; j < ih; ++j) {
                gradientX[j][0] = (double)(a[j][1] - a[j][0])/(column[1]-column[0]);
                gradientX[j][iw-1] = (double)(a[j][iw-1] - a[j][iw-2])/(column[iw-1]-column[iw-2]);
            }
        }

        if (iw > 2) {
            for (int i = 0; i < ih; ++i) {
                for (int j = 1; j < iw-1; ++j) {
                    gradientX[i][j] = (double)(a[i][j+1] - a[i][j-1])/(column[j+1]-column[j-1]);
                }
            }
        }
    }

    /**
     * Function to calculate RGB values of the pixel
     * @param img
     * @param i
     * @param j
     */
    void extractRGB(int img[][], int i, int j)
    {
        r = 0xff & (img[i][j] >> 16);
        g = 0xff & (img[i][j] >> 8);
        b = 0xff & (img[i][j]);
    }
}


Main.java

package imageselection;

import com.permission.Display;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import javax.swing.*;
import javax.swing.UIManager;
import java.awt.*;
import java.awt.event.ComponentAdapter;
import java.awt.event.ComponentEvent;
import java.awt.image.*;
import java.io.File;
import java.io.IOException;
import java.util.logging.Level;
import java.util.logging.Logger;
import javax.imageio.ImageIO;
import inpaint.*;

/**
 * Class to make UI and communicate the ImageInpainting module
 * @author Pulkit & Sapan
 */
public class Main extends JFrame implements Runnable{
   
    public Entry entry;                             // Defines the object of Entry class
    protected Image entryImage;                     // variable of type Image
    JScrollPane pictureScrollPane;                  // Scrollpane for scrolling Images(whenever required)
    File outputFile;                                // file used while saving the image
    String fileExtension;                           // string to store the extension of a image
    ImageInpaint Inpainter;                         // Defines the object of ImageInpaint class
    Thread inpaintThread = null;                    // thread for the inpaint process
    private Boolean fastInpaint = false;            // flag to determine fastInpaint
    //Button playbtn;

    private javax.swing.JButton jButton1;
    private javax.swing.JButton jButton2;
    private javax.swing.JButton jButton3;
    private javax.swing.JButton jButton4;
    private javax.swing.JButton jButton6;
    private javax.swing.JButton jButton7;
    private javax.swing.JFrame jFrame5;
    private javax.swing.JFrame jFrame6;
    private javax.swing.JFrame jFrame7;
    private javax.swing.JToolBar jToolBar1;
   
    @SuppressWarnings("deprecation")
    Main()
    {
        Inpainter = new ImageInpaint(this);         // Creates an instance of the ImageInpaint class

        /**
         * sets look and feel for the UI
         */
        try {
            UIManager.setLookAndFeel(UIManager.getSystemLookAndFeelClassName());
        } catch (Exception e) {
            e.printStackTrace();
        }

        /**
         * Action to be performed on clicking the close button
         */
        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);
        initComponents();

        /**
         * Reads the image to be inpainted.
         */
        try {
            entryImage = ImageIO.read(getClass().getResource("/imageselection/Images/defaultImage.png"));
        } catch (IOException ex) {
            Logger.getLogger(Entry.class.getName()).log(Level.SEVERE, null, ex);
        }
       
        setTitle("Image Inpainting");                         // Set the title of the Window
        getContentPane().setLayout(null);
        setSize(500,500);                                     //Set Window Width and Height
      
        entry = new Entry(entryImage);                        // creates an instance of the entry class
       
        entry.setPreferredSize(new Dimension(entryImage.getWidth(this),entryImage.getHeight(this)));
       
        getContentPane().add(entry);
        entry.initImage(); 
        pictureScrollPane = new JScrollPane();                // creates an instance of the JScrollPane class
        getContentPane().add(pictureScrollPane);
        int w = Math.min(entryImage.getWidth(this)+3,getContentPane().getWidth());
        int h = Math.min(entryImage.getHeight(this)+3,getContentPane().getHeight()-50);
        if (h == entryImage.getHeight(this)+3) {
            pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
        } else {
            pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h+25);
        }
       
        pictureScrollPane.setAlignmentY(CENTER_ALIGNMENT);
        pictureScrollPane.setVerticalScrollBarPolicy(JScrollPane.VERTICAL_SCROLLBAR_AS_NEEDED);
        pictureScrollPane.setHorizontalScrollBarPolicy(JScrollPane.HORIZONTAL_SCROLLBAR_AS_NEEDED);
        pictureScrollPane.setViewportView(entry);
       
        this.addComponentListener(new ComponentAdapter() {
            @Override
            public void componentResized(ComponentEvent e) {
                int w = Math.min(entryImage.getWidth(entry)+3,getContentPane().getWidth());
                int h = Math.min(entryImage.getHeight(entry)+3,getContentPane().getHeight()-50);
                if (h == entryImage.getHeight(entry)+3) {
                    pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
                } else {
                    pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h+25);
                }
         //       pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
                pictureScrollPane.setViewportView(entry);
            }
        });
    }

    /**
     * main function execution starts from here
     * @param args command line argument
     */
    @SuppressWarnings("deprecation")
    public static void main(String[] args) {
        (new Main()).show();
    }
   
     private void initComponents() {
        JFrame jFrame1 = new javax.swing.JFrame();
        JFrame jFrame2 = new javax.swing.JFrame();
        JFrame jFrame3 = new javax.swing.JFrame();
        JFrame jFrame4 = new javax.swing.JFrame();



        JMenuBar jMenuBar1 = new javax.swing.JMenuBar();
       
        JMenu jMenu1 = new javax.swing.JMenu();
        JMenu jMenu2 = new javax.swing.JMenu();
        JMenu jMenu3 = new javax.swing.JMenu();
        JMenu jMenu4 = new javax.swing.JMenu();
       
        JMenuItem jMenuItem1 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem2 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem3 = new javax.swing.JMenuItem();       
        JMenuItem jMenuItem4 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem5 = new javax.swing.JMenuItem();       
        JMenuItem jMenuItem6 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem7 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem8 = new javax.swing.JMenuItem();
        JMenuItem jMenuItem9 = new javax.swing.JMenuItem();

        javax.swing.GroupLayout jFrame1Layout = new javax.swing.GroupLayout(jFrame1.getContentPane());
        jFrame1.getContentPane().setLayout(jFrame1Layout);
        jFrame1Layout.setHorizontalGroup(
            jFrame1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame1Layout.setVerticalGroup(
            jFrame1Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        javax.swing.GroupLayout jFrame2Layout = new javax.swing.GroupLayout(jFrame2.getContentPane());
        jFrame2.getContentPane().setLayout(jFrame2Layout);
        jFrame2Layout.setHorizontalGroup(
            jFrame2Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame2Layout.setVerticalGroup(
            jFrame2Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        javax.swing.GroupLayout jFrame3Layout = new javax.swing.GroupLayout(jFrame3.getContentPane());
        jFrame3.getContentPane().setLayout(jFrame3Layout);
        jFrame3Layout.setHorizontalGroup(
            jFrame3Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame3Layout.setVerticalGroup(
            jFrame3Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        javax.swing.GroupLayout jFrame4Layout = new javax.swing.GroupLayout(jFrame3.getContentPane());
        jFrame4.getContentPane().setLayout(jFrame4Layout);
        jFrame4Layout.setHorizontalGroup(
            jFrame4Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame4Layout.setVerticalGroup(
            jFrame4Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);

        jMenu1.setText("File");

        jMenuItem1.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_O, java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem1.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/open.png")));
        jMenuItem1.setText("Open Image");
        jMenu1.add(jMenuItem1);
        jMenuItem1.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem1ActionPerformed(evt);
            }
        });

        jMenuItem2.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_S, java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem2.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/save.png")));
        jMenuItem2.setText("Save");
        jMenu1.add(jMenuItem2);
        jMenuItem2.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem2ActionPerformed(evt);
            }
        });

        jMenuItem7.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_S, java.awt.event.InputEvent.ALT_MASK | java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem7.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/saveAs.png")));
        jMenuItem7.setText("Save As");
        jMenu1.add(jMenuItem7);
        jMenuItem7.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem7ActionPerformed(evt);
            }
        });
       
        jMenuItem3.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_F4, java.awt.event.InputEvent.ALT_MASK));
        jMenuItem3.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/exit.png")));
        jMenuItem3.setText("Exit");
        jMenu1.add(jMenuItem3);
        jMenuItem3.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem3ActionPerformed(evt);
            }
        });

        jMenuBar1.add(jMenu1);

        jMenu2.setText("Edit");

        jMenuItem4.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_Z, java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem4.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/undo.png")));
        jMenuItem4.setText("Undo");
        jMenu2.add(jMenuItem4);
        jMenuItem4.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem4ActionPerformed(evt);
            }
        });

        jMenuItem5.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_Y, java.awt.event.InputEvent.CTRL_MASK));
        jMenuItem5.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/redo.png")));
        jMenuItem5.setText("Redo");
        jMenu2.add(jMenuItem5);
        jMenuItem5.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem5ActionPerformed(evt);
            }
        });

        jMenuBar1.add(jMenu2);


        jMenu3.setText("Inpaint");

        jMenuItem8.setAccelerator(javax.swing.KeyStroke.getKeyStroke(java.awt.event.KeyEvent.VK_R, java.awt.event.InputEvent.CTRL_MASK));
        //jMenuItem8.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/open.png")));
        jMenuItem8.setText("Run");
        jMenu3.add(jMenuItem8);
        jMenuItem8.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem8ActionPerformed(evt);
            }
        });

        jMenu4.setText("Help");

     
       
       
        jMenuBar1.add(jMenu3);

        jMenuBar1.add(jMenu4);

        setJMenuBar(jMenuBar1);

        javax.swing.GroupLayout layout = new javax.swing.GroupLayout(getContentPane());
        getContentPane().setLayout(layout);
        layout.setHorizontalGroup(
            layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        layout.setVerticalGroup(
            layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 279, Short.MAX_VALUE)
        );

        jFrame5 = new javax.swing.JFrame();
        jFrame6 = new javax.swing.JFrame();
        jFrame7 = new javax.swing.JFrame();
        jToolBar1 = new javax.swing.JToolBar();
        jButton1 = new javax.swing.JButton();
        jButton2 = new javax.swing.JButton();
        jButton3 = new javax.swing.JButton();
        jButton4 = new javax.swing.JButton();
        jButton6 = new javax.swing.JButton();
        jButton7 = new javax.swing.JButton();

        javax.swing.GroupLayout jFrame5Layout = new javax.swing.GroupLayout(jFrame5.getContentPane());
        jFrame5.getContentPane().setLayout(jFrame5Layout);
        jFrame5Layout.setHorizontalGroup(
            jFrame5Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame5Layout.setVerticalGroup(
            jFrame5Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        javax.swing.GroupLayout jFrame6Layout = new javax.swing.GroupLayout(jFrame6.getContentPane());
        jFrame6.getContentPane().setLayout(jFrame6Layout);
        jFrame6Layout.setHorizontalGroup(
            jFrame6Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame6Layout.setVerticalGroup(
            jFrame6Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        javax.swing.GroupLayout jFrame7Layout = new javax.swing.GroupLayout(jFrame7.getContentPane());
        jFrame7.getContentPane().setLayout(jFrame7Layout);
        jFrame7Layout.setHorizontalGroup(
            jFrame7Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 400, Short.MAX_VALUE)
        );
        jFrame7Layout.setVerticalGroup(
            jFrame7Layout.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGap(0, 300, Short.MAX_VALUE)
        );

        setDefaultCloseOperation(javax.swing.WindowConstants.EXIT_ON_CLOSE);

        jToolBar1.setRollover(true);

        jButton1.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/open.png"))); // NOI18N
        jButton1.setFocusable(false);
        jButton1.setHorizontalTextPosition(javax.swing.SwingConstants.CENTER);
        jButton1.setVerticalTextPosition(javax.swing.SwingConstants.BOTTOM);
        jToolBar1.add(jButton1);
        jButton1.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem1ActionPerformed(evt);
            }
        });

        jButton2.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/save.png"))); // NOI18N
        jButton2.setFocusable(false);
        jButton2.setHorizontalTextPosition(javax.swing.SwingConstants.CENTER);
        jButton2.setVerticalTextPosition(javax.swing.SwingConstants.BOTTOM);
        jToolBar1.add(jButton2);
        jButton2.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem2ActionPerformed(evt);
            }
        });

        jButton3.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/undo.png"))); // NOI18N
        jButton3.setFocusable(false);
        jButton3.setHorizontalTextPosition(javax.swing.SwingConstants.CENTER);
        jButton3.setVerticalTextPosition(javax.swing.SwingConstants.BOTTOM);
        jToolBar1.add(jButton3);
        jButton3.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem4ActionPerformed(evt);
            }
        });

        jButton4.setIcon(new javax.swing.ImageIcon(getClass().getResource("/imageselection/Images/redo.png"))); // NOI18N
        jButton4.setFocusable(false);
        jButton4.setHorizontalTextPosition(javax.swing.SwingConstants.CENTER);
        jButton4.setVerticalTextPosition(javax.swing.SwingConstants.BOTTOM);
        jToolBar1.add(jButton4);
        jButton4.addActionListener(new java.awt.event.ActionListener() {
            public void actionPerformed(java.awt.event.ActionEvent evt) {
                jMenuItem5ActionPerformed(evt);
            }
        });


        javax.swing.GroupLayout layout1 = new javax.swing.GroupLayout(getContentPane());
        getContentPane().setLayout(layout1);
        layout1.setHorizontalGroup(
            layout1.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addComponent(jToolBar1, javax.swing.GroupLayout.DEFAULT_SIZE, 400, Short.MAX_VALUE)
        );
        layout1.setVerticalGroup(
            layout1.createParallelGroup(javax.swing.GroupLayout.Alignment.LEADING)
            .addGroup(layout1.createSequentialGroup()
                .addComponent(jToolBar1, javax.swing.GroupLayout.PREFERRED_SIZE, 25, javax.swing.GroupLayout.PREFERRED_SIZE)
                .addContainerGap(275, Short.MAX_VALUE))
        );

        pack();
    }

     /**
      * action performed for opening file
      * @param evt
      */
     private void jMenuItem1ActionPerformed(ActionEvent evt) {

         if (!entry.isDisabled) {

             BufferedImage selectedImage;

             JFileChooser _fileChooser = new JFileChooser();
             int retval = _fileChooser.showOpenDialog(Main.this);

             /**
              * extensions of images user is allowed to choose
              */
             final String[] okFileExtensions = new String[] {"jpg", "png", "gif", "bmp", "jpeg"};
             File file;

             if (retval == JFileChooser.APPROVE_OPTION) {
                try {
                    file = _fileChooser.getSelectedFile();
                    Boolean flag = false;
                    for (String extension : okFileExtensions) {
                        if (file.getName().toLowerCase().endsWith(extension)) {
                            outputFile = file;
                            fileExtension = extension;
                            flag = true;
                        }
                    }
                    if (!flag) {
                        JOptionPane.showMessageDialog(this, "Please choose a jpg, jpeg, png, bmp or gif file only.","Error",JOptionPane.ERROR_MESSAGE);
                        return;
                    }

                    entry.SavedImages.clear();
                    entry.RedoImages.clear();
                    selectedImage = ImageIO.read(file);
                    Image tmg = createImage(((Image)selectedImage).getWidth(this),((Image)selectedImage).getHeight(this));
                    Graphics tg = tmg.getGraphics();
                    tg.drawImage((Image)selectedImage,0,0,null);
                    entry.SavedImages.push(selectedImage);
                    entryImage = tmg;
                    entry.showImage(entryImage);
                    entry.setPreferredSize(new Dimension(entryImage.getWidth(this),entryImage.getHeight(this)));
                    int w = Math.min(entryImage.getWidth(this)+3,getContentPane().getWidth());
                    int h = Math.min(entryImage.getHeight(this)+3,getContentPane().getHeight());
                    pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
                    pictureScrollPane.setViewportView(entry);
                } catch (IOException ex) {
                    Logger.getLogger(Main.class.getName()).log(Level.SEVERE, null, ex);
                }
          }
         }
     }

     /**
      * action performed for undo event
      * @param evt
      */
    @SuppressWarnings("unchecked")
      private void jMenuItem4ActionPerformed(ActionEvent evt)
      {
        Boolean flag = false;

         if ((!entry.isDisabled) && (entry.SavedImages.size()>1)) {
            if (entry.getPressed()) {
                entry.entryReset();
                flag = true;
            }
            entry.RedoImages.push(entry.SavedImages.pop());
            Image tmg = createImage(((Image)entry.SavedImages.peek()).getWidth(this),((Image)entry.SavedImages.peek()).getHeight(this));
            Graphics tg = tmg.getGraphics();
            tg.drawImage((Image)entry.SavedImages.peek(),0,0,null);
            entry.showImage(tmg);
           
            entry.setPreferredSize(new Dimension(entryImage.getWidth(this),entryImage.getHeight(this)));
            int w = Math.min(entryImage.getWidth(this)+3,getContentPane().getWidth());
            int h = Math.min(entryImage.getHeight(this)+3,getContentPane().getHeight());
            pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
            pictureScrollPane.setViewportView(entry);

            if (flag) {
                jMenuItem5ActionPerformed(evt);
            }
         }
      }

    /**
      * action performed for redo event
      * @param evt
      */
    @SuppressWarnings("unchecked")
     private void jMenuItem5ActionPerformed(ActionEvent evt)
     {
         if ((!entry.isDisabled) && (entry.RedoImages.size()>0)) {
           
            Image tmg = createImage(((Image)entry.RedoImages.peek()).getWidth(this),((Image)entry.RedoImages.peek()).getHeight(this));
            Graphics tg = tmg.getGraphics();
            tg.drawImage((Image)entry.RedoImages.peek(),0,0,null);
            entry.showImage(tmg);
            entry.SavedImages.push(entry.RedoImages.pop());

            entry.setPreferredSize(new Dimension(entryImage.getWidth(this),entryImage.getHeight(this)));
            int w = Math.min(entryImage.getWidth(this)+3,getContentPane().getWidth());
            int h = Math.min(entryImage.getHeight(this)+3,getContentPane().getHeight());
            pictureScrollPane.setBounds((getContentPane().getWidth()-w)/2,(getContentPane().getHeight()-h)/2, w, h);
            pictureScrollPane.setViewportView(entry);
         }
      }
  

    /**
      * action performed for saving image
      * @param evt
      */
     private void jMenuItem2ActionPerformed(ActionEvent evt)
     {        
         if (outputFile == null) {
             System.err.println("Error!! No file to save");
             return;
         }
        
         try {
             BufferedImage bi = (BufferedImage)entry.getImage();
             ImageIO.write(bi, fileExtension, outputFile);
         } catch (IOException e) {
             System.err.println("Error!! File not saved");
         }
     }

     /**
      * action performed when user wants to save image with different name
      * @param evt
      */
     @SuppressWarnings("unchecked")
      private void jMenuItem7ActionPerformed(ActionEvent evt)
      {        
         JFileChooser _fileChooser = new JFileChooser();
         int retval = _fileChooser.showSaveDialog(Main.this);

         final String[] okFileExtensions = new String[] {"jpg", "png", "gif", "bmp", "jpeg"};
         File file;

         if (retval == JFileChooser.APPROVE_OPTION) {            
            file = _fileChooser.getSelectedFile();
            Boolean flag = false;
            for (String extension : okFileExtensions) {
                if (file.getName().toLowerCase().endsWith(extension)) {
                    if (outputFile == null) {                           
                        System.err.println("Error!! No file to save");
                        return;
                    }
                    try {
                        outputFile = file;
                        fileExtension = extension;
                        BufferedImage bi = (BufferedImage)entry.getImage();
                        ImageIO.write(bi, fileExtension, outputFile);
                        System.out.println("Saved");
                    } catch (IOException e) {
                        System.err.println("Error!! File not saved");
                    }


                    flag = true;
                }
            }
            if (!flag) {
                JOptionPane.showMessageDialog(this, "Please choose a jpg, jpeg, png, bmp or gif file only.","Error",JOptionPane.ERROR_MESSAGE);
                return;
            }
        }
      
      }

     /**
      * action performed for inpaint
      * @param evt
      */
     private void jMenuItem8ActionPerformed(ActionEvent evt) {
        if(Display.diplaay()==true){
         Inpainter.halt = false;
         Inpainter.completed = false;
          if ( inpaintThread==null ) {
              entry.setDisabled();
              inpaintThread = new Thread(this);
              inpaintThread.start();
          }
        }else{
            JOptionPane.showMessageDialog(this, "Can't Run");
        }
     }

     /**
      * action performed for exit
      * @param evt
      */
     private void jMenuItem3ActionPerformed(ActionEvent evt)
     {
               System.exit(0);
     }
    
     class SymAction implements java.awt.event.ActionListener {
         public void actionPerformed(java.awt.event.ActionEvent event)
         {
             Object object = event.getSource();            
         }
     }

     /**
      * calls the method to start inpainting
      */
    public void run()
    {
        Inpainter.init((BufferedImage)entry.getImage(), (BufferedImage)entry.getImage(),fastInpaint);
    }

    /**
     * Method to communicate with the Inpainting module
     * @param toShow updated image
     */
    public void updateStats(BufferedImage toShow)
    {
        UpdateStats stats = new UpdateStats();
        stats.toShow = toShow;
        try {
            SwingUtilities.invokeAndWait(stats);
        } catch ( Exception e ) {
            JOptionPane.showMessageDialog(this,"Error: " + e,"Training",
            JOptionPane.ERROR_MESSAGE);
        }


        if (Inpainter.completed) {                               // Inpainting completes
           
            JOptionPane.showMessageDialog(this,
                                    "                      Inpainting is completed.","Inpainting",
                                    JOptionPane.PLAIN_MESSAGE);
        }
        if (Inpainter.completed || Inpainter.halt) {
            System.out.println ("Inpainting completed or halted");
            inpaintThread = null;
            Image tmg = createImage(((Image)toShow).getWidth(this),((Image)toShow).getHeight(this));
            Graphics tg = tmg.getGraphics();
            tg.drawImage((Image)toShow,0,0,null);
            entry.SavedImages.push(tmg);
            entry.setEnabled();
            entry.RedoImages.clear();
            fastInpaint = false;
        }
    }

    public class UpdateStats implements Runnable {

        BufferedImage toShow;

        public void run()
        {
            entry.showImage(toShow);
        }
    }
}

PACKAGE 

INPAINT

IMAGEINPAINT.JAVA

package inpaint;

import imageselection.GradientCalculator;
import java.util.*;                     // Packages for basic Java utilities, e.g. Vector etc.
import java.awt.*;                      // Packages for Java Images
import java.awt.image.*;                // Packages for Image processing in Java
import java.awt.image.PixelGrabber;     // Package for grabbing Pixels from the Image
import imageselection.Main;             // Application's Main class


/**
 * Class to Inpaint the given Image.
 * Call Function init with original image, fillImage and quickPaint as arguments
 * @author Sapan & Pulkit
 */


public class ImageInpaint {
    BufferedImage origImg;              // BufferedImage Object to represent updated image at every step
    Image fillImg;                      // Image Object to represent Image with target region marked
    BufferedImage img;                  // BufferedImage Object to represent original Image
    WritableRaster raster;              // Raster to write to the image
    int iw, ih;                         // iw: Width of Image, ih: Height of Image
    int pixels[];                       // Temporary array that will initially store the grabbed pixels
    int pixelmap[][];                   // Matrix to store the pixels of original image
    int fillPixelmap[][];               // Matrix to store the pixels of image marked with target region
    int sourceRegion[][];               // Matrix to store Boolean values for Source Region (is updated after each iteration)
    int initialSourceRegion[][];        // Matrix to store Boolean values for Initial Source Region
    double fillRegion[][];              // Matrix to store Boolean values for Target Region
    double gradientX[][];               // Matrix to represent the perpendicular Gradient at each point in X direction
    double gradientY[][];               // Matrix to represent the perpendicular Gradient at each point in Y direction
    double confidence[][];              // Matrix that stores the Confidence values for each pixel
    double data[][];                    // Matrix to store the data terms for each pixel
    GradientCalculator gc;              // Object of GradientCalculator class used to calculate gradient
    double omega = 0.7;                 // Represents the omega used in confidence term
    double Alpha = 0.2;                 // Represents the multiplier for confidence term in calculating priority
    double Beta = 0.8;                  // Represents the multiplier for data term in calculating priority
    int maxX;                           // The maximum X coordinate of the target region
    int maxY;                           // The maximum Y coordinate of the target region
    int minX;                           // The minimum X coordinate of the target region
    int minY;                           // The minimum Y coordinate of the target region
    int continuousCol = 0;              // The maximum number of continuous green pixels in one column
    int continuousRow = 0;              // The maximum number of continuous green pixels in one row
    protected Main owner;               // Object to the Main Class. Used in passing updates after every iteration
    public Boolean halt = false;        // Used to stop inpainting.
    public Boolean completed = false;        // Used to report when the inpainting is complete.
    final int diamX = 50;               // Diameter (in X direction) of the Region to be searched for
    final int diamY = 30;               // Diameter (in Y direction) of the Region to be searched for
    private int pixelPosX;              // The X-coordinate of current pixel being processed
    private int pixelPosY;              // The Y-coordinate of current pixel being processed
    int w = 3;                          // Patch size is 2*w + 1
    double con[][] = new double[][] {{1, 1, 1}, {1, -8, 1}, {1, 1, 1}}; //Laplacian filter used in founding boundary of target region

    /**
     * Constructor for the ImageInpaint Class
     * @param owner Object to the Main class representing the owner of ImageInpaint object.
     */
    public ImageInpaint(Main owner) {
        this.owner = owner;
    }

    /**
     * Function to inpaint the given image.
     * @param a_origImg Original Image (without the target region marked)
     * @param a_fillImg Image with the target region marked in green color
     * @param quickInpaint  Flag to specify whether quick painting is enabled or not
     */
    @SuppressWarnings("unchecked")
    public void init(BufferedImage a_origImg, BufferedImage a_fillImg, Boolean quickInpaint)
    {
        halt = false;
        int i, j;
        try {
            origImg = a_origImg;
            img = a_origImg;
            fillImg = a_fillImg;
            raster = origImg.getRaster();

            iw = img.getWidth(null);
            ih = img.getHeight(null);

            /**
             * Grab the pixels of Original Image.
             */
            pixels = new int[iw*ih];
            PixelGrabber pg = new PixelGrabber(img, 0, 0, iw, ih, pixels, 0, iw);
            pg.grabPixels();
            pixelmap = new int[ih][iw];

            for (i = 0; i < ih; i ++) {
                for (j = 0; j < iw; j ++) {
                    pixelmap[i][j] = pixels [i*iw+j];
                }
            }

             /**
             * Grab the pixels of Fill Image.
             */
           
           
            PixelGrabber fillPg = new PixelGrabber(fillImg, 0, 0, iw, ih, pixels, 0, iw);
            fillPg.grabPixels();
            fillPixelmap = new int[ih][iw];

            for (i = 0; i < ih; i ++) {
                for (j = 0; j < iw; j ++) {
                    fillPixelmap[i][j] = pixels [i*iw+j];
                }
            }

        } catch (InterruptedException e1) {
            javax.swing.JOptionPane.showMessageDialog(owner, "Error " + e1,"Error!!",
                                    javax.swing.JOptionPane.ERROR_MESSAGE);
            System.out.println("Error " + e1);
        }

        /**
         * Create instance of GradientCalculator to calculate gradients.
         */
       
       
        gc = new GradientCalculator();
        gc.calculateGradientFromImage(pixelmap, ih, iw);        // Calculate Perpendicular Gradient for original image.
        gradientX = gc.gradientX;
        gradientY = gc.gradientY;
       
        initialize_confidence_term();                           // Initialize the confidence term
        initialize_data_term();                                 // Initialize the data term

        fillRegion = new double[ih][iw];                        // Allocate Space for fillRegion
        sourceRegion = new int[ih][iw];                         // Allocate Space for SourceRegion
        initialSourceRegion = new int[ih][iw];                  // Allocate Space for initialSourceRegion

        /**
         * Initialize minX, minY, maxX and maxY to find these coordinates
         */
        minX = iw;
        minY = ih;
        maxX = maxY = 0;

        int pixel, r, g, b, countrow, countcol;
        continuousRow = continuousCol = 0;
        for (i = 0; i < ih; i ++) {
            countrow = 0;
            for (j = 0; j < iw; j ++) {
                pixel = fillPixelmap[i][j];
                r = 0xff & (pixel >> 16);                       // Obtain the red Component
                g = 0xff & (pixel >> 8);                        // Obtain the green Component
                b = 0xff & (pixel);                             // Obtain the blue Component
                /**
                 * If the color is green, mark it as fillRegion
                 * Otherwise mark it as SourceRegion.
                 */
                if (r == 0 && g == 255 && b == 0) {
                    countrow ++;                                // Increase the number of continuous green pixels
                    fillRegion[i][j] = 1;
                    sourceRegion[i][j] = 0;
                    initialSourceRegion[i][j] = 0;
                    if (j < minX) {
                        minX = j;
                    }
                    if (i < minY) {
                        minY = i;
                    }
                    if (j > maxX) {
                        maxX = j;
                    }
                    if (i > maxY) {
                        maxY = i;
                    }
                } else {
                    if (countrow > continuousRow) {
                        continuousRow = countrow;
                    }
                    countrow = 0;
                    fillRegion[i][j] = 0;
                    sourceRegion[i][j] = 1;
                    initialSourceRegion[i][j] = 1;
                }
            }
        }

        for (i = 0; i < iw; ++i) {
            countcol = 0;
            for (j = 0; j < ih; ++j) {
                pixel = fillPixelmap[j][i];
                r = 0xff & (pixel >> 16);                       // Obtain the red Component
                g = 0xff & (pixel >> 8);                        // Obtain the green Component
                b = 0xff & (pixel);                             // Obtain the blue Component
                /**
                 * If pixel is green, increase the countcol by 1.
                 */
                if (r == 0 && g == 255 && b == 0) {
                    countcol ++;                                // Increase the number of continuous green pixels
                } else {
                    if (countcol > continuousCol) {
                        continuousCol = countcol;
                    }
                    countcol = 0;
                }
            }
        }

        //System.out.println("Col = " + continuousCol);
        //System.out.println("Col = " + continuousRow);

        Boolean flag = true;                        // Flag to represent the completion of the inpainting process

        double[][] temp = new double[ih][iw];
        double[][] sourceGradX = new double[ih][iw];
        double[][] sourceGradY = new double[ih][iw];

        Vector dR = new Vector();                   // Vector to store pixels of the boundary
        Vector Nx = new Vector();                   // Vector to store corresponding gradient in X direction
        Vector Ny = new Vector();                   // Vector to store corresponding gradient in Y direction

        double count;                               // Temporary variable used in calculating confidence terms for patches
        double Rcp;                                 // Temporary variable used in calculating confidence terms for patches
        double tempPriority;                         // Temporary variable used in calculating maximum priority

        while (flag) {

            /**
             * Filter the fillRegion with Laplacian Filter. Done to find the pixels on the boundary.
             */
            temp = conv2 (fillRegion, con);

            /**
             * Find the gradient of Source Region.
             */
            gc.calculateGradient(sourceRegion, ih, iw);
            sourceGradX = gc.gradientX;
            sourceGradY = gc.gradientY;

            dR.clear();
            Nx.clear();
            Ny.clear();

            /**
             * Add the coordinates of boundary pixels to dR and their gradients to Nx and Ny.
             */
            for (i = 0; i < temp[0].length; i ++) {
                for (j = 0; j < temp.length; j ++) {
                    if (temp[j][i] > 0) {
                            dR.add(i*temp.length + j);
                            Nx.add(sourceGradX[j][i]);
                            Ny.add(sourceGradY[j][i]);
                    }
                }
            }

            double[][] N = normr(Nx, Ny);       // Normalize Nx and Ny and store in N

            Vector q = new Vector();
            count = 0.0;

            /**
             * Now calculate Confidence Terms for all the pixels in the boundary of target region.
             */
            for (i = 0; i < dR.size(); i ++) {
                int[][] Hp = getpatch(pixelmap, (Integer)dR.get(i));
                System.out.println();
                for (j = 0; j < Hp.length; j ++) {
                    for (int k = 0; k < Hp[0].length; k ++) {
                        int col = (int)Hp[j][k]/ih;
                        int row = (int)Hp[j][k]%ih;
                        System.out.println(col+"   "+row);
                        if (fillRegion[row][col] == 0) {
                           
                            count += confidence[row][col];
                            q.add(Hp[j][k]);
                        }
                    }
                }
               
                int col = (Integer)dR.get(i)/ih;
                int row = (Integer)dR.get(i)%ih;
                confidence[row][col] = count/(double)(Hp.length*Hp[0].length);
                count = 0;
            }

            double maxPriority = 0.0;
            int maxPriorityIndex = -1;
            /**
             * After calculating confidence terms, now calculate data term and then find the patch with maximum priority.
             * This is done using the additive priority term.
             */
            for (i = 0; i < dR.size(); i ++) {
                int col = (Integer)dR.get(i)/ih;
                int row = (Integer)dR.get(i)%ih;

                /**
                 * Calculate Data term. +0.001 to prevent it from getting value Zero.
                 */
                data[row][col] = Math.abs(gradientX[row][col]*N[i][0] + gradientY[row][col]*N[i][1]) + 0.001;
   
                /**
                 * Elevated Confidence term. This prevents it from reaching Zero to soon.
                 */
                Rcp = (1-omega)*confidence[row][col]+omega;

                tempPriority = Alpha*Rcp + Beta*data[row][col];     // Calculate Priority using additive function

                /**
                 * find the patch with maximum priority
                 */
                if (tempPriority >= maxPriority) {
                    maxPriority = tempPriority;
                    maxPriorityIndex = i;
                }
            }

            if (maxPriorityIndex == -1) break;          // If no patch is found, then inpainting is complete.

            /**
             * Obtain the patch with maximum priority.
             * @Hp: Coordinates of patch with maximum priority
             * @toFill: Represents which pixels (from the patch) are to be filled
             * @toFillTrans: Transpose of ToFill matrix
             */
            int[][] Hp = getpatch (pixelmap, (Integer)dR.get(maxPriorityIndex));
            double[][] toFill = new double[Hp.length][Hp[0].length];
            double[][] toFillTrans = new double[Hp[0].length][Hp.length];

            /**
             * Calculate ToFill and ToFillTrans
             */
            for (i = 0; i < Hp.length; i ++) {
                for (j = 0; j < Hp[0].length; j ++) {
                    int col = (int)Hp[i][j]/ih;
                    int row = (int)Hp[i][j]%ih;
                    toFill[i][j] = fillRegion[row][col];
                    toFillTrans[j][i] = fillRegion[row][col];
                }
            }

            pixelPosX = (Integer)dR.get(maxPriorityIndex)/ih;
            pixelPosY = (Integer)dR.get(maxPriorityIndex)%ih;
            /**
             * Find the Best Exemplar. if quickInpaint is set, then only a small region of complete image is scanned.
             * We now get the starting and ending X and Y coordinates of the best Patch.
             */
            int[] best = bestExemplar(Hp, toFillTrans, initialSourceRegion, quickInpaint);

            int nRows = best[3] - best[2] + 1;
            int nCols = best[1] - best[0] + 1;

            /**
             * Find the patch represented by best.
             */
            int[][] X = new int[nRows][nCols];
            int[][] Y = new int[nRows][nCols];
            int[][] Hq = new int[nRows][nCols];
            for (i = 0; i < nRows; i ++) {
                for (j = 0; j < nCols; j ++) {
                    X[i][j] = best[0] + j;
                    Y[i][j] = best[2] + i;
                    Hq[i][j] = X[i][j] + Y[i][j]*ih;
                }
            }

            int p = (Integer)dR.get(maxPriorityIndex);

            /**
             * Update the image, data values and confidence values.
             */
            for (i = 0; i < toFill.length; i ++) {
                for (j = 0; j < toFill[0].length; j ++) {
                    if (toFill[i][j] != 0) {
                        toFill[i][j] = 1;
                        int col = (int)Hp[i][j]/ih;
                        int row = (int)Hp[i][j]%ih;

                        int col1 = Hq[i][j]/ih;
                        int row1 = Hq[i][j]%ih;
                       
                        fillRegion[row][col] = 0;       // Since the region is reconstructed, remove from fillRegion
                        sourceRegion[row][col] = 1;     // Since the region is reconstructed, add to SourceRegion

                        /** Propagate Confidence and Isophote Values */
                        confidence[row][col] = confidence[p%ih][p/ih];
                        gradientX[row][col] = gradientX[row1][col1];
                        gradientY[row][col] = gradientY[row1][col1];
                        pixelmap[row][col] = pixelmap[row1][col1];

                        /**
                         * Update the image pixels
                         */
                        int[] color = new int[3];
                        color[0] = 0xff & (pixelmap[row1][col1] >> 16);
                        color[1] = 0xff & (pixelmap[row1][col1] >> 8);
                        color[2] = 0xff & (pixelmap[row1][col1]);
                        raster.setPixel(col, row, color);
                    }
                }
            }

            if (halt) break;
            owner.updateStats(origImg);         // Inform the owner about the updated image.
            Thread.yield();

            /**
             * Find whether any other pixels are remaining to be inpainted.
             */
            flag = false;
            for (i = 0; i < fillRegion.length; ++i) {
                for (j = 0; j < fillRegion[0].length; ++j) {
                    if (fillRegion[i][j] == 1) {
                        flag = true;
                        break;
                    }
                }
                if (flag) break;        // If none of the pixels are in target region, break
            }

            if (halt) break;     // Stop Inpainting if owner halts the process.
        }
        /**
         * No need of gradient calculator now.
         * Mark it as null to drop a hint to garbageCollector that it can now be removed.
         */
        gc = null;
        if (!halt) {
            completed = true;
        } else {
            completed = false;
        }
        owner.updateStats(origImg);     // Inform the owner about the updates.
        Thread.yield();
    }

    /**
     * Initializes the confidence term to zero for pixels in target region and 1 for pixels in source region.
     */
    void initialize_confidence_term()
    {
        confidence = new double[ih][iw];
        for(int i = 0; i < ih; i ++) {
            for (int j = 0; j < iw; j ++) {
                int p = pixelmap[i][j];
                int r = 0xff & (p >> 16);
                int g = 0xff & (p >> 8);
                int b = 0xff & (p);
                if (r == 0 && g == 255 && b == 0) {
                    confidence[i][j] = 0;
                } else {
                    confidence[i][j] = 1;
                }
            }
        }
    }

    /**
     * Initialize the data term.
     */
    void initialize_data_term()
    {
        data = new double[ih][iw];
        for (int i = 0; i < ih; i ++) {
            for (int j = 0; j < iw; j ++) {
                data[i][j] = -0.1;
            }
        }
    }

    /**
     * Function to convolve the given matrix with another matrix
     * @param a Matrix to be convolved
     * @param b Matrix to be convolved with
     * @return The result after convolution.
     */
    double[][] conv2 (double a[][], double b[][])
    {
        int ra = a.length;
        int ca = a[0].length;
        int rb = b.length;
        int cb = b[0].length;

        // do full convolution
        double c[][] = new double[ra+rb-1][ca+cb-1];
        for (int i = 0; i < rb; i ++) {
            for (int j = 0; j < cb; j ++) {
                int r1 = i;
                int r2 = r1 + ra - 1;
                int c1 = j;
                int c2 = c1 + ca - 1;
                for (int k = r1; k < r2; k ++) {
                    for (int l = c1; l < c2; l ++) {
                        c[k][l] = c[k][l] + b[i][j]*a[k-r1+1][l-c1+1];
                    }
                }
            }
        }

        double out[][] = new double[ra][ca];
        //extract region of size(a) from c
        int r1 = rb/2;
        int r2 = r1 + ra - 1;
        int c1 = cb/2;
        int c2 = c1 + ca - 1;
        for (int i = r1; i < r2; i ++) {
            for (int j = c1; j < c2; j ++) {
                out[i-r1+1][j-c1+1] = c[i][j];
            }
        }      
        return out;
    }

    /**
     * Function to normalize the given vectors and return them concatenated in one single matrix
     * @param X Vector 1
     * @param Y Vector 2
     * @return  The result after normalizing and then concatenating the two vectors.
     */
    @SuppressWarnings("unchecked")
    double[][] normr (Vector X, Vector Y )
    {
        double normalized[][] = new double[X.size()][2];
        for (int i = 0; i < X.size(); i ++) {
            double temp1 = (Double)X.get(i);
            double temp2 = (Double)Y.get(i);
           
            temp1 *= temp1;
            temp2 *= temp2;
           
            double temp3;
           
            if(temp1 + temp2 == 0) {
                temp3 = 0;
            } else {
                temp3 = (double)1/(temp1 + temp2);
                temp3 = Math.sqrt(temp3);
            }
           
            normalized[i][0] = temp3*(Double)X.get(i);
            normalized[i][1] = temp3*(Double)Y.get(i);
        }
    
        return normalized;
    }

    /**
     * Function that returns the patch around the given pixel.
     * @param pixelmap  Matrix representing the pixelmap of the image.
     * @param p Position of the pixel around which patch is to be calculated
     * @return  a matrix that contains the coordinates of the pixels that form the patch.
     */
    @SuppressWarnings("unchecked")
    int[][] getpatch (int[][] pixelmap, int p)
    {
       
        int y;
        int x;

        p = p-1;
        y = (p/ih);//+ 1;
        p = p % ih;
        x = p+1;
       
        int temp1 = Math.max(x-w, 0);           // Starting X coordinate
        int temp2 = Math.min(x+w,ih-1);         // Ending X coordinate
        int temp3 = Math.max(y-w,0);            // Starting Y Coordinate
        int temp4 = Math.min(y+w,iw-1);         // Ending Y Coordinate
           
        //int[][] X = new int[temp4-temp3+1][temp2-temp1+1];
        //int[][] Y = new int[temp4-temp3+1][temp2-temp1+1];
        int [][] N = new int[temp4-temp3+1][temp2-temp1+1];
       
        for (int i = 0; i < temp4-temp3+1; i ++) {
            for (int j = 0; j < temp2-temp1+1; j ++) {
                //X[i][j] = temp1 + j;
                //Y[i][j] = temp3 + i;
               
                System.out.println(i+"Patching pixel value :- "+N[i][j]);
                N[i][j] = temp1+j + (temp3+i)*ih;
            }
        }
        return N;
    }

    /**
     * Function that finds the best exemplar from the image that matches the patch given to the fucntion.
     * By default, it finds the complete image to look for the best exemplar.
     * But if quickInpaint is set, it only looks for the patch inside a region defined by minX, maxX, minY, maxY, diamX and diamY
     * @param Hp    Patch for which the best exemplar is to be found
     * @param toFill    Pixels that are to be filled from the patch.
     * @param sourceRegion  the source region from where to look for the best exemplar
     * @param quickInpaint  Flag that represents whether or not to perform quick Inpainting.
     * @return  Returns an integer array with four elements representing starting x coordinate, ending x coordinate, starting y coordinate and ending y coordinate respectively.
     */
 int[] bestExemplar (int [][] Hp, double[][] toFill, int[][] sourceRegion, Boolean quickInpaint)
    {
        int [][] Ip = new int[toFill.length][toFill[0].length];
        for (int i = 0; i < toFill[0].length; i ++) {
            for (int j = 0; j < toFill.length; j ++) {
                int col = (int)Hp[i][j]/ih;
                int row = (int)Hp[i][j]%ih;
                Ip[j][i] = pixelmap[row][col];              // transpose
            }
        }

        int[][] rIp = new int[Ip.length][Ip[0].length];
        int[][] gIp = new int[Ip.length][Ip[0].length];
        int[][] bIp = new int[Ip.length][Ip[0].length];

        for (int i = 0; i < Ip.length; ++i) {
            for (int j = 0; j < Ip[0].length; ++j) {
                /**
                 * Extract the RGB components of the image.
                 */
                rIp[i][j] = 0xff & (Ip[i][j] >> 16);
                gIp[i][j] = 0xff & (Ip[i][j] >> 8);
                bIp[i][j] = 0xff & (Ip[i][j]);
            }
        }

        int mm, nn;
        int startX;
        int startY;
        int endX;
        int endY;
        int m = Ip.length;
        int n = Ip[0].length;
        if (quickInpaint) {

            startX = Math.max(0, pixelPosX - n/2 - continuousRow - diamX/2);      // Set the start X coordinate above the pixel's X-coordinate by some amount
            startY = Math.max(0, pixelPosY - m/2 - continuousCol -diamY/2);       // Set the start Y coordinate above the pixel's Y-coordinate by some amount
            endX = Math.min(pixelmap[0].length-1, pixelPosX + n/2 + continuousRow + diamX/2);
            endY = Math.min(pixelmap.length-1, pixelPosY + m/2 + continuousCol + diamY/2);
            mm = endY - startY + 1;
            nn = endX - startX + 1;
        } else {
            mm = pixelmap.length;
            nn = pixelmap[0].length;
            startX = 0;
            startY = 0;
            endX = endY = 0;
        }
        int i,j,ii,jj,ii2,jj2,M,N,I,J;
        double patchErr=0.0,err=0.0,bestErr = 99999999999999999999.0, bestPatchErr1 = 99999999999999999999.0;
        int[] best = {0, 0, 0, 0};
        Boolean skipPatchFlag = false;
       
        /** for each patch */
        N= startX + nn-n+1;  M= startY + mm-m+1;
        for (j = startX; j < N; j++) {
            J = j+n-1;
            for (i = startY; i < M; i++) {
                I = i+m-1;
               
                skipPatchFlag = false;
               
                double meanR = 0.0;
                double meanG = 0.0;
                double meanB = 0.0;
               
               
                /**
                 * Calculate patch error
                 */
                for (jj = j, jj2 = 0; jj <= J; jj++,jj2++) {
                    for (ii = i,ii2 = 0; ii <= I; ii++,ii2++) {

                        /**
                         * If any pixels does not belong to the source region, then skip the patch.
                         */
                        if (sourceRegion[ii][jj]!=1) {
                            skipPatchFlag = true;
                            break;
                        }
                        if (toFill[ii2][jj2]==0) {
                            int rImage = 0xff & (pixelmap[ii][jj] >> 16);
                            int gImage = 0xff & (pixelmap[ii][jj] >> 8);
                            int bImage = 0xff & (pixelmap[ii][jj]);

                            /**
                             * Calculate the mean square error for the patch.
                             */
                            err = rImage - rIp[ii2][jj2]; patchErr += err*err;
                            err = gImage - gIp[ii2][jj2]; patchErr += err*err;
                            err = bImage - bIp[ii2][jj2]; patchErr += err*err;
                            /**
                             * Calculate the mean of the color values. Used when we get two patches with same best error.
                             */
                            meanR += rImage;
                            meanG += gImage;
                            meanB += bImage;
                        }
                    }
                    if (skipPatchFlag) {
                        break;
                    }
                }
               
                /**
                 * Update
                 */
                if (!skipPatchFlag && patchErr < bestErr ) {
                    bestErr = patchErr;
                    best[0] = i; best[1] = I;
                    best[2] = j; best[3] = J;

                    /**
                     * Calculate the variance of the patch.
                     */
                    double patchErr1 = 0.0;
                    for (jj = j, jj2 = 0; jj <= J; jj++,jj2++) {
                        for (ii = i,ii2 = 0; ii <= I; ii++,ii2++) {
                            if (toFill[ii2][jj2]==1) {
                                int rImage = 0xff & (pixelmap[ii][jj] >> 16);
                                int gImage = 0xff & (pixelmap[ii][jj] >> 8);
                                int bImage = 0xff & (pixelmap[ii][jj]);

                                err = rImage - meanR; patchErr1 += err*err;
                                err = gImage - meanG; patchErr1 += err*err;
                                err = bImage - meanB; patchErr1 += err*err;
                            }
                        }
                    }
                    bestPatchErr1 = patchErr1;      // Update the variance of the best patch found so far.
                } else if (!skipPatchFlag && patchErr == bestErr) {
                    /**
                     * If the current patch has same error as the previous best patch
                     * we find the variance of this patch.
                     * The patch with minimum variance is now selected as the best patch.
                     */
                    double patchErr1 = 0.0;
                    for (jj = j, jj2 = 0; jj <= J; jj++,jj2++) {
                        for (ii = i,ii2 = 0; ii <= I; ii++,ii2++) {
                            if (toFill[ii2][jj2]==1) {
                                int rImage = 0xff & (pixelmap[ii][jj] >> 16);
                                int gImage = 0xff & (pixelmap[ii][jj] >> 8);
                                int bImage = 0xff & (pixelmap[ii][jj]);

                                err = rImage - meanR; patchErr1 += err*err;
                                err = gImage - meanG; patchErr1 += err*err;
                                err = bImage - meanB; patchErr1 += err*err;
                            }
                        }
                    }
                    /**
                     * Select the new patch if it has lower variance.
                     * Otherwise discard.
                     */
                    if (bestPatchErr1 > patchErr1) {
                        best[0] = i; best[1] = I;
                        best[2] = j; best[3] = J;
                        bestPatchErr1 = patchErr1;
                    }
                }
               
                patchErr = 0.0;
            }
        }

        if (best[0] == 0 && best[1] == 0 && best[2] == 0 && best[3] == 0) {
            return (bestExemplar (Hp, toFill, sourceRegion, false));
        }
        return best;
    }
}




No comments:

Post a Comment