android:pdfdocument:commandmirroring
commandmirroring
차이
문서의 선택한 두 판 사이의 차이를 보여줍니다.
다음 판 | 이전 판 | ||
android:pdfdocument:commandmirroring [2024/12/17 17:57] – 만듦 이거니맨 | android:pdfdocument:commandmirroring [2024/12/24 19:55] (현재) – 이거니맨 | ||
---|---|---|---|
줄 8: | 줄 8: | ||
^ 본문 역시 벡터로 그려진 PDF ^^ | ^ 본문 역시 벡터로 그려진 PDF ^^ | ||
- | | {{: | + | | {{: |
+ | |||
+ | ===== 명령어 모음 만들기 ===== | ||
+ | |||
+ | ==== 1. 원리 ==== | ||
+ | |||
+ | 지금까지 PDF로 그린 것을 명령어로 저장하여 다시 반복하려는 이유는 페이지가 여러장일 때에는 구역화(헤더와 푸터, 본문)를 할 때에전체 장수를 미리 알 수가 없기 때문이다. 즉, 한번 싹 다 그려야만 전체의 페이지수를 알 수 있기 때문에, 다시 반복하는 것이다. | ||
+ | |||
+ | 그런데 지금까지 PDF를 그릴 떄에는, 정해진 사이즈의 캔버스 위에 특정한 좌표에다가 드로잉을 했다. 그리고 그 드로잉에는 인쇄될 글자들이 포함된다. 따라서 좌표와 명령어, 그리고 문구를 변수로 넘겨주면 명령어의 모음이 된다. | ||
+ | |||
+ | |||
+ | ==== 2. 구조 ==== | ||
+ | |||
+ | 따라서 명령어의 구조를 다음과 같이 만들었다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // 명령어 구조 | ||
+ | data class COMMAND( | ||
+ | val command : String, | ||
+ | val page : Int, | ||
+ | val textFirst : String, | ||
+ | val textSecond : String, | ||
+ | val pos : POS | ||
+ | ) | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==== 3. 명령어 집합 선언 ==== | ||
+ | |||
+ | 다음과 같이 명령어 집합을 선언했다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // 명령어 집합 | ||
+ | val commandList : ArrayList< | ||
+ | var isRecording = true | ||
+ | </ | ||
+ | |||
+ | isRecording을 통하여 명령어를 기록하는 스위치를 만들었다. 이에 대하여는 다음 항목에서 서술하겠다. | ||
+ | |||
+ | ==== 4. isRecording을 통한 제어 ==== | ||
+ | |||
+ | === 가. 제어 함수 만들기 === | ||
+ | |||
+ | isRecording이라는 스위치를 통하여, 명령어를 저장할 때에 현재 상태의 변화(예 : 좌표의 변화)를 적용할지 여부를 고를 수 있게 했다. | ||
+ | |||
+ | isRecording은 다음의 함수로 호출할 것이다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // Recording commannd | ||
+ | private fun recording(recoding : Boolean) { | ||
+ | isRecording = recoding | ||
+ | } | ||
+ | |||
+ | </ | ||
+ | |||
+ | |||
+ | === 나. 내가 했던 일을 기록 === | ||
+ | |||
+ | 다음과 같이 기존 그리기 함수에다가 내가 하는 그리기를 명령어 셋트로 기록하게 했다. | ||
+ | |||
+ | isRecording이 필요한 이유는, 아래에서와 같이 기록할떄의 좌표 다음에도 좌표를 옮길 필요가 있을 수 있기 때문이다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // Draw Title Text | ||
+ | fun titleText(text : String) { | ||
+ | |||
+ | nextPage() | ||
+ | linePlus(50) | ||
+ | canvasList[pageNum].drawText(text, | ||
+ | |||
+ | if (isRecording) { | ||
+ | commandList.add(COMMAND(" | ||
+ | } | ||
+ | |||
+ | linePlus(50) | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | 위의 코드에서는 제목그리기(titleText)를 실행한 후,. 이를 기록한 이후에도 다시 좌표를 세로로 50포인트 내리게 했다. | ||
+ | |||
+ | 그리기 이후의 좌표 설정은 굳이 명령어를 기록할 떄 필요 없는 일이다. | ||
+ | |||
+ | 커맨드는 첫번째 인자로 사용할 컴맨드(=TitleText), | ||
+ | |||
+ | ==== 5. 두번째 PDF 문서를 그리기 ==== | ||
+ | |||
+ | === 가. 새로운 그리기 함수 === | ||
+ | |||
+ | 첫번째 그리기와 동일한 기능을 하는 두번째의 그리기 함수가 필요하다. | ||
+ | |||
+ | 첫번째 그릴 때에는 그리면서 페이지수를 알게 된 것이지만, | ||
+ | |||
+ | <code kotlin> | ||
+ | // Draw Title Finally | ||
+ | fun titleText(text : String, canvas: Canvas, pos : POS) { | ||
+ | |||
+ | linePlus(50) | ||
+ | canvas.drawText(text, | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | 사실 두번째에 그릴 떄에는 첫번쨰 그릴때에 확정한 위치(pos)를 가져와서 canvas에 그리는 것이므로, | ||
+ | |||
+ | === 나. 전체 페이지 중 본문을 그리기 === | ||
+ | |||
+ | [[android: | ||
+ | |||
+ | |||
+ | <code kotlin> | ||
+ | // PDF 페이지 만들기 | ||
+ | fun makePDFDocument() { | ||
+ | |||
+ | recording(false) | ||
+ | for (i in 0 until canvasList.size) { | ||
+ | val newPage: PdfDocument.Page = pdfDocument.startPage(A4) | ||
+ | |||
+ | val canvas = newPage.canvas | ||
+ | |||
+ | drawPageHeader(canvas) | ||
+ | // canvas.drawBitmap(bitmapList[i], | ||
+ | canvas.save() | ||
+ | canvas.translate(marginLeft.toFloat(), | ||
+ | todoList(commandList, | ||
+ | canvas.restore() | ||
+ | drawPageFooter(canvas, | ||
+ | pdfDocument.finishPage(newPage) | ||
+ | |||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | todoList에는 cavnas의 페이지 순서를 인자로 내보내 준다. | ||
+ | |||
+ | |||
+ | === 다. 기록한 명령어를 호출하기 === | ||
+ | |||
+ | 위에서 본문 부분에 그리기로 한 todoList는 다음과 같이 만들었다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // Command Function | ||
+ | fun todoList(commands : ArrayList< | ||
+ | val todo = commands.filter { it.page == page } | ||
+ | todo.forEach() { | ||
+ | when (it.command) { | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | when (it.textSecond) { | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | } | ||
+ | } | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | " | ||
+ | } | ||
+ | |||
+ | } | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | todoList는 제일 처음에 각 페이지별로 filter를 한다. 만약 페이지별로 커맨드를 나누지 않으면, 동일한 페이지, 동일한 좌표에 여러개의 커맨드가 중첩적용될 수 있기 때문이다. 즉, 페이지를 구분하여 특정의 canvas에 적용하는 캔버스가 무엇인지를 명확히 했다. | ||
+ | |||
+ | 그 이후에는 COMMAND배열의 첫번쨰 인자인 command항목에 따라서 특정의 명령어를 실행하도록 하였다. | ||
+ | |||
+ | 이런식으로 저장한 명령어 모음을 다시 실행하도록 하였다. | ||
+ | |||
+ | |||
+ | ===== 고소인과 피고소인 테이블 만들기 ===== | ||
+ | |||
+ | ==== 1. 목적 ==== | ||
+ | |||
+ | 고소인과 피고소인은 한번에 그릴 때에 여러개의 변수가 같이 넘어간다. | ||
+ | |||
+ | 이를테면, | ||
+ | |||
+ | 따라서 이들을 JSON으로 변환하여 위의 COMMAND클래스에 2번쨰 스트링인자에 변수로 넘겨주어야 한다. | ||
+ | |||
+ | |||
+ | ==== 2. JSON 만들기와 가져오기 ==== | ||
+ | |||
+ | === 가. JSON 만들기 === | ||
+ | |||
+ | 다음과 같이 고소인 연락처를 JSON 스트링으로 만드는 함수를 만들었다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // 고소인, 피고소인 연락처 JSON으로 만들기 | ||
+ | fun setJsonContactPlaintiff(contact : PlaintiffContact) : JSONObject { | ||
+ | |||
+ | val jsonContact = JSONObject() | ||
+ | try { | ||
+ | |||
+ | jsonContact.put(" | ||
+ | jsonContact.put(" | ||
+ | jsonContact.put(" | ||
+ | jsonContact.put(" | ||
+ | |||
+ | } catch (e: Exception) { | ||
+ | // TODO Auto-generated catch block | ||
+ | e.printStackTrace() | ||
+ | } | ||
+ | |||
+ | return jsonContact | ||
+ | } | ||
+ | |||
+ | </ | ||
+ | |||
+ | [[android: | ||
+ | |||
+ | |||
+ | === 나. JSON 스트링을 변환하기 === | ||
+ | |||
+ | 다음과 같이 스트링을 가져와서 변환하게 할 수 있다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // 고소인, 피고소인 연락처 가져오기 | ||
+ | fun getJsonContactPlaintiff(jsonObject: | ||
+ | |||
+ | val valueOfContact = PlaintiffContact( | ||
+ | pName = jsonObject.optString(" | ||
+ | socialNumber = jsonObject.optString(" | ||
+ | pAddress = jsonObject.optString(" | ||
+ | pPhoneNumber = jsonObject.optString(" | ||
+ | pPhoto = jsonObject.optString(" | ||
+ | |||
+ | ) | ||
+ | |||
+ | return valueOfContact | ||
+ | |||
+ | } | ||
+ | |||
+ | </ | ||
+ | |||
+ | ==== 3. JSON 보내기 ==== | ||
+ | |||
+ | === 가. 코드 === | ||
+ | |||
+ | 다음과이 고소인, 피고소인 주소록 명단 중에서 한명의 정보를 JSON 형태로 내보낸다. | ||
+ | |||
+ | <code kotlin> | ||
+ | if (isRecording) { | ||
+ | |||
+ | val jsonContact = setJsonContactPlaintiff(it) | ||
+ | commandList.add(COMMAND(" | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | === 나. 전체 코드 === | ||
+ | |||
+ | 이해를 위한 전체코드는 다음과 같다. | ||
+ | |||
+ | <file kotlin " | ||
+ | // 표 : 고소인 | ||
+ | fun tablePlaintiff() { | ||
+ | |||
+ | nextPage() | ||
+ | /** 변수 **/ | ||
+ | val cellHeight = 30 // Row Height : 30 | ||
+ | val columnWidth = 60 // 내부선 | ||
+ | val secondColumn = 260 // 두번쨰 열 | ||
+ | val vcOffset = 4f // 세로 중간을 맞추기 위한 오프셋 | ||
+ | |||
+ | plaintiffList? | ||
+ | |||
+ | nextPage(cellHeight * 3) | ||
+ | // 첫째 행 | ||
+ | // Rect | ||
+ | val rect1 = Rect(currentPOS.X, | ||
+ | canvasList[pageNum].drawRect(rect1, | ||
+ | val rect1head1 = Rect(currentPOS.X, | ||
+ | canvasList[pageNum].drawRect(rect1head1, | ||
+ | val rect1head2 = Rect(currentPOS.X + secondColumn, | ||
+ | canvasList[pageNum].drawRect(rect1head2, | ||
+ | |||
+ | // Text | ||
+ | canvasList[pageNum].drawText(" | ||
+ | canvasList[pageNum].drawText(it.pName, | ||
+ | canvasList[pageNum].drawText(" | ||
+ | canvasList[pageNum].drawText(it.socialNumber, | ||
+ | |||
+ | |||
+ | // 둘재 행 | ||
+ | // Rect | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvasList[pageNum].drawText(" | ||
+ | canvasList[pageNum].drawText(it.pAddress, | ||
+ | |||
+ | // 셋쨰 행 | ||
+ | // Rect | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvasList[pageNum].drawText(" | ||
+ | canvasList[pageNum].drawText(it.pPhoneNumber, | ||
+ | |||
+ | if (isRecording) { | ||
+ | |||
+ | val jsonContact = setJsonContactPlaintiff(it) | ||
+ | commandList.add(COMMAND(" | ||
+ | } | ||
+ | |||
+ | // 아래 여백 | ||
+ | linePlus(cellHeight * 3 + 20) | ||
+ | }?:{ | ||
+ | |||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | canvasList[pageNum].drawText(" | ||
+ | |||
+ | if (isRecording) { | ||
+ | commandList.add(COMMAND(" | ||
+ | } | ||
+ | // 아래 여백 | ||
+ | linefeed() | ||
+ | } | ||
+ | } | ||
+ | |||
+ | </ | ||
+ | |||
+ | |||
+ | ==== 4. JSON 가져오기 ==== | ||
+ | |||
+ | JSON으로 가져온 스트링 데이터를 다시 나누어서 표로 그리는건 다음과 같이 하면 된다. | ||
+ | |||
+ | <file kotlin " | ||
+ | fun tablePlaintiff(canvas: | ||
+ | |||
+ | /** 변수 **/ | ||
+ | val cellHeight = 30 // Row Height : 30 | ||
+ | val columnWidth = 60 // 내부선 | ||
+ | val secondColumn = 260 // 두번쨰 열 | ||
+ | val vcOffset = 4f // 세로 중간을 맞추기 위한 오프셋 | ||
+ | |||
+ | val contact = getJsonContactPlaintiff(JSONObject(jsonObject)) | ||
+ | |||
+ | // 첫째 행 | ||
+ | // Rect | ||
+ | val rect1 = Rect(pos.X, pos.Y, body.Width, pos.Y + cellHeight) | ||
+ | canvas.drawRect(rect1, | ||
+ | val rect1head1 = Rect(pos.X, pos.Y, pos.X + columnWidth, | ||
+ | canvas.drawRect(rect1head1, | ||
+ | val rect1head2 = Rect(pos.X + secondColumn, | ||
+ | canvas.drawRect(rect1head2, | ||
+ | |||
+ | // Text | ||
+ | canvas.drawText(" | ||
+ | canvas.drawText(contact.pName, | ||
+ | canvas.drawText(" | ||
+ | canvas.drawText(contact.socialNumber, | ||
+ | |||
+ | |||
+ | // 둘재 행 | ||
+ | // Rect | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvas.drawText(" | ||
+ | canvas.drawText(contact.pAddress, | ||
+ | |||
+ | // 셋쨰 행 | ||
+ | // Rect | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvas.drawText(" | ||
+ | canvas.drawText(contact.pPhoneNumber, | ||
+ | } | ||
+ | |||
+ | // 명단이 없을 떄 | ||
+ | fun noneContact(canvas: | ||
+ | /** 변수 **/ | ||
+ | val cellHeight = 30 // Row Height : 30 | ||
+ | val vcOffset = 4f // 세로 중간을 맞추기 위한 오프셋 | ||
+ | |||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | canvas.drawText(" | ||
+ | } | ||
+ | |||
+ | </ | ||
+ | |||
+ | |||
+ | ===== 여러개의 정보를 스트링으로 넘기기 ===== | ||
+ | |||
+ | ==== 1. 텍스트 합치기와 나누기 ==== | ||
+ | |||
+ | === 가. 텍스트 합치기 === | ||
+ | |||
+ | 여러 줄의 데이터를 하나의 텍스트로 합치려면 다음과 같이 하면 될 것이다. 이 때 구분자는 세미콜론(; | ||
+ | |||
+ | <code kotlin> | ||
+ | var totalStr = "" | ||
+ | bodyStr.forEach (){ | ||
+ | if (totalStr != "" | ||
+ | totalStr += it | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | === 나. 텍스트 나누기 === | ||
+ | |||
+ | 텍스트를 나누는 함수는 kotlin에서 split함수로 제공한다. 다음과 같이 하면 된다. | ||
+ | |||
+ | <code kotlin> | ||
+ | val wordsBits = bodyStr.split(";" | ||
+ | val line = wordsBits.size | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==== 2. 텍스트 합쳐서 데이터로 전송해주기 ==== | ||
+ | |||
+ | 처음에는 다음과 같이 데이터를 뿌려준 다음 이를, command 모음에 다음과 같이 전송해준다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // MultiLine Table Text | ||
+ | fun tableText(header : String, bodyStr : MutableList< | ||
+ | /** 변수 **/ | ||
+ | val cellHeight = 20 // Row Height : 40 | ||
+ | val columnWidth = 180f // 내부선 | ||
+ | val vcOffset = 4f // 세로 중간을 맞추기 위한 오프셋 | ||
+ | val line = bodyStr.size | ||
+ | nextPage(cellHeight * line + 20) | ||
+ | // Rect | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | canvasList[pageNum].drawRect(currentPOS.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvasList[pageNum].drawText(header, | ||
+ | |||
+ | // 칸 내에 여러줄의 글을 쓰기 | ||
+ | for (i : Int in 0 until line ) { | ||
+ | canvasList[pageNum].drawText(bodyStr[i], | ||
+ | } | ||
+ | |||
+ | var totalStr = "" | ||
+ | bodyStr.forEach (){ | ||
+ | if (totalStr != "" | ||
+ | totalStr += it | ||
+ | } | ||
+ | |||
+ | if (isRecording) { | ||
+ | commandList.add(COMMAND(" | ||
+ | } | ||
+ | |||
+ | |||
+ | // 아래 여백 | ||
+ | totalPOS += POS(0, cellHeight * line + 20) | ||
+ | currentPOS += POS(0, cellHeight * line + 20) | ||
+ | } | ||
+ | </ | ||
+ | |||
+ | |||
+ | ==== 3. 2차 그리기에서 텍스트 나눠서 다시 그리기 ==== | ||
+ | |||
+ | 이렇게 묶음으로 받은 텍스트를 나눠서 다시 뿌려주면 된다. | ||
+ | |||
+ | <code kotlin> | ||
+ | // Table Text 2nd paint | ||
+ | fun tableTextMultiLine(header : String, bodyStr : String, canvas: Canvas, pos: POS) { | ||
+ | /** 변수 **/ | ||
+ | val cellHeight = 20 // Row Height : 40 | ||
+ | val columnWidth = 180f // 내부선 | ||
+ | val vcOffset = 4f // 세로 중간을 맞추기 위한 오프셋 | ||
+ | |||
+ | val wordsBits = bodyStr.split(";" | ||
+ | val line = wordsBits.size | ||
+ | |||
+ | // Rect | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | canvas.drawRect(pos.X.toFloat(), | ||
+ | |||
+ | // Text | ||
+ | canvas.drawText(header, | ||
+ | |||
+ | // 칸 내에 여러줄의 글을 쓰기 | ||
+ | for (i : Int in 0 until line ) { | ||
+ | canvas.drawText(wordsBits[i], | ||
+ | } | ||
+ | |||
+ | } | ||
+ | </ | ||
+ | |||
+ | ===== 최종 결과물 ===== | ||
+ | |||
+ | 이렇게 PDF 2차로 그리면 본문과 머릿말 꼬릿말을 모두 벡터형식으로 그릴 수가 있다. | ||
+ | |||
+ | 최종 결과물은 다음과 같다. 실제 PDF로 파일을 다운 받아 보면, 스케일을 키워도 글자가 깨지지 않고 자연스럽게 크기가 커지는 것을 볼 수 있다. | ||
+ | |||
+ | ^ 벡터로 저장된 PDF 문서 ^^ | ||
+ | | {{android: |
android/pdfdocument/commandmirroring.1734425821.txt.gz · 마지막으로 수정됨: 2024/12/17 17:57 저자 이거니맨