使用musicg库在android中创建频谱图

时间:2014-07-10 14:56:55

标签: java android audio

我正在尝试在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);*/
        }
    }

2 个答案:

答案 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);
        }

    }
}