one.empty3.feature.CurveFitting Maven / Gradle / Ivy
/*
* Copyright (c) 2022-2023. Manuel Daniel Dahmen
*
*
* Copyright 2012-2023 Manuel Daniel Dahmen
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package one.empty3.feature;
import one.empty3.feature.app.replace.java.awt.Color;
import one.empty3.feature.shape.Rectangle;
import one.empty3.io.ProcessFile;
import one.empty3.library.Axe;
import one.empty3.library.Circle;
import one.empty3.library.ColorTexture;
import one.empty3.library.Point3D;
import one.empty3.library.core.nurbs.CourbeParametriquePolynomialeBezier;
import javax.imageio.ImageIO;
import java.awt.*;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.HashMap;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;
public class CurveFitting extends ProcessFile {
CourbeN11 curvePoints;
/*
*
* Morphing
* Mask
* Paste
*
* On donne une courbe initiale on essaie de la faire correspondre à une portion de l'image
* Commencer par une courbe initiale comme un cercle
* Déformer en courbe de Bézier de longueur N = N points de contrôle correspondant aux pixels
* de la carte initiale
* Carte des distances à la courbe
* Vu dans Dr Lingrand, et R. Szeliski.
*/
PixM pix = null;
private double lambda1 = 0.01, lambda2 = 0.01, lambda3 = 1.0;
double[] outAvg = new double[]{0, 0, 0}, inAvg = new double[]{0, 0, 0};
private double arcLength;
private double[][] distancesArr;
private CourbeParametriquePolynomialeBezier curveResult;
private CourbeParametriquePolynomialeBezier curveInitial;
private double[][] deltaE;
private HashMap> border;
private PixM inPix;
private PixM outPix;
public void init() {
border = new HashMap<>();
deltaE = new double[pix.getColumns()][pix.getLines()];
lambda1 = 0.01;
lambda2 = 0.02;
lambda3 = 1.0;
distancesArr = new double[pix.getColumns()][pix.getLines()];
outAvg = new double[]{0., 0., 0.};
inAvg = new double[]{0., 0., 0.};
curveResult = new CourbeParametriquePolynomialeBezier();
}
public double E() {
double t1 = 0.0;
double t2 = 0.0;
double arcLengthPart = 0;
double e = 0.0;
inAvg = mean(true);
outAvg = mean(false);
for (int j = 0; j < pix.getColumns(); j++) {
for (int k = 0; k < pix.getLines(); k++) {
t1 += inPix.get(j, k) * lambda1 * (pix.luminance(j, k) - Math.sqrt(inAvg[0] * inAvg[0]
+ inAvg[1] * inAvg[1] + inAvg[2] * inAvg[2])) * (pix.luminance(j, k)
- Math.sqrt(inAvg[0] * inAvg[0]
+ inAvg[1] * inAvg[1] + inAvg[2] * inAvg[2]));
t2 += outPix.get(j, k) * lambda2 * (pix.luminance(j, k)
+ Math.sqrt(outAvg[0] * outAvg[0] + outAvg[1] * outAvg[1] + outAvg[2] * outAvg[2]))
* (pix.luminance(j, k) -
Math.sqrt(outAvg[0] * outAvg[0] + outAvg[1] * outAvg[1] + outAvg[2] * outAvg[2]));
}
}
arcLength = 0.0;
for (int i = 1; i < curvePoints.getCoefficients().getData1d().size() - 1; i++)
arcLength += lambda3 * curvePoints.getCoefficients().getData1d().get(i + 1)
.moins(curvePoints.getCoefficients().getElem(i)).norme();
e = t1 - t2 + lambda3 * arcLength;
System.out.printf("\nE terme intérieur t1 %f\n", t1);
System.out.printf("t2 terme extérieur t2 %f\n", t2);
System.out.printf("L Longueur de la courbe %f\n", arcLength);
return e;
}
public double[] mean(boolean in) {
double[] res = new double[]{0., 0., 0.};
for (int i = 0; i < pix.getCompNo(); i++) {
for (int j = 0; j < pix.getColumns(); j++) {
for (int k = 0; k < pix.getLines(); k++) {
int isIn = in ? 1 : 0;
pix.setCompNo(i);
res[i] += pix.get(j, k) / Math.sqrt(3)
* (in ? inPix.get(j, k) : (outPix.get(j, k)));
}
}
}
return res;
}
public double[][] distances() {
border = new HashMap<>();
int IN = 0, OUT = 0;
arcLength = 0.0;
double dist = 0.0;
double inEnergy, outEnergy;
arcLength = 0.0;
distancesArr = new double[pix.getColumns()][pix.getLines()];
for (int i = 0; i < distancesArr.length; i++) {
Arrays.fill(distancesArr[i], 100000);
}
//distancesArr[i][j] = 0.0;
for (int j = 0; j < pix.getLines(); j++) {
border.put(j, new ArrayList<>());
for (int i = 0; i < pix.getColumns(); i++) {
for (int k = 0; k < curvePoints.getCoefficients().getData1d().size(); k++) {
// Si vous habitez dans une petite rue, ça n'ira pas tout seul. Je le redis une fois. NON. Autographe d'un mongol.
// Dédié à Lucky Luke.
dist = Point3D.distance(pix.getRgb(i, j), curvePoints.getCoefficients().getElem(k));
if (dist < distancesArr[i][j])
distancesArr[i][j] = dist;
if (dist < 1.0) {
// Point de la courbe
border.get(j).add(i);
}
}
if (inPix.luminance(i, j) > 0) {
IN++;
} else if (outPix.luminance(i, j) > 0) {
OUT++;
}
}
}
for (int i = 1; i < curvePoints.getCoefficients().getData1d().size(); i++) {
arcLength += curvePoints.getCoefficients().getData1d().get(i)
.moins(curvePoints.getCoefficients().getData1d().get(i - 1)).norme();
}
System.out.printf("In points count %d\n", IN);
System.out.printf("Out points count %d\n", OUT);
return distancesArr;
}
/*
Générer une approximation avec points de contrôle.
*/
public CourbeN11 approx() {
List p = new ArrayList<>();
double incr = (curveInitial.getEndU() - curveInitial.getStartU()) * 0.0001 /
Point3D.distance(curveInitial.calculerPoint3D(0.0001),
curveInitial.calculerPoint3D(0.0)); // Garder l'incrément en chaque point? pour l'itération suivante
double incr0 = incr;
double lastIncr = incr;
final int MAX_ITER = 100;
int iter = 0;
for (double t = 0.0; t < 1.0; t += incr) {
Point3D pt1 = curveInitial.calculerPoint3D(t);
Point3D pt2 = curveInitial.calculerPoint3D(t + incr);
double dist = Point3D.distance(pt1, pt2);
//if (dist >= 1.0 && dist < 2.0) {
if (dist >= 0.9 && dist < 1.1) {
//p.add(Point3D.P.n((int) (double) pt2.get(0),
// (int) (double) pt2.get(1), (int) (double) pt2.get(2)));
iter = 0;
lastIncr = incr;
p.add(pt2);
arcLength += dist;
} else if (dist < 1.0) {
t = t - incr0;
incr *= 1.5;
incr0 = incr;
iter++;
} else if (iter > MAX_ITER) {
incr = lastIncr;
incr0 = incr;
t += incr;
iter = 0;
continue;
} else {
t = t - incr0;
incr /= 1.5;
incr0 = incr;
iter++;
} // à détailler
if (p.size() > maxRes * 4) {
Logger.getAnonymousLogger().log(Level.INFO, "ArrayList.size too long : " + p.size());
break;
}
}
Point3D[] point3DS = new Point3D[p.size()];
p.toArray(point3DS);
curvePoints = new CourbeN11(point3DS);
return curvePoints;
}
@Override
public boolean process(File in, File out) {
try {
pix = PixM.getPixM(ImageIO.read(in), maxRes);
init();
inPix = new PixM(pix.getImage());
outPix = new PixM(pix.getImage());
curveInitial = new CourbeN11();
int imgX1;
int imgY1;
int imgX2;
int imgY2;
int moy = (pix.getColumns() + pix.getLines()) / 2;
int square;
if (pix.getColumns() > pix.getLines()) {
square = pix.getLines();
imgX1 = pix.getColumns() / 2 - square / 2;
imgY1 = 0;
imgX2 = pix.getColumns() / 2 + square / 2;
imgY2 = pix.getLines() - 1;
} else {
square = pix.getColumns();
imgX1 = 0;
imgY1 = pix.getLines() / 2 - square / 2;
imgX2 = pix.getColumns() - 1;
imgY2 = pix.getLines() / 2 + square / 2;
}
int N = 12;
for (int i = 0; i < N; i++) {
curveInitial.getCoefficients().getData1d().add(
new Point3D(pix.getColumns() / 2
+ square / 2. * Math.cos(i * 2 * Math.PI / N),
pix.getLines() / 2
+ square / 2. * Math.sin(i * 2 * Math.PI / N),
0.0));
}
Point3D center = new Point3D(pix.getColumns() / 2., pix.getLines() / 2., 0.0);
Circle circle = new Circle(new Axe(center.plus(Point3D.Z), center.plus(Point3D.Z.mult(-1))), square / 2.);
//System.out.printf("%s", curveInitial.toString());
//curveInitial = new Circle(new Axe(Point3D.Z.plus(Point3D.n(pix.getColumns()/2,
// pix.getLines()/2, 0)), Point3D.Z.mult(-1)), (imgX2-imgX1)/3.*2);
inPix.paintAll(new double[]{0, 0, 0});
inPix.fillIn(curveInitial, new ColorTexture(Color.WHITE), new ColorTexture(Color.RED));
outPix.paintAll(new double[]{1, 1, 1});
outPix.fillIn(curveInitial, new ColorTexture(Color.BLACK), new ColorTexture(Color.RED));
curvePoints = approx();
curvePoints.getClosed().setElem(false);
//double[][] distances = distances();
//inPix.paintAll(new double[]{0, 0, 0});
//outPix.paintAll(new double[]{1, 1, 1});
inPix.fillIn(curvePoints, new ColorTexture(Color.WHITE), new ColorTexture(Color.WHITE));
outPix.fillIn(curvePoints, new ColorTexture(Color.BLACK), new ColorTexture(Color.BLACK));
double e = E();
//curveResult = modify();
PixM p = new PixM(pix.getColumns(), pix.getLines());
curvePoints.setIncrU(1. / maxRes / curvePoints.getCoefficients().data1d.size());
Logger.getAnonymousLogger().log(Level.INFO, "Courbe 4/5");
p.plotCurve(curvePoints, new ColorTexture(Color.WHITE));
Logger.getAnonymousLogger().log(Level.INFO, "Courbe 5/5");
p.plotCurve(circle, new ColorTexture(Color.BLUE));
for (Point3D c : curvePoints.getCoefficients().getData1d()) {
Rectangle rectangle = new Rectangle(c.getX() - 3, c.getY() - 3, 6, 6);
rectangle.setIncrU(0.1);
}
PixM normalize = p.normalize(0.0, 1.0, 0.0, 1.0);
// E = Sout + Sin + Scourbe :
// Convergence vers ?
// Changer l1,l2,l3
// Convergence de courbe . Modification de la courbe.
//normalize.plotCurve(courbeParametriquePolynomialeBezier, Color.GREEN);
//normalize.plotCurve(curvePoints, Color.BLUE);
//normalize.fillIn(courbeParametriquePolynomialeBezier, Color.BLACK, Color.WHITE);
System.out.printf("curveResult %d\n", curveResult.getCoefficients().getData1d().size());
System.out.printf("curveInitial %d\n", ((CourbeParametriquePolynomialeBezier) curveInitial)
.getCoefficients().getData1d().size());
System.out.printf("curvePoints %d\n", curvePoints.getCoefficients().getData1d().size());
System.out.printf("Energy %f\n", e);
Logger.getAnonymousLogger().log(Level.INFO,""+ curvePoints);
BufferedImage image = normalize.getImage();
Graphics graphics = image.getGraphics();
graphics.setColor(Color.RED);
graphics.drawArc(image.getWidth() / 2 - square / 2, image.getHeight() / 2 - square / 2,
square, square, 0, 360);
String absolutePath = out.getAbsolutePath();
absolutePath = absolutePath.substring(0, absolutePath.length() - 4);
ImageIO.write(image, "jpg", new File(absolutePath + ".jpg"));
ImageIO.write(outPix.normalize(0.0, 1.0, 0., 1.).getImage(), "jpg", new File(
absolutePath + "-outPixels.jpg"));
ImageIO.write(inPix.normalize(0.0, 1.0, 0., 1.).getImage(), "jpg", new File(
absolutePath + "-inPixels.jpg"));
return true;
} catch (IOException e) {
e.printStackTrace();
}
return false;
}
private CourbeParametriquePolynomialeBezier modify() {
return new CourbeParametriquePolynomialeBezier();
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy