- android - 多次调用 OnPrimaryClipChangedListener
- android - 无法更新 RecyclerView 中的 TextView 字段
- android.database.CursorIndexOutOfBoundsException : Index 0 requested, 光标大小为 0
- android - 使用 AppCompat 时,我们是否需要明确指定其 UI 组件(Spinner、EditText)颜色
我正在用 Java 编写简单的 3D 游戏,我需要用于纹理变形的代码/lib。我不想使用 OpenGL。
示例:
我有纹理:
我需要将其变形为:
希望大家能够理解我。谢谢!
最佳答案
好的,我们开始吧。作为一个小型、独立的示例实现,您可以在其中用鼠标拖动角落:
+1 至...
代码如下:
import java.awt.Color;
import java.awt.Graphics;
import java.awt.GridLayout;
import java.awt.Point;
import java.awt.event.MouseEvent;
import java.awt.event.MouseListener;
import java.awt.event.MouseMotionListener;
import java.awt.geom.Point2D;
import java.awt.image.BufferedImage;
import java.io.File;
import java.io.IOException;
import javax.imageio.ImageIO;
import javax.swing.ImageIcon;
import javax.swing.JFrame;
import javax.swing.JLabel;
import javax.swing.JPanel;
import javax.swing.SwingUtilities;
public class Pseudo3DTest
{
public static void main(String[] args)
{
SwingUtilities.invokeLater(new Runnable()
{
@Override
public void run()
{
createAndShowGUI();
}
});
}
private static void createAndShowGUI()
{
JFrame f = new JFrame();
f.setDefaultCloseOperation(JFrame.EXIT_ON_CLOSE);
BufferedImage image = null;
try
{
image = ImageIO.read(new File("lena512color.png"));
}
catch (IOException e)
{
e.printStackTrace();
return;
}
f.getContentPane().setLayout(new GridLayout(1,2));
f.getContentPane().add(new JLabel(new ImageIcon(image)));
f.getContentPane().add(new Pseudo3DImagePanel(image));
f.pack();
f.setLocationRelativeTo(null);
f.setVisible(true);
}
}
class Pseudo3DImagePanel extends JPanel
implements MouseListener, MouseMotionListener
{
private final BufferedImage inputImage;
private final Point2D p0;
private final Point2D p1;
private final Point2D p2;
private final Point2D p3;
private Point2D draggedPoint;
Pseudo3DImagePanel(BufferedImage inputImage)
{
this.inputImage = inputImage;
this.p0 = new Point2D.Double(30,20);
this.p1 = new Point2D.Double(50,400);
this.p2 = new Point2D.Double(450,300);
this.p3 = new Point2D.Double(430,100);
addMouseListener(this);
addMouseMotionListener(this);
}
@Override
protected void paintComponent(Graphics g)
{
super.paintComponent(g);
BufferedImage image = Pseudo3D.computeImage(inputImage, p0, p1, p2, p3);
g.drawImage(image, 0, 0, null);
int r = 8;
g.setColor(Color.RED);
g.fillOval((int)p0.getX()-r, (int)p0.getY()-r, r+r, r+r);
g.fillOval((int)p1.getX()-r, (int)p1.getY()-r, r+r, r+r);
g.fillOval((int)p2.getX()-r, (int)p2.getY()-r, r+r, r+r);
g.fillOval((int)p3.getX()-r, (int)p3.getY()-r, r+r, r+r);
}
@Override
public void mousePressed(MouseEvent e)
{
Point p = e.getPoint();
int r = 8;
if (p.distance(p0) < r) draggedPoint = p0;
if (p.distance(p1) < r) draggedPoint = p1;
if (p.distance(p2) < r) draggedPoint = p2;
if (p.distance(p3) < r) draggedPoint = p3;
}
@Override
public void mouseDragged(MouseEvent e)
{
if (draggedPoint != null)
{
draggedPoint.setLocation(e.getX(), e.getY());
repaint();
}
}
@Override
public void mouseReleased(MouseEvent e)
{
draggedPoint = null;
}
@Override
public void mouseMoved(MouseEvent e) {}
@Override
public void mouseClicked(MouseEvent e) {}
@Override
public void mouseEntered(MouseEvent e) {}
@Override
public void mouseExited(MouseEvent e) {}
}
class Pseudo3D
{
static BufferedImage computeImage(
BufferedImage image,
Point2D p0, Point2D p1, Point2D p2, Point2D p3)
{
int w = image.getWidth();
int h = image.getHeight();
BufferedImage result =
new BufferedImage(w, h, BufferedImage.TYPE_INT_ARGB);
Point2D ip0 = new Point2D.Double(0,0);
Point2D ip1 = new Point2D.Double(0,h);
Point2D ip2 = new Point2D.Double(w,h);
Point2D ip3 = new Point2D.Double(w,0);
Matrix3D m = computeProjectionMatrix(
new Point2D[] { p0, p1, p2, p3 },
new Point2D[] { ip0, ip1, ip2, ip3 });
Matrix3D mInv = new Matrix3D(m);
mInv.invert();
for (int y = 0; y < h; y++)
{
for (int x = 0; x < w; x++)
{
Point2D p = new Point2D.Double(x,y);
mInv.transform(p);
int ix = (int)p.getX();
int iy = (int)p.getY();
if (ix >= 0 && ix < w && iy >= 0 && iy < h)
{
int rgb = image.getRGB(ix, iy);
result.setRGB(x, y, rgb);
}
}
}
return result;
}
// From https://math.stackexchange.com/questions/296794
private static Matrix3D computeProjectionMatrix(Point2D p0[], Point2D p1[])
{
Matrix3D m0 = computeProjectionMatrix(p0);
Matrix3D m1 = computeProjectionMatrix(p1);
m1.invert();
m0.mul(m1);
return m0;
}
// From https://math.stackexchange.com/questions/296794
private static Matrix3D computeProjectionMatrix(Point2D p[])
{
Matrix3D m = new Matrix3D(
p[0].getX(), p[1].getX(), p[2].getX(),
p[0].getY(), p[1].getY(), p[2].getY(),
1, 1, 1);
Point3D p3 = new Point3D(p[3].getX(), p[3].getY(), 1);
Matrix3D mInv = new Matrix3D(m);
mInv.invert();
mInv.transform(p3);
m.m00 *= p3.x;
m.m01 *= p3.y;
m.m02 *= p3.z;
m.m10 *= p3.x;
m.m11 *= p3.y;
m.m12 *= p3.z;
m.m20 *= p3.x;
m.m21 *= p3.y;
m.m22 *= p3.z;
return m;
}
private static class Point3D
{
double x;
double y;
double z;
Point3D(double x, double y, double z)
{
this.x = x;
this.y = y;
this.z = z;
}
}
private static class Matrix3D
{
double m00;
double m01;
double m02;
double m10;
double m11;
double m12;
double m20;
double m21;
double m22;
Matrix3D(
double m00, double m01, double m02,
double m10, double m11, double m12,
double m20, double m21, double m22)
{
this.m00 = m00;
this.m01 = m01;
this.m02 = m02;
this.m10 = m10;
this.m11 = m11;
this.m12 = m12;
this.m20 = m20;
this.m21 = m21;
this.m22 = m22;
}
Matrix3D(Matrix3D m)
{
this.m00 = m.m00;
this.m01 = m.m01;
this.m02 = m.m02;
this.m10 = m.m10;
this.m11 = m.m11;
this.m12 = m.m12;
this.m20 = m.m20;
this.m21 = m.m21;
this.m22 = m.m22;
}
// From http://www.dr-lex.be/random/matrix_inv.html
void invert()
{
double invDet = 1.0 / determinant();
double nm00 = m22 * m11 - m21 * m12;
double nm01 = -(m22 * m01 - m21 * m02);
double nm02 = m12 * m01 - m11 * m02;
double nm10 = -(m22 * m10 - m20 * m12);
double nm11 = m22 * m00 - m20 * m02;
double nm12 = -(m12 * m00 - m10 * m02);
double nm20 = m21 * m10 - m20 * m11;
double nm21 = -(m21 * m00 - m20 * m01);
double nm22 = m11 * m00 - m10 * m01;
m00 = nm00 * invDet;
m01 = nm01 * invDet;
m02 = nm02 * invDet;
m10 = nm10 * invDet;
m11 = nm11 * invDet;
m12 = nm12 * invDet;
m20 = nm20 * invDet;
m21 = nm21 * invDet;
m22 = nm22 * invDet;
}
// From http://www.dr-lex.be/random/matrix_inv.html
double determinant()
{
return
m00 * (m11 * m22 - m12 * m21) +
m01 * (m12 * m20 - m10 * m22) +
m02 * (m10 * m21 - m11 * m20);
}
final void mul(double factor)
{
m00 *= factor;
m01 *= factor;
m02 *= factor;
m10 *= factor;
m11 *= factor;
m12 *= factor;
m20 *= factor;
m21 *= factor;
m22 *= factor;
}
void transform(Point3D p)
{
double x = m00 * p.x + m01 * p.y + m02 * p.z;
double y = m10 * p.x + m11 * p.y + m12 * p.z;
double z = m20 * p.x + m21 * p.y + m22 * p.z;
p.x = x;
p.y = y;
p.z = z;
}
void transform(Point2D pp)
{
Point3D p = new Point3D(pp.getX(), pp.getY(), 1.0);
transform(p);
pp.setLocation(p.x / p.z, p.y / p.z);
}
void mul(Matrix3D m)
{
double nm00 = m00 * m.m00 + m01 * m.m10 + m02 * m.m20;
double nm01 = m00 * m.m01 + m01 * m.m11 + m02 * m.m21;
double nm02 = m00 * m.m02 + m01 * m.m12 + m02 * m.m22;
double nm10 = m10 * m.m00 + m11 * m.m10 + m12 * m.m20;
double nm11 = m10 * m.m01 + m11 * m.m11 + m12 * m.m21;
double nm12 = m10 * m.m02 + m11 * m.m12 + m12 * m.m22;
double nm20 = m20 * m.m00 + m21 * m.m10 + m22 * m.m20;
double nm21 = m20 * m.m01 + m21 * m.m11 + m22 * m.m21;
double nm22 = m20 * m.m02 + m21 * m.m12 + m22 * m.m22;
m00 = nm00;
m01 = nm01;
m02 = nm02;
m10 = nm10;
m11 = nm11;
m12 = nm12;
m20 = nm20;
m21 = nm21;
m22 = nm22;
}
}
}
关于java - 纹理变形,4 点,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/47820921/
我有一个未定义数量的显示上下文,每个都将显示一个纹理。当我调用 glGenTextures 时,我会在所有显示上下文中返回相同的名称。这会起作用吗?即使它们具有相同的名称,它们仍会存储和显示不同的纹理
我在 SVG 中看到过:文本填充是图像而不是颜色;我一直想知道使用 CSS3 是否可以实现这样的事情。 我浏览了整个网络,到目前为止只找到了基本上将图像覆盖在文本上的解决方法(请参阅 this ,这对
我是 WebGL 的新手。 :)我知道顶点数据和纹理不应该经常更新,但是当它们确实发生变化时,首选哪个:- 通过调用 gl.deleteBuffer 销毁先前的缓冲区 (static_draw) 并创
我需要将 GL_RGBA32F 作为内部格式,但我在 OpenGL ES 实现中没有得到它。相反,我只得到 GL_FLOAT 作为纹理数据类型。 OES_texture_float 规范没有说明里面的
当我执行某些几何体的渲染时,我可以在控制台中看到此警告: THREE.WebGLRenderer: Texture is not power of two. Texture.minFilter sho
我正在尝试使用阴影贴图实现阴影,因此我需要将场景渲染到单独的帧缓冲区(纹理)。我无法让它正常工作,因此在剥离我的代码库后,我留下了一组相对简单的指令,这些指令应该将场景渲染到纹理,然后简单地渲染纹理。
我在 XNA 中使用带有自定义着色器的标准 .fbx 导入器。当我使用 BasicEffect 时,.fbx 模型被 UV 正确包裹并且纹理正确。但是,当我使用我的自定义效果时,我必须将纹理作为参数加
如果我创建一个 .PNG 1024 x 1024 的纹理并在中间画一个 124 x 124 的圆,它周围是空的,它使用的 RAM 量是否与我画一个 124 x 的圆一样124 x 124 空间上的 1
我试图在 Android 中绘制一个地球仪,为此我使用了 OpenGL。然而,为了让它更容易理解,我将从制作一个简单的 3D 立方体开始。我使用 Blender 创建我的 3D 对象(立方体),并在我
文本本身的背景图像层是否有任何 JS/CSS 解决方案? 示例 最佳答案 检查这个http://lea.verou.me/2012/05/text-masking-the-standards-way/
非功能代码: if sprite.texture == "texture" { (code) } 当 Sprite 具有特定纹理时,我正在尝试访问 Sprite 的纹理以运行代码。目前纹理仅在我的
我正在尝试学习适用于 iOS 的 SceneKit 并超越基本形状。我对纹理的工作原理有点困惑。在示例项目中,平面是一个网格,并对其应用了平面 png 纹理。你如何“告诉”纹理如何包裹到物体上?在 3
基本上, 这有效: var expl1 = new THREE.ImageUtils.loadTexture( 'images/explodes/expl1.png' ); this.material
我正在尝试将各种场景渲染为一组纹理,每个场景都有自己的纹理到应该绘制的位置...... 问题: 创建 512 个 FBO,每个 FBO 绑定(bind)了 512 个纹理,这有多糟糕。只使用一个 FB
我正在使用文本 protobuf 文件进行系统配置。 我遇到的一个问题是序列化的 protobuf 格式不支持注释。 有没有办法解决? 我说的是文本序列化数据格式,而不是方案定义。 这个问题是有人在某
我想将我的 3D 纹理的初始化从 CPU 移到 GPU。作为测试,我编写了一个着色器将所有体素设置为一个常数值,但纹理根本没有修改。我如何使它工作? 计算着色器: #version 430 layou
我可以像这样用 JavFX 制作一个矩形: Rectangle node2 = RectangleBuilder.create() .x(-100) .
我在 iPhone 上遇到了 openGL 问题,我确信一定有一个简单的解决方案! 当我加载纹理并显示它时,我得到了很多我认为所谓的“色带”,其中颜色,特别是渐变上的颜色,似乎会自动“优化”。 只是为
假设我有一个域类 class Profile{ String name byte[] logo } 和一个 Controller : class ImageController {
我正在开发一款使用 SDL 的 2D 游戏。由于某些系统的 CPU 较弱而 GPU 较强,因此除了普通的 SDL/软件之外,我还有一个使用 OpenGL 的渲染器后端。 渲染器界面的简化版本如下所示:
我是一名优秀的程序员,十分优秀!