Android开发之在软件内录屏录画面,直接CV就能用

关于Android开发录屏的资料,真的是很难找到那种拿来就能用的,不是付费专栏就是要积分才能下载源码,起不到任何帮助。我需要的是录制软件画面,找到的几个例子都是录制时返回桌面,然后弹出小窗口开始录制桌面,而且源码内容都太多了很难get到录屏重点。没办法最后只好去GitHub找找有没有简单点的源码,可算是找到一个简单易用的,随机改改就能实现需求,这里整理下分享给大家,希望看了可以少走弯路。

public class MainActivity extends AppCompatActivity {

    private static final String TAG = "MainActivity";
    private static final int PERMISSION_CODE = 1;
    private int mScreenDensity;
    private MediaProjectionManager mProjectionManager;
    private static final int DISPLAY_WIDTH = 480;
    private static final int DISPLAY_HEIGHT = 640;
    private MediaProjection mMediaProjection;
    private VirtualDisplay mVirtualDisplay;
    private MediaProjectionCallback mMediaProjectionCallback;
    private MediaRecorder mMediaRecorder;
    private Button btnRecord,btnReady;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        DisplayMetrics metrics = new DisplayMetrics();
        getWindowManager().getDefaultDisplay().getMetrics(metrics);
        mScreenDensity = metrics.densityDpi;

        initRecorder();
        prepareRecorder();

        mProjectionManager = (MediaProjectionManager) getSystemService
                (Context.MEDIA_PROJECTION_SERVICE);

        btnReady = findViewById(R.id.btnReady);
        btnReady.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                shareScreen();//会打开新页面
                btnReady.setText("录制准备就绪");
            }
        });
		btnRecord = findViewById(R.id.btnRecord);
        btnRecord.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View view) {
                if(btnRecord.getText().equals("开始录制")) {
                    btnRecord.setText("停止录制");

                    if (mVirtualDisplay == null) {
                        mVirtualDisplay = createVirtualDisplay();
                    }
                    mMediaRecorder.start();
                }
                else{
                    btnRecord.setText("开始录制");
                    mMediaRecorder.stop();
                    mMediaRecorder.reset();
                }
            }
        });
        mMediaProjectionCallback = new MediaProjectionCallback();
    }

    @Override
    public void onDestroy() {
        super.onDestroy();
        if (mMediaProjection != null) {
            mMediaProjection.stop();
            mMediaProjection = null;
        }
    }

    @Override
    public void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (requestCode != PERMISSION_CODE) {
            Log.e(TAG, "Unknown request code: " + requestCode);
            return;
        }
        if (resultCode != RESULT_OK) {
            Toast.makeText(this,
                    "Screen Cast Permission Denied", Toast.LENGTH_SHORT).show();
            return;
        }
        mMediaProjection = mProjectionManager.getMediaProjection(resultCode, data);
        mMediaProjection.registerCallback(mMediaProjectionCallback, null);
        mVirtualDisplay = createVirtualDisplay();
        mMediaRecorder.start();
    }

    private void shareScreen() {
        if (mMediaProjection == null) {
            startActivityForResult(mProjectionManager.createScreenCaptureIntent(), PERMISSION_CODE);
            return;
        }
    }

    private void stopScreenSharing() {
        if (mVirtualDisplay == null) {
            return;
        }
        mVirtualDisplay.release();
        //mMediaRecorder.release();
    }

    private VirtualDisplay createVirtualDisplay() {
        return mMediaProjection.createVirtualDisplay("MainActivity",
                DISPLAY_WIDTH, DISPLAY_HEIGHT, mScreenDensity,
                DisplayManager.VIRTUAL_DISPLAY_FLAG_AUTO_MIRROR,
                mMediaRecorder.getSurface(), null /*Callbacks*/, null /*Handler*/);
    }

    private class MediaProjectionCallback extends MediaProjection.Callback {
        @Override
        public void onStop() {
            if (btnRecord.getText().equals("停止录制")) {
                mMediaRecorder.stop();
                mMediaRecorder.reset();
                Log.v(TAG, "Recording Stopped");
            }
            mMediaProjection = null;
            stopScreenSharing();
            Log.i(TAG, "MediaProjection Stopped");
        }
    }

    private void prepareRecorder() {
        try {
            mMediaRecorder.prepare();
        } catch (IllegalStateException | IOException e) {
            e.printStackTrace();
            finish();
        }
    }

    public String getFilePath() {
        final String directory = Environment.getExternalStorageDirectory() + File.separator + "Recordings";
        if (!Environment.MEDIA_MOUNTED.equals(Environment.getExternalStorageState())) {
            Toast.makeText(this, "Failed to get External Storage", Toast.LENGTH_SHORT).show();
            return null;
        }
        final File folder = new File(directory);
        boolean success = true;
        if (!folder.exists()) {
            success = folder.mkdir();
        }
        String filePath;
        if (success) {
            String videoName = ("capture_" + getCurSysDate() + ".mp4");
            filePath = directory + File.separator + videoName;
        } else {
            Toast.makeText(this, "Failed to create Recordings directory", Toast.LENGTH_SHORT).show();
            return null;
        }
        return filePath;
    }

    public String getCurSysDate() {
        return new SimpleDateFormat("yyyy-MM-dd_HH-mm-ss").format(new Date());
    }

    private void initRecorder() {
        if (mMediaRecorder == null) {
            mMediaRecorder = new MediaRecorder();
            mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
            mMediaRecorder.setVideoSource(MediaRecorder.VideoSource.SURFACE);
            mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.MPEG_4);
            mMediaRecorder.setVideoEncoder(MediaRecorder.VideoEncoder.H264);
            mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);
            mMediaRecorder.setVideoEncodingBitRate(10 * 1024 * 1024);
            mMediaRecorder.setVideoFrameRate(60);
            mMediaRecorder.setVideoSize(DISPLAY_WIDTH, DISPLAY_HEIGHT);
            mMediaRecorder.setOutputFile(getFilePath());
        }
    }
}

只需要自己做两个按钮,就可以完成录制功能了。之所以需要两个按钮是因为准备录制时会重新打开页面,屏幕可能会顿一下或者闪一下,如果准备后直接录制,会把屏幕闪一下的画面也录下,所以这里放两个按钮,准备好了再录制,确保能录到自己想要的画面。

### 如何在制抖音直播时保存带有礼物特效的视频 为了实现在制抖音直播过程中保留礼物特效的功能,可以从技术角度考虑几个方面: 对于直播平台而言,在实现这一功能的过程中涉及到多个模块之间的协作。具体来说,推流端负责采集、美颜处理、编码以及推流;服务端则承担转码、制、截图及内容审核的任务;播放器完成拉流、解码与渲染工作;而互动系统支持聊天室交流、礼物赠送和点赞等功能[^4]。 当谈及礼物特效的具体呈现方式时,这类视觉增强通常是在客户端即时生成并叠加于原始视频画面上显示给观众看,并不会直接嵌入到实际传输的数据流之中。因此如果想要确保能够捕捉到这些动态效果,则需特别注意以下几点: #### 客户端本地合成方案 一种可行的做法是在应用程序内部先将礼物动画等附加元素绘制好之后再与其他媒体数据一起打包发送出去之前就做好最终版本的画面输出准备。这意味着所有的特效都已经被融合进了每一帧图像里边去了,从而使得任何形式上的幕记工具都能如实反映所见即所得的结果。 #### 使用官方API或SDK特性 部分成熟的直播解决方案提供商可能会提供专门用于控制像行为的应用编程接口(API)或者是软件开发包(SDK),允许开发者自定义哪些组件应该被纳入最后形成的文件当中去。利用这样的机制可以帮助更精确地管理整个过程而不必担心遗漏任何重要的组成部分。 #### 制方法的选择 考虑到不同操作系统下可用的技术手段有所差异,选择合适的策略至关重要。例如,在Android平台上可以借助MediaProjection API来获取包含UI层在内的全内容副本;而在iOS环境中可能就需要依靠ReplayKit框架或者其他类似的途径达成目的。 ```python import cv2 from datetime import datetime def record_screen_with_effects(output_path='output.mp4', duration=60): fourcc = cv2.VideoWriter_fourcc(*'XVID') out = cv2.VideoWriter(output_path, fourcc, 20.0, (1920, 1080)) start_time = datetime.now() while True: elapsed_seconds = (datetime.now() - start_time).total_seconds() if int(elapsed_seconds) >= duration: break # Capture frame-by-frame with effects applied here. frame = capture_frame_including_gifts_and_other_overlays() # Write the frame into video file. out.write(frame) out.release() ``` 上述代码片段展示了如何创建一段固定时长内持续抓取含有特定覆盖物(比如礼物图标)的连续画面序列,并将其保存成标准格式的多媒体资源文件的过程。需要注意的是`capture_frame_including_gifts_and_other_overlays()`函数代表了一个假设性的操作,它应当由具体的业务逻辑决定怎样从当前活动窗口提取经过装饰后的像素信息。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值