안 드 로 이 드 는 녹화 스크린 의 작은 기능 을 실현 한다.
사고의 방향
안 드 로 이 드 가 스크린 기능 을 실현 하 는 것 은 두 가지 방안 이 있 는데 하 나 는 안 드 로 이 드 가 자체 적 으로 가지 고 있 는 MediaProjectionManager 를 직접 사용 하여 스크린 기능 을 실현 하 는 것 이다.두 번 째 는 음성 만 녹음 하고 사용자 의 조작 은 특정한 방식 으로 기록 하고 저장 하 며 마지막 으로 특정한 협 의 를 통 해 재생 하 는 것 이다.
두 가지 방안 은 각각 장단 점 이 있 는데 전 자 는 실현 방식 이 간단 하지만 특정한 지역 의 화면 만 녹화 할 수 없고 생 성 된 영상 파일 이 비교적 크다.후 자 는 비교적 번 거 로 운 것 을 실현 합 니 다.오디 오 녹음 안 드 로 이 드 7.0 전에 일시 정지 방법 이 없 으 면 여러 개의 파일 만 생 성 한 다음 에 오디 오 를 합성 할 수 있 습 니 다.사용자 의 작업 은 스스로 저장 하고 재생 할 때 복원 해 야 합 니 다.재생 기 는 사용자 정의 생 성 이 필요 합 니 다.그러나 후자 의 장점 은 확장 성 이 높 고 특정 지역 의 녹 화 를 지원 하 며 생 성 된 오디 오 파일 이 비교적 작다 는 것 이다.
수요
화판 을 녹화 하고 화판 은 색깔 의 굵기 를 바 꿀 수 있 고 지 울 수 있 도록 요구한다.화판 밑부분 은 화이트 보드,그림 일 수 있 습 니 다.사진 요 구 는 카메라 촬영 이나 로 컬 사진 이다.녹화 내용 재생 가능;업로드 가 필요 하기 때문에 파일 이 작 아야 합 니 다.모든 것 은 두 번 째 방식 만 선택 할 수 있 습 니 다.
github 주소
전체 항목 은 하나의 폴 더 를 생 성 합 니 다.폴 더 에는 MP3 파일,cw 프로 토 콜 파일(사용자 의 작업 저장),그림 이 포함 되 어 있 습 니 다.전체 화판 은 recyclerView 이 며,item 에는 낙서 화판,그림 컨트롤 이 포함 되 어 있 습 니 다.재생 할 때 cw 프로 토 콜 파일 을 읽 고 시간 에 따라 하나씩 그립 니 다.프로 토 콜 내용 은 화판 각 페이지 의 내용 이 공백 화판 인지 그림 인지,시간,작업(그림/선 전환)을 포함 합 니 다.
오디 오
//
if (mMediaRecorder == null) {
mMediaRecorder = new MediaRecorder();
}
mMediaRecorder.setAudioSource(MediaRecorder.AudioSource.MIC);
mMediaRecorder.setOutputFormat(MediaRecorder.OutputFormat.AMR_NB);
mMediaRecorder.setOutputFile(mRecordFilePath);
mMediaRecorder.setAudioEncoder(MediaRecorder.AudioEncoder.AMR_NB);//amr_nb 6
try {
mMediaRecorder.prepare();
mMediaRecorder.start();
isRunning = true;
AudioUtil.startAudio();
mHandler.sendEmptyMessageDelayed(MSG_TYPE_COUNT_DOWN, 1000);
} catch (IOException e) {
e.printStackTrace();
}
/**
* amr_nb
* @param partsPaths
* @param unitedFilePath
*/
public static void uniteAMRFile(List<String> partsPaths, String unitedFilePath) {
try {
File unitedFile = new File(unitedFilePath);
FileOutputStream fos = new FileOutputStream(unitedFile);
RandomAccessFile ra = null;
for (int i = 0; i < partsPaths.size(); i++) {
ra = new RandomAccessFile(partsPaths.get(i), "rw");
if (i != 0) {
ra.seek(6);
}
byte[] buffer = new byte[1024 * 8];
int len = 0;
while ((len = ra.read(buffer)) != -1) {
fos.write(buffer,0,len);
}
File file = new File(partsPaths.get(i));
if(file.exists()){
file.delete();
}
}
if(ra!=null){
ra.close();
}
fos.close();
} catch (Exception e) {
e.printStackTrace();
}
}
오디 오 재생
mediaPlayer = new MediaPlayer();
mediaPlayer.setDataSource(path);
mediaPlayer.prepare();
mediaPlayer.start();
recyclerView미끄럼 금지 여부
public class ForbitLayoutManager extends LinearLayoutManager {
private boolean canScrollHorizon = true;
private boolean canScrollVertical = true;
public ForbitLayoutManager(Context context) {
super(context);
}
public ForbitLayoutManager(Context context, int orientation, boolean reverseLayout) {
super(context, orientation, reverseLayout);
}
public ForbitLayoutManager(Context context, AttributeSet attrs, int defStyleAttr, int defStyleRes) {
super(context, attrs, defStyleAttr, defStyleRes);
}
public void setCanScrollHorizon(boolean canScrollHorizon) {
this.canScrollHorizon = canScrollHorizon;
}
public void setCanScrollVertical(boolean canScrollVertical) {
this.canScrollVertical = canScrollVertical;
}
@Override
public boolean canScrollHorizontally() {
return canScrollHorizon && super.canScrollHorizontally();
}
@Override
public boolean canScrollVertically() {
return canScrollVertical && super.canScrollVertically();
}
}
미 끄 러 질 때 view Page 와 유사 한 페이지 만 미 끄 러 집 니 다.
mPagerSnapHelper = new PagerSnapHelper();
mPagerSnapHelper.attachToRecyclerView(recyclerView);
현재 몇 페이지,viewPage 와 유사 한 페이지 Select 가 져 오기
public class RecyclerViewPageChangeListenerHelper extends RecyclerView.OnScrollListener {
private SnapHelper snapHelper;
private OnPageChangeListener onPageChangeListener;
private int oldPosition = -1;// Position
public RecyclerViewPageChangeListenerHelper(SnapHelper snapHelper, OnPageChangeListener onPageChangeListener) {
this.snapHelper = snapHelper;
this.onPageChangeListener = onPageChangeListener;
}
@Override
public void onScrolled(RecyclerView recyclerView, int dx, int dy) {
super.onScrolled(recyclerView, dx, dy);
if (onPageChangeListener != null) {
onPageChangeListener.onScrolled(recyclerView, dx, dy);
}
}
@Override
public void onScrollStateChanged(RecyclerView recyclerView, int newState) {
super.onScrollStateChanged(recyclerView, newState);
int position = 0;
RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
// itemView
View view = snapHelper.findSnapView(layoutManager);
if (view != null) {
// itemView position
position = layoutManager.getPosition(view);
}
if (onPageChangeListener != null) {
onPageChangeListener.onScrollStateChanged(recyclerView, newState);
//newState == RecyclerView.SCROLL_STATE_IDLE
if (newState == RecyclerView.SCROLL_STATE_IDLE && oldPosition != position) {
oldPosition = position;
onPageChangeListener.onPageSelected(position);
}
}
}
public interface OnPageChangeListener {
void onScrollStateChanged(RecyclerView recyclerView, int newState);
void onScrolled(RecyclerView recyclerView, int dx, int dy);
void onPageSelected(int position);
}
}
현재 선택 한 아 이 템 가 져 오기(보기 페이지 아 이 템 만 가 져 오기)
View view = forbitLayoutManager.findViewByPosition(position);
// null, ,
recyclerView.getViewTreeObserver().addOnGlobalLayoutListener(new ViewTreeObserver
.OnGlobalLayoutListener() {
@Override
public void onGlobalLayout() {
// ,
recyclerView.getViewTreeObserver().removeOnGlobalLayoutListener(this);
}
});
시간 에 따라 재생
private void convertCWACT(CW cw, int seconds,boolean isSeek) {
List<CWACT> cwacts = cw.getACT();
// , , item ,
if(isSeek){
position =0;
forbitLayoutManager.scrollToPosition(position);
forbitLayoutManager.setStackFromEnd(true);
for(int i=0;i<recyclerViewList.size();i++){
View view = recyclerViewList.get(i);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.clear();
}
}
}
for (CWACT cwact : cwacts) {
int time = cwact.getTime();
if(isSeek?time > seconds:time != seconds){
continue;
}
if ("switch".equals(cwact.getAction())) {//
position = cwact.getCwSwitch().getIndex();
forbitLayoutManager.scrollToPosition(position);
forbitLayoutManager.setStackFromEnd(true);
} else if ("line".equals(cwact.getAction())) {//
if(position>recyclerViewList.size()-1){
continue;
}
View view = recyclerViewList.get(position);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.setDrawPath(cwact.getLine());
}
} else if ("clear".equals(cwact.getAction())) {//
if(position>recyclerViewList.size()-1){
continue;
}
View view = recyclerViewList.get(position);
if(view!=null){
SimpleDoodleView doodleView = view.findViewById(R.id.doodleView);
doodleView.clear();
}
}
}
}
이상 이 바로 본 고의 모든 내용 입 니 다.여러분 의 학습 에 도움 이 되 고 저 희 를 많이 응원 해 주 셨 으 면 좋 겠 습 니 다.
이 내용에 흥미가 있습니까?
현재 기사가 여러분의 문제를 해결하지 못하는 경우 AI 엔진은 머신러닝 분석(스마트 모델이 방금 만들어져 부정확한 경우가 있을 수 있음)을 통해 가장 유사한 기사를 추천합니다:
Kotlin의 기초 - 2부지난 글에서는 Kotlin이 무엇인지, Kotlin의 특징, Kotlin에서 변수 및 데이터 유형을 선언하는 방법과 같은 Kotlin의 기본 개념에 대해 배웠습니다. 유형 변환은 데이터 변수의 한 유형을 다른 데이터...
텍스트를 자유롭게 공유하거나 복사할 수 있습니다.하지만 이 문서의 URL은 참조 URL로 남겨 두십시오.
CC BY-SA 2.5, CC BY-SA 3.0 및 CC BY-SA 4.0에 따라 라이센스가 부여됩니다.