All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.hipparchus.samples.clustering.ImageClusteringExample Maven / Gradle / Ivy

/*
 * Licensed to the Apache Software Foundation (ASF) under one or more
 * contributor license agreements.  See the NOTICE file distributed with
 * this work for additional information regarding copyright ownership.
 * The ASF licenses this file to You 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
 *
 *      https://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.
 */

/*
 * This is not the original file distributed by the Apache Software Foundation
 * It has been modified by the Hipparchus project
 */
package org.hipparchus.samples.clustering;

import java.awt.BorderLayout;
import java.awt.Color;
import java.awt.Component;
import java.awt.Dimension;
import java.awt.FlowLayout;
import java.awt.Graphics;
import java.awt.GridLayout;
import java.awt.event.ActionEvent;
import java.awt.event.ActionListener;
import java.awt.image.BufferedImage;
import java.awt.image.Raster;
import java.awt.image.WritableRaster;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;

import javax.imageio.ImageIO;
import javax.swing.BorderFactory;
import javax.swing.Box;
import javax.swing.ImageIcon;
import javax.swing.JButton;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.JSpinner;
import javax.swing.SpinnerNumberModel;

import org.hipparchus.clustering.CentroidCluster;
import org.hipparchus.clustering.Clusterable;
import org.hipparchus.clustering.KMeansPlusPlusClusterer;
import org.hipparchus.samples.ExampleUtils;
import org.hipparchus.samples.ExampleUtils.ExampleFrame;

/**
 * This example shows how clustering can be applied to images.
 */
@SuppressWarnings("serial")
public class ImageClusteringExample {

    /** Empty constructor.
     * 

* This constructor is not strictly necessary, but it prevents spurious * javadoc warnings with JDK 18 and later. *

* @since 3.0 */ public ImageClusteringExample() { // NOPMD - unnecessary constructor added intentionally to make javadoc happy // nothing to do } /** Main frame for displaying clusters. */ public static class Display extends ExampleFrame { /** Reference image. */ private BufferedImage referenceImage; /** Cluster image. */ private BufferedImage clusterImage; /** Reference raster. */ private Raster referenceRaster; /** Painter for the clusters. */ private ImagePainter painter; /** Spinner. */ private JSpinner clusterSizeSpinner; /** Simple constructor. * @throws IOException if image cannot be created */ public Display() throws IOException { setTitle("Hipparchus: Image Clustering Example"); setSize(900, 350); setLayout(new FlowLayout()); Box bar = Box.createHorizontalBox(); ClassLoader classLoader = ExampleUtils.class.getClassLoader(); referenceImage = ExampleUtils.resizeImage( ImageIO.read(classLoader.getResourceAsStream("ColorfulBird.jpg")), 350, 240, BufferedImage.TYPE_INT_RGB); referenceRaster = referenceImage.getData(); clusterImage = new BufferedImage(referenceImage.getWidth(), referenceImage.getHeight(), BufferedImage.TYPE_INT_RGB); JLabel picLabel = new JLabel(new ImageIcon(referenceImage)); bar.add(picLabel); painter = new ImagePainter(clusterImage.getWidth(), clusterImage.getHeight()); bar.add(painter); JPanel controlBox = new JPanel(); controlBox.setLayout(new GridLayout(5, 1)); controlBox.setBorder(BorderFactory.createLineBorder(Color.black, 1)); JPanel sizeBox = new JPanel(); JLabel sizeLabel = new JLabel("Clusters:"); sizeBox.add(sizeLabel); SpinnerNumberModel model = new SpinnerNumberModel(3, 2, 10, 1); clusterSizeSpinner = new JSpinner(model); sizeLabel.setLabelFor(clusterSizeSpinner); sizeBox.add(clusterSizeSpinner); controlBox.add(sizeBox, BorderLayout.NORTH); JButton startButton = new JButton("Cluster"); startButton.setActionCommand("cluster"); controlBox.add(startButton, BorderLayout.CENTER); bar.add(controlBox); add(bar); startButton.addActionListener(new ActionListener() { public void actionPerformed(ActionEvent e) { clusterImage(); } }); } /** Display clusters. */ private void clusterImage() { List pixels = new ArrayList(); for (int row = 0; row < referenceImage.getHeight(); row++) { for (int col = 0; col < referenceImage.getWidth(); col++) { pixels.add(new PixelClusterable(col, row)); } } int clusterSize = ((Number) clusterSizeSpinner.getValue()).intValue(); KMeansPlusPlusClusterer clusterer = new KMeansPlusPlusClusterer(clusterSize); List> clusters = clusterer.cluster(pixels); WritableRaster raster = clusterImage.getRaster(); for (CentroidCluster cluster : clusters) { double[] color = cluster.getCenter().getPoint(); for (PixelClusterable pixel : cluster.getPoints()) { raster.setPixel(pixel.x, pixel.y, color); } } Display.this.repaint(); } /** Container for one pixel that can be used in clusters. */ private class PixelClusterable implements Clusterable { /** Abscissa. */ private final int x; /** Ordinate. */ private final int y; /** Color. */ private double[] color; /** Simple constructor. * @param x abscissa * @param y ordinate */ PixelClusterable(int x, int y) { this.x = x; this.y = y; this.color = null; } /** {@inheritDoc} */ @Override public double[] getPoint() { if (color == null) { color = referenceRaster.getPixel(x, y, (double[]) null); } return color; } } /** Painter for clusters. */ private class ImagePainter extends Component { /** Width. */ private int width; /** Height. */ private int height; /** Simple constructor. * @param width width * @param height height */ ImagePainter(int width, int height) { this.width = width; this.height = height; } /** {@inheritDoc} */ @Override public Dimension getPreferredSize() { return new Dimension(width, height); } /** {@inheritDoc} */ @Override public Dimension getMinimumSize() { return getPreferredSize(); } /** {@inheritDoc} */ @Override public Dimension getMaximumSize() { return getPreferredSize(); } /** {@inheritDoc} */ @Override public void paint(Graphics g) { g.drawImage(clusterImage, 0, 0, this); } } } /** Program entry point. * @param args program arguments (unused here) * @throws IOException if display frame cannot be created. */ public static void main(String[] args) throws IOException { ExampleUtils.showExampleFrame(new Display()); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy