이렇게 해서 바코드 스캐너 기능을 가진 앱을 만들 수 있는데, 약간 꺼림칙한 부분이 있어요. 단말기에 바코드 스캐너 앱이 여러개 설치되어있는 경우, 어떤걸 열건지 물어보는 창이 뜨게되는데 아마 intent 이름이 같아서 선택하라고 뜨는것 같습니다. 우리는 앱 내부에 있는 클래스(CaptureActivity)를 열기 때문에 다른 앱은 필요가 없죠. 이걸 없애는 방법은 다음과 같습니다.
IntentIntegrator 클래스에서
Intent intentScan = new Intent("com.google.zxing.client.android.SCAN"); 부분을
Intent intentScan = new Intent(activity, CaptureActivity.class); 로 바꾸고
CaptureActivity클래스에서
if (intent != null && action != null) 부분의 else 부분에
모두 설치하셨다면, 프로젝트를 만들어야 할텐데요. 맨땅에다가 만들지 않고 예제를 수정하는 방법으로 만드려고 합니다.
OpenCV 예제 중 OpenCV Tutorial 2 - Mixed Processing 이라는 예제가 있을텐데요, 이 안에서는 CameraBridgeViewBase로 카메라를 다루지만 AutoFocus기능을 사용하기 까다롭더라구요 (Camera객체를 만들어서 start시키려고 하면 CameraBridgeViewBase와 충돌하는 듯 합니다). 하지만 NDK개발이 가능합니다.
그리고 OpenCV Tutorial 3 - Camera Control 이라는 예제는 CameraBridgeViewBase를 상속받은 JavaCameraView를 사용해서 카메라를 다루는데, 이 클래스 안에 mCamera가 Camera 객체이기 때문에 AutoFocus기능을 사용하기 수월합니다. 하지만 NDK개발 구축이 안되어있습니다.
이 두 예제를 합칠건데요, 일단 그냥 Tutorial 2 의 패키지를 지워버리고 Tutorial 3의 패키지를 가져오는 식으로 하셔도 됩니다.
다만 두 프로젝트간에 다른건 조정 해줘야합니다. Menifest파일의 패키지와 액티비티 이름, 옮긴 후 액티비티의 onCreate에서 레이아웃 이름 등등은 알아서 조절하시길 바랍니다.
이렇게 해서 만들어진 프로젝트는 JavaCameraView의 mCamera를 이용한 AutoFocus가 가능하고, JavaCameraView의 onCameraFrame에서 OpenCV영상처리도 가능하며 onCameraFrame에서 native 함수를 호출함으로써 NDK 영상처리 개발도 가능합니다.
Tip : native 함수는 Java_com_abc_core_MainActivity_func() 처럼 생겼는데 맨 처음 Java는 고정이고 다음은 com.abc.core.MainActivity라는 액티비티의 func()함수로 사용하겠다는 의미입니다. 즉 위처럼 선언된 네이티브 함수는 com.abc.core.MainActivity 액티비티에 public native void func(); 이렇게 선언되어있어야 합니다. 오타가 나거나 경로가 틀리면 native 함수를 찾을 수 없다고 출력되며 앱이 종료됩니다.
그리고 패킷 제작은 사용하시는 분 마음대로 규격을 정하셔도 괜찮을거 같아요. 저는 모든 정보를 문자로 변환한 후 알파벳으로 구분했습니다.
"a1.23b2.34c3.45z" 이렇게요. 그러면 전송받은 측에서 a, b사이의 문자("1.23")을 빼 낸 다음에 숫자(1.23)로 변환하는 방법을 사용했습니다.
패킷을 송신 했다면, 수신을 해야할텐데 코드 작성 전에 문득 이런 생각이 들더군요
'한 패킷 전송 속도와 안드로이드 반복문의 속도가 다르면 패킷이 잘릴까?' '패킷이 잘린다면 다음번에 수신된 나머지 패킷과 연결할 순 없을까?'
그래서 고민하다 링 버퍼를 만들었습니다. 패킷보다 긴(2배이상) 충분한 저장공간(배열)을 만들어 놓고 수신한 바이트들을 배열 안에 순서대로
차곡차곡 쌓아 두는거죠. 차례대로 쌓여있다면 그 배열을 검사하면서 내가 정한 패킷에 맞는 데이터가 있다면 빼서 사용하는겁니다.
밑에 소스 첨부 하겠습니다~ 어떤 함수를 사용했는지 참고하시고 자신만의 통신을 구현해 보시길~
private String catchNumber(String start, String end, String str, int state) { // a와 z의 위치를 저장 int pos_a = str.indexOf(start); int pos_z = str.indexOf(end);
if(pos_a != -1 && pos_z != -1) { if(pos_a >= pos_z) { // z가 a보다 앞에 있을 경우 str = str.substring(pos_a, str.length()); // a바로 앞까지 제거
// a와 z의 위치를 저장 pos_a = str.indexOf(start); pos_z = str.indexOf(end); }
밑에 소스는 구글링에서 퍼온건데.. 기억이 안나네요 ㅠㅠ (아시는분 댓글 달아주시면 바로 출처 올리겠습니다~)
4x4 역행렬 연산에 쓰이는 Matrix 클래스입니다.
Matrix.java
public class Matrix { public int row; // 행 전역변수 선언 public int col; // 열 전역변수 선언 public double[][] values; // 2차원 배열 행렬 전역변수 선언
public Matrix(int row, int col) { // 생성자 Matrix 생성 this.row = row; // 행의 값을 받는다. this.col = col; // 열의 값을 받는다. values = new double[row][col];// 전역변수 values에 row값과 col값을 넣어준다. }
class Invers // Invers 메소드 { determi det = new determi(); // 행렬식을 이용하기 위해 불러옴
public double[][] Invers(Matrix mat) { Matrix inv = new Matrix(mat.row, mat.col); // 역행렬을 저장한 행렬 생성 Matrix cofactor = new Matrix(mat.row, mat.col); double de = (double) 1 / det.data(mat);
if (det.data(mat) == 0) // 행렬식이 0일 경우 { System.out.println("역행렬이 있지 않습니다.");// 메세지 툴력 System.exit(0); // 그리고 시스템 exit }
for (int i = 0; i < mat.row; i++) { // 0부터 행 값까지 반복 for (int j = 0; j < mat.col; j++) {// 0부터 열 값까지 반복 cofactor.values[i][j] = Mino(mat, i, j); inv.values[j][i] = de * Math.pow(-1, i + j) * cofactor.values[i][j]; } } return inv.values;// inv를 리턴한다. }
public double Mino(Matrix mat, int r, int c) {// 여인수전개를 위한 메소드 double cof; Matrix minor = new Matrix(mat.row - 1, mat.col - 1); int k = 0; if (k == r) { k++; } // C_ij의 소행렬은 i행j열을 제외하므로 이런 if를 쓴다. for (int i = 0; i < minor.row; i++) // 0부터 행값까지 반복 { int l = 0; // l은 초기값으로 0 if (l == c) { l++; }// l값이 열값과 같다면 l증가 if (k == r) { k++; }// k값이 행값과 같다면 k증가 for (int j = 0; j < minor.col; j++) { if (l == c) { l++; } // 열값과 같아면 l증가 minor.values[i][j] = mat.values[k][l]; l++; } k++; } cof = det.data(minor);// 소행렬을 구해 반환 return cof; } }
static class determi { static int order; static double[][] matrix;
public double data(Matrix mat) // 행렬식을 구할 행렬을 받는다. { matrix = new double[mat.row][mat.col]; order = mat.row; // order는 받은 행렬의 행의 수이다. for (int i = 0; i < mat.row; i++)// i를 행의수만큼 돌리고 { for (int j = 0; j < mat.col; j++) // j는 열의수만큼 돌려서 { matrix[i][j] = mat.values[i][j]; // matrix에 mat과 같은 값을 넣는다. } } return compute(); // 리턴을 compute함수로 시켜준다. }
public static double compute() // 그럼 여기로 오는데, { double check[] = new double[20]; // check란 20칸짜리 배열을 만든다. double det = 0; // det는 초기화하고 int row, col = 0; // row와 col도 초기화선언한다. for (row = 0; row < order; row++) // row는 받은 행의 수 만큼 돌리고 { check[row] = 1; // 이때 check의 행의수만큼의 index를 1로 지정 }
if (order == 1)// 만약 order가 1이면 { det = matrix[0][0]; // det는 a11값이 된다. }
for (row = 0; row < order; row++) // 다시 row를 수만큼 돌림 { check[row] = 0;// row[i]에 0을 집어넣음 det += matrix[row][0] * Math.pow(-1, row) * minor(row, col + 1, check); // det를 1열의 수 곱하기 minor함수에서 받은 값으로 온다 check[row] = 1; // 이렇게 계산 되면 다시 check를 1로 해준다. } return det; // det 값을 반환 }
public static double minor(int Row, int Col, double trans[]) { double result = 0;// 결과값으로 초기값을 0을 준다. int i = 0; if ((order - Col) == 0) { return 1; } for (int row = 0; row < order; row++)// 0부터 { if (trans[row] != 0)// 만약 trans[row]가 0이 아니라면 { trans[row] = 0; // 0을 넣어준다. result += matrix[row][Col] * Math.pow(-1, (double) i) * minor(row, Col + 1, trans); // 여인수전개에 의한 식이다. 재귀함수로 계속 돌리게된다. trans[row] = 1; // trans[row]에 1을 넣어준다. i++;// i 값 증가 } } return result; // result 반환 } }