我正在尝试在Android中创建频谱图可视化。我使用Android设备以.PCM格式录制声音,然后将其转换为.WAV,以便可以使用musicg库https://code.google.com/p/musicg/进行分析。
使用musicg我可以创建录制的.WAV的频谱图,并且从该频谱图中我可以将频率时域数据提取为double [] []。
接下来我要解决的是如何在Android中可视化这些数据。任何帮助将不胜感激。
在用户3161880回答之后,我尝试了以下操作,但我没有得到任何屏幕。我可以画一个圆圈,如user3161880回答所示。任何想法为什么这不起作用?
private static class SpectrogramView extends View {
private Paint paint = new Paint();
private double [][] data;
public SpectrogramView(Context context, double [][] data) {
super(context);
this.data = data;
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
if (data != null) {
paint.setStrokeWidth(1);
canvas.drawColor(Color.WHITE);
int width = data.length;
int height = data[0].length;
for(int i = 0; i < width; i++) {
for(int j = 0; j < height; j++) {
int value;
value = 255 - (int)(data[i][j] * 255);
paint.setColor(value<<16|value<<8|value|255<<24);
canvas.drawPoint(i, height-1-j, paint);
}
}
} else {
System.err.println("Data Corrupt");
}
//draw circle
/*paint.setColor(Color.RED);
canvas.drawColor(Color.BLUE);
canvas.drawCircle(100.0f, 100.0f, 50.0f, paint);*/
}
}
答案 0 :(得分:1)
比我原来的帖子改进了答案。
private static class SpectrogramView extends View {
private Paint paint = new Paint();
private Bitmap bmp;
public SpectrogramView(Context context, double [][] data) {
super(context);
if (data != null) {
paint.setStrokeWidth(1);
int width = data.length;
int height = data[0].length;
int[] arrayCol = new int[width*height];
int counter = 0;
for(int i = 0; i < height; i++) {
for(int j = 0; j < width; j++) {
int value;
int color;
value = 255 - (int)(data[j][i] * 255);
color = (value<<16|value<<8|value|255<<24);
arrayCol[counter] = color;
counter ++;
}
}
bmp = Bitmap.createBitmap(arrayCol, width, height, Config.ARGB_8888);
} else {
System.err.println("Data Corrupt");
}
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
canvas.drawBitmap(bmp, 0, 100, paint);
}
}
这将通过在绘制到画布之前创建位图来提高性能。
答案 1 :(得分:0)
如果您愿意显示预先录制的音频的静态频谱图,那么简单的View
就足够了。您可以覆盖视图的onDraw
方法。在这里,您将可以访问将在其上绘制背景的Canvas
。
您的显示逻辑是什么,它将double
映射到颜色和时间 - 频率网格到画布的x-y坐标。
public class MainActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(new MyView(this));
}
private static class MyView extends View {
Paint paint = new Paint();
public MyView(Context context) {
super(context);
}
@Override
protected void onDraw(Canvas canvas) {
super.onDraw(canvas);
// draw circle
paint.setColor(Color.RED);
canvas.drawColor(Color.BLUE);
canvas.drawCircle(100.f, 100.f, 50.f, paint);
}
}
}