이 글을 보고 저도 따라해봤습니다:
https://www.clien.net/service/board/park/18025380
-> https://forum.samygo.tv/viewtopic.php?t=14101

# 적용한 모델

21년형 75UA8000 (그냥 4K LCD 모델)

# 참고사항

가이드 원문을 보니, 기본앱이 알아서 재설치될 수 있으며 이를 예방하려면 "TV 앱 관련 웹사이트 도메인"을 차단하라더군요. 필요하다고 하면 공유기에서 차단해야 할 것 같은데요, 아직은 괜찮으니 넘어갑니다.

리모컨에 "프라임비디오" 버튼 대신 외부입력(전환) 버튼이나 넣어줬으면 하는 마음인데요, "프라임비디오" 앱을 삭제한 뒤에는 해당 버튼을 누르면 앱스토어의 앱 정보 페이지로 갑니다.. 버튼 리매핑은 (그 버튼 만드는 데 아마존-삼성 간의 계약이 있을 것 같으니) 안 해줄 것 같고, 가짜 앱이 실행되도록 해서 해당 버튼의 기능을 바꿀 수 있으면 좋겠네요.

# 하면 좋아지나......?

성능 개선은 잘... 모르겠고요, 그냥 깔끔하니 기분이 좋습니다.

# 절차

## 1. TV의 개발자모드 활성화

 - 웹 검색해보시면 익히 나오는 방법입니다

 - 공식 가이드: https://developer.samsung.com/smarttv/develop/getting-started/using-sdk/tv-device.html

1. TV에서 "앱(스토어)" 앱으로 진입하고, (가상)번호키로 PIN번호 "12345"를 입력하여 "개발자 모드" 대화상자를 띄웁니다

 - 저는 이게 잘 안 되었는데요, 여러 번 TV를 껐다 켜보면서 시험하니 어느 순간 되기는 하였습니다
 - 리모컨 전원 버튼을 길게 (2초 이상) 누르면 재시작됩니다 ("삼성 스마트TV" 로고가 뜸)
 - "전원 플러그를 아예 빼고 몇 분 있다가 다시 꽂아라, USB키보드를 연결해서 번호를 입력해봐라, PIN번호를 "00000"으로 입력해봐라"와 같은 조언이 있었으나, 그냥 껐다 켜기 반복이 어느 순간 먹혔습니다

2. "개발자 모드" 대화상자에서 "개발자 모드"를 켠 뒤 TV에 연결할 PC의 IP주소를 적어주고 "확인"을 선택합니다.

- PC의  IP 주소는 윈도 PC면 명령창에서 "ipconfig" 명령을 쓰든, 네트워크 설정창에서 확인하세요
- PC IP 주소를 잘못 쓰면 어떻게 될까요...?

3. 그러면 "TV를 재시작하면 개발자 모드가 활성화됩니다"하는 안내창이 뜹니다. 리모컨 전원 버튼을 길게 눌러 TV를 재시작해줍니다.

## 2. PC에 Tizen SDK (Tizen Studio) 설치

윈도 PC 기준입니다

- 공식 다운로드 페이지: https://developer.tizen.org/development/tizen-studio/download

저는 IDE가 필요없기 때문에 용량 작은 "CLI" 버전을 골랐습니다. 기본 경로에 설치하면 "C:\tizen-studio"에 깔리며, 그러면 작업에 필요한 "sdb" 프로그램의 경로는 "C:\tizen-studio\tools\sdb.exe"가 됩니다

## 3. TV의 IP 주소 확인

TV의 [설정 > 일반 > 네트워크 > 네트워크 정보]에서 확인했습니다. 공유기 설정 페이지나 다른 경로에서도 확인할 수 있겠습니다.

## 4. PC에서 TV에 연결하고, 원하는 앱 삭제

윈도 PC 기준입니다

1. 명령줄 창을 띄우고, "C:\tizen-studio\tools" 경로로 갑니다 (또는 SDK를 설치한 경로). 여기에 "sdb" 프로그램이 있습니다.

 - TMI로 sdb는 "Smart Device Bridge"의 약자라고 합니다. Android SDK의 ADB가 떠오르네요..

2. "sdb connect <TV IP 주소>" 명령으로 TV에 연결합니다

3. "sdb devices" 명령으로 TV에 연결되었는지 확인합니다

4. "sdb shell 0 vd_applist" 명령으로 설치된 앱 명단을 얻습니다. 아래와 같은 출력이 뜹니다

get app list...
apps count: 18
---------------------------------------------------------------------------------------------
--------------app_id                                =3201907018807-------------
--------------app_index                             =1-------------
--------------app_title                             =Netflix-------------
--------------app_installed_path                    =/opt/usr/home/owner/apps_rw/org.tizen.netflix-app/-------------
--------------app_panel_icon_path                   =/opt/share/webappservice/apps_icon/all/3201907018807/-------------
--------------app_version                           =5.3.27-------------
--------------install_date                          =2021-11-13 14:04:59-------------
--------------app_tizen_id                          =org.tizen.netflix-app-------------
--------------app_package_name                      =org.tizen.netflix-app-------------
--------------app_widget_category                   =-------------
--------------app_runtitle                          =Netflix-------------
--------------type                                  =normal-------------
--------------app_size                              =19977282-------------
--------------use_count                             =591-------------
--------------app_featured(1:myapp,2:recommended)   =2-------------
--------------app_type                              =2-------------
--------------installState                          =3-------------
--------------is_hidden                             =0-------------
--------------is_support_voice                      =0-------------
--------------is_support_gesture                    =0-------------
--------------is_network                            =1-------------
--------------is_removable                          =1-------------
--------------is_lock                               =0-------------
--------------is_need_login                         =0-------------
--------------is_need_update                        =0-------------
--------------installed_source_type                 =1-------------
--------------is_updated                            =0-------------
--------------is_ticker                             =0-------------
--------------is_multi_screen2                      =0-------------
--------------is_multi_tasking                      =1-------------
--------------is_usb_app                            =0-------------
--------------is_age_auth                           =0-------------
--------------app_age_limit                         =0-------------
--------------gamepad                               =none-------------
--------------is_need_uninstall                     =0-------------
--------------firstscreen_icon_dir                  =/opt/share/webappservice/apps_icon/all/3201907018807/-------------
--------------ambient_screen_support                =-------------
--------------icon_mutlilang_support                =0-------------
--------------ambient_icon_path                     =/opt/share/webappservice/apps_icon/all/3201907018807/-------------
---------------------------------------------------------------------------------------------
...후략....

  - "sdb shell 0 vd_applist > apps.txt" 명령과 같은 식으로 출력을 텍스트 파일에 써넣으면 확인하기 좋습니다
  - 아래와 같은 앱 ID에 관심이 갔습니다. 지워도 되는지는 모르겠네요. 특히 "e-manual"은 지우기 조심스럽습니다

3201910019365 prime_video
3201612011352 watcha
111399000128 wavve
3201901017752 tving
3201909019211 samsung_health
3202107024558 samsung_chat
3201702011788 bugs
3201604009105 melon
3202203026799 xboxgamepass
3202004020674 google_assistant
3202101022815 coupangplay
3201901017640 disneyplus
3201807016597 appletv (aria-video)
20212100001 e-manual

5. "sdb shell 0 vd_appuninstall <app_id>" 명령으로 원하는 앱을 하나씩 지웁니다

uninstall 3201910019365
app_id[3201910019365] uninstall start
app_id[3201910019365] uninstalling[3]
app_id[3201910019365] uninstalling[7]
app_id[3201910019365] uninstalling[11]
app_id[3201910019365] uninstalling[15]
app_id[3201910019365] uninstalling[19]
app_id[3201910019365] uninstalling[23]
app_id[3201910019365] uninstalling[26]
app_id[3201910019365] uninstalling[30]
app_id[3201910019365] uninstalling[34]
app_id[3201910019365] uninstalling[38]
app_id[3201910019365] uninstalling[42]
app_id[3201910019365] uninstalling[46]
app_id[3201910019365] uninstalling[50]
app_id[3201910019365] uninstalling[53]
app_id[3201910019365] uninstalling[57]
app_id[3201910019365] uninstalling[61]
app_id[3201910019365] uninstalling[65]
app_id[3201910019365] uninstalling[69]
app_id[3201910019365] uninstalling[73]
app_id[3201910019365] uninstalling[76]
app_id[3201910019365] uninstalling[80]
app_id[3201910019365] uninstalling[84]
app_id[3201910019365] uninstalling[88]
app_id[3201910019365] uninstalling[92]
app_id[3201910019365] uninstalling[96]
app_id[3201910019365] uninstalling[100]
app_id[3201910019365] uninstall completed
spend time for wascmd is [1596]ms

끝!

반응형
# 네이버카페 엘지모바일 사용자 카페를 보시면, 다른 분께서 아예 폰트 APK 생성기를 공유하셨더군요.
그 쪽을 사용해보세요.

지난 글 "LG폰 폰트 패키지 직접 만들기는 잘 안 됨"에 이어서,

그동안 LG폰에 사용자 폰트를 넣으려는 시도는 다들 폰트 정보 파일의 벽을 넘기가 어려웠습니다. 물론 별로 관심없는 쪽이 가깝긴 한 것 같지요. 스토어의 유/무료 폰트 패키지에서 폰트 파일을 열어서 글리프만 바꿔치기하는 수법이 있었으나 번거로웠죠.

그런데 요새 오픈소스 분석도구가 워낙 좋아서 실력없는 목수인 저도 노오력하면 분석이 가능했네요.
(포맷 분석 결과로부터,) 공개 폰트로 만든 폰트 패키지를 올려드려봅니다. 한 번 써보세요!

※ 폰트별로 링크된 APK 파일을 휴대폰에서 다운로드 및 설치 후, 휴대폰 폰트 설정에서 고르세요.

1. 리디바탕체
전자책 서점사 리디북스에서 만든 "리디바탕체" 사용자 폰트입니다.
보기 좋고 읽기 좋은 바탕체 글꼴이 자랑입니다.

※ 원래 버전은 가늘다고 느껴서 FontForge로 조금 굵게 고친 버전입니다.

RidiBatang_LGFont.apk
1.20MB

※ 폰트 배포처 링크: https://www.ridicorp.com/branding/fonts/ridibatang/

리디바탕 글꼴은 개인 및 기업 사용자를 포함한 모든 사용자에게 무료로 제공되며 자유롭게 수정하고 재배포하실 수 있습니다. 단, 글꼴 자체를 유료로 판매하는 것은 금지하며, 리디바탕 글꼴은 본 저작권 안내와 라이선스 전문을 포함해서 다른 소프트웨어와 번들하거나 재배포 또는 판매가 가능합니다. 라이선스 전문을 포함하기 어려울 경우, 리디바탕 글꼴의 출처 표기를 권장합니다. 예) 이 페이지는 리디주식회사에서 제공한 리디바탕 글꼴이 사용되어 있습니다.

This Font Software is licensed under the SIL Open Font License, Version 1.1.
리디바탕의 저작권은 리디주식회사가 소유하고 있습니다.

 

2. Neo둥근모 (v1.41 (19.12.19))

Neo둥근모는 90년대에 퍼블릭 도메인으로 공개된 비트맵 한글 글꼴 둥근모꼴을 트루타입 형식으로 변환하고, 여기에 원래 글꼴에는 없던 여러가지 기호를 추가함으로써 여러 환경에서 유용하게 쓰일 수 있는 글꼴을 만드는 프로젝트입니다.

NeoDGM_v1.41_LGFont.apk
0.14MB

※ 폰트 배포처 링크: https://neodgm.dalgona.dev

Neo둥근모는 SIL Open Font License 1.1 하에 배포됩니다. 라이선스의 전체 내용은 GitHub에서 보실 수 있습니다.

 

반응형

이전 글에 이어서,

(추가) SSH/셸스크립트 실행을 지원하는 커스텀 펌웨어인 경우!

번거로운 SCP 파일 복사 작업 없이도 간편하게 안내 음성 파일을 바꿀 수 있다 하여 소개해 봅니다.
나온지 오래되어 보이는데 이제야 알게 되었네요.

덤: 중국어 vs. 글로벌(영문) 펌웨어 차이 (뇌피셜)

공통으로 안 되는 것

  • GPS 신호로 현재 시간 설정하기 (시간 자동 동기화 방법은 휴대폰 연동 뿐입니다)

중국어판 특징

  • 타임랩스 주차모드 지원
    • 감지되는 동작, 충격 없다면 1fps 저속 촬영으로 메모리 공간, 전력 절감 (국내 기기에도 다 있는 기능이죠)
    • ACC/상시전원을 구분할 전용 전원 케이블이 필요합니다
  • GPS 좌표 기록조차 안 됨
  • WDR 기능 v1.0.8에서 글로벌판에도 메뉴가 추가된 것 같습니다 아직 못 써봐서..
    • (주: 주야간을 고려한 자동 화면 대비 조절 기능인 것 같습니다)
중국어 버전에서만 되는 기능이 있어서, 4PDA 포럼 자료를 보시면 "중국어 펌웨어 + 영문 화면+음성" 짜깁기한 펌웨어가 배포되고 있습니다.

글로벌판 특징

  • GPS 좌표가 영상 파일에 메타데이터 형태로 기록됨

덤: 20년 3월 현재 최신 v1.0.8 영문판 커스텀 펌웨어 (불펌)

기왕 작업하는 김에 최신 펌웨어이면 좋겠죠.

러시아 4PDA 포럼(Shkiper-vr6님)에서 무단 불펌한... v1.0.8 영문 버전의 최신 커스텀 펌웨어입니다.

v1.0.8에서 달라진 점은:

  • 언어 선택 기능 추가
  • 일본어 화면 언어+음성 언어 추가... 한국어는??
  • WDR 메뉴 추가 (이상하게도 앱에서는 안 나왔습니다)

그와 별개로 커스텀 펌웨어의 장점 중 제가 주목한 점은 다음과 같습니다:
(1) GPS 모듈 장착 시 MP4 영상에 기록된 좌표를 SRT 자막 파일로 추출
(2) SSH 접속 가능 -- 안내 음성 파일 바꿀 때 사용
(3) USB 장치 인식 가능 (주: 안 써봐서 어디 쓰는지는 잘..)
(4) 시작 시 "autorun_"으로 시작하는 셸스크립트 실행 기능 ← 보다 간편한 음성 교체를 위해 쓸 기능입니다

아래 링크에서 다운로드하시고,

OTA_38_FULL_70mai_d05_v1.0.8na_20200226_1828_full_ssh_gps_usb.7z

4PDA 포럼에서 직접 다운로드하실 분들은(비회원일 시 다운로드 링크가 동작하지 않습니다),
아래 두 링크를 참조하셔서 회원가입/로그인 시 필요한 키릴 문자 캡챠 연옥에서 탈출하세요:
러시아어 숫자 강제 공부중... - 토픽게시판 - XETOWN
4pda.ru 에 가입했습니다. - 아무 의미 없는 삽질기

  • 압축 파일의 "update" 폴더를 통째로 블랙박스의 SD카드에 넣으신 후,
    • SD카드의 최상위 폴더에 넣으시면 됩니다. 즉, 'event' 폴더 같은 기본 폴더와 같은 경로에 두세요.
  • 블랙박스의 전원이 꺼진 상태에서
    • (1) 화면 하단 버튼 4개 중 가장 우측 버튼과
    • (2) 그 아래 가로로 길쭉한 전원 버튼을
    • 동시에 누르고 계시다가, "업데이트 중" 화면이 뜨면 손을 떼세요.

(20년 3월 기준) 최신 버전이 되었습니다!

네이버 클로바 표 한국어 안내 음성으로 교체하기

네이버 클로바 음성 합성 서비스를 이용하여 만든 음성입니다: https://clova.ai/voice

(추가) 위 문단처럼, SSH/셸스크립트 실행을 지원하는 커스텀 펌웨어를 먼저 기기에 설치하세요!

음성 파일 중 "storage/low_speed.aac" 파일은 한국어 버전이 없습니다. 다른 분 게시글을 참조하시거나 하세요.
추천: 샤오미 블랙박스, 70Mai Pro 에 한국어 음성 적용하기 / 이런저런 블로그

 

70maiPro_NaverClova_Voice_v2.zip
0.62MB

이전 첨부파일(v1)의 스크립트에서는, 파일 복사를 마친 후 스크립트 자신은 지우지 못하는 문제가 있었습니다.

첨부된 파일에는 "voiceplay" 폴더, 그리고 두 개 "sh" 확장자의 파일이 있습니다.

먼저, 블랙박스의 SD카드를 PC에 연결하시고,
음성파일이 들어있는 "voiceplay" 폴더를 블랙박스의 SD카드에 넣으세요.

다음으로, "sh" 확장자의 스크립트 파일은 역시 4PDA 포럼에서 불펌한 것인데요(v_max님),
4PDA/음성 파일 교체

기존 v1.0.7까지와 달리, v1.0.8에서 다국어 지원 기능이 추가되면서, 음성 파일이 기존 경로에서 다음 경로로 바뀌었습니다:

  • 기존: /liteos/voiceplay/
  • 변경: /liteos/voiceplay/english/

따라서 스크립트도 그에 맞게 가져가야 하기 때문에..

  • v1.0.7까지는: autorun_copy_voiceplay_107.sh 파일을
  • v1.0.8부터는: autorun_copy_voiceplay_108.sh 파일을
    블랙박스의 SD카드에 넣어주세요.

그러면 블랙박스의 SD카드에 "voiceplay" 폴더와 둘 중 하나 골라서 넣은 "sh" 파일이 들어있게 되겠습니다.

💡 부팅 후 자동으로 음성 파일을 복사하는 중 전원이 꺼지면 기기가 손상될 수 있으니, 블랙박스 배터리가 충분히 충전된 상태, 또는 USB 전원이 공급되는 상태에서 진행해보세요.

이제 준비된 SD카드를 다시 블랙박스에 넣고, 전원을 켜면 별다른 메시지는 뜨지 않지만

  • 알아서 "voiceplay" 폴더의 파일이 블랙박스 시스템 공간으로 복사되고 나서,
  • 방금 전 SD카드에 넣은 위 파일은, 할 일을 다했으니 용량 절감을 위해 알아서 삭제됩니다

이 작업은 순식간에 진행이 되고요, 새 버전 (커스텀) 펌웨어를 설치하실 때마다 해 주시면 되겠습니다.

이런 고생 하지 않게 얼른 한국 정발이 되어야할 텐데요. 물론 국내 제품들이 현재 가격대에서 화질이 크게 개선되는 게 제일 좋고요.

💡 펌웨어 업그레이드하는 김에 음성 파일 교체도 한 번에 되도록, 새 펌웨어 파일과 위의 음성 교체 파일을 한꺼번에 넣을 수도 있겠습니다.

그러나 저는 권장하지 않는데요, 준비된 SD카드를 기기에 꽂고 펌웨어 업그레이드 버튼 조합을 누르기 실패할 경우 정상 부팅이 되어 사용중인 기존 펌웨어에 "음성파일 교체 작업"을 하는 헛짓이 될 수 있기 때문입니다.

한방에 업그레이드 + 음성 파일 교체를 하시려면 펌웨어 파일, 음성 교체 파일을 한꺼번에 넣어주시되..., 업그레이드 버튼 조합을 잘 눌러주세요!

반응형

리소스 오버레이 프로젝트 폴더 준비: AndroidManifest.xml 파일, res 폴더

안드로이드 스튜디오 프로젝트로 만드는 분들도 있던데, 저는 잘 몰라서 손수 해 보겠습니다.
잘 찾아보면 괜찮은 템플릿 많이 있는 것 같아요. 요런 것도 있고요

프로젝트 폴더로 삼을 폴더 하나를 만들고, 그 안에 아래와 같은 파일 및 폴더를 만드세요.

(1) AndroidManifest.xml

<?xml version="1.0" encoding="utf-8" standalone="no"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android" android:compileSdkVersion="28" android:compileSdkVersionCodename="9" package="<오버레이 패키지명>">
    <overlay android:priority="<우선순위>" android:targetPackage="<대상 패키지명>"/>
    <application android:allowBackup="false" android:hasCode="false"/>
</manifest>
  • <오버레이 패키지명>
    리소스 오버레이 패키지의 이름을 적당히 지어주시면 됩니다.
  • <대상 패키지명>
    리소스를 수정하려는 대상 앱의 패키지 이름을 적으세요.
  • <우선순위>
    양수 자연수 숫자 값이고요, 같은 대상 패키지에 대해 오버레이가 여러 개 있을 때... 낮은 우선순위 값을 지닌 오버레이가 나중에 적용됩니다. 최후의 승자가 되는 셈이죠.
    예를 들어, 쓰고 있는 오버레이의 우선순위가 9인데, 같은 앱을 대상으로 하는 내 오버레이가 꼭 적용되어야 한다면 우선순위를 9보다 작은 8 또는 그 이하 값으로 주면 되겠죠.

(2) res 폴더
요기다 수정하려는 리소스를 넣어두게 됩니다. 하위 폴더 구조는 일반적인 안드로이드 앱 리소스와 같으니 안드로이드 문서, 또는 수정 대상과 똑같이 해 두세요. 아래에서 이어집니다.

수정할 리소스 만들어 넣기

💡 리소스 폴더명 뒤에 옵션이 붙을 수 있습니다:

  • 화면 밀도 (xhdpi, xxhdpi, ...), 회전 여부 (land), API 버전 (v21), 야간 모드 (dark), 언어 (ko) 등등

✔ 각각 다 고쳐주셔야 합니다!

  • 덮어쓰려는 리소스 파일과 같은 경로, 같은 파일 이름으로 만드세요.

해본 것 위주로 예시를 적어보겠습니다.

숫자, 문자열 값 /res/values/...

원본과 같은 이름의 폴더에 같은 이름의 파일을 만들고 (e.g. dimens.xml), 파일 내용으로는 덮어쓰려는 키-값 쌍만 나열해 주세요. 원래 파일의 키-값 전체를 반복해서 쓸 필요 없습니다. 파일 내용으로 예를 들면 이렇습니다.

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <dimen name="operator_text_font_size">0.0dip</dimen>
    <dimen name="operator_text_kt_image_width">0.0dip</dimen>
    <!-- 몇 개 더 덮어썼지만 생략합니다. -->
</resources>

위험도: 낮음 ✅
다른 곳에서 정의된 ID를 참조하는 형태가 아니므로 _없는 ID 참조 예외_가 발생할 일 없으며, 어지간히 잘못된 값을 주지 않는 한 앱 오류를 발생시킬 가능성도 낮으므로 그럭저럭 실수해도 안전하다 하겠습니다.

그림 리소스: /res/drawable/... 또는 /res/mipmap/...

drawable 폴더에 바로 그림이 있을수도 있고, drawable 폴더에는 상황별 리소스를 참조하는 XML 파일이 있고 mipmap에 실제 그림 리소스가 있을 수 있습니다. XML 파일을 고치는 것은 위험하다고 생각되니, 대신 이미지 파일을 수정해보세요.

e.g. /res/drawables-xxxhdpi-v4/stat_sys_operator_kt.png 파일을 투명 파일로 교체, xxxhdpi 대신 xxhdpi, xhdpi 폴더의 그림도 마찬가지.

💡 화면 밀도, 회전 여부 등 폴더명 베리에이션을 잊지 마세요.

위험도: (그림 자체) 낮음 ✅
다른 곳에서 정의된 ID를 참조하는 형태가 아니므로 _없는 ID 참조 예외_가 발생할 일 없으며, 그림이 비정상이 아닌 한 역시 안전하다 하겠습니다.
위험도: (XML 파일) ..? ❔
(안 해봤습니다. 다른 리소스명을 참조하게 되기 때문에 혹시나 새 버전에서 사라지면.. 오류를 일으킬 것 같습니다.)

스타일 /res/values/styles.xml

TODO 스타일 상속 받는 법을 잘 모르겠더라구요.. 그걸 제외하고는 앞과 비슷하게 (1) 같은 이름의 스타일을 적고, (2) 그 안에 덮어쓸 item 을 나열하세요.

<?xml version="1.0" encoding="utf-8"?>
<resources>
    <style name="TextAppearance.Material.Notification.Emphasis" parent="@*android:style/TextAppearance.Material.Notification.Emphasis">
        <item name="android:textColor">#66ffffff</item>
    </style>
</resources>

💡 정의된 다른 ID를 참조할 때는 아래와 같이 적으세요.
@*<패키지명>:<리소스 분류>/<리소스 ID>
오버레이 대상 패키지에 속한 ID를 참조할 때, 안드로이드 프레임워크 (위의 android) 안의 내장 ID를 참조할 때 모두 위와 같이 @* 기호와 함께 패키지명을 적어 주세요.
(왜 그래야 할까요... :()

위험도: 잘 모름
음... 참조 잘못하면 위험하겠죠.

레이아웃

한 발 나아가서 레이아웃도 굳이 고치려면 고칠 수 있습니다. 그런데 XML 파일에 다른 곳에서 정의된 ID를 참조하거나, 새 ID를 정의하는 내용이 들어갑니다. 앱 업데이트 후 있던 ID가 없어지거나 ID 이름이 바뀌었을 때, 레이아웃에서 또는 소스코드에서 엉뚱한 ID를 참조하려 하면 오류가 발생하겠습니다. 시스템 리소스를 수정한다면 오류 때문에 더 위험하겠죠?

  • 원본 레이아웃 파일을 오버레이 프로젝트 폴더로 복사해 오세요.

  • 파일을 열고, ID 참조 부분을 모두 수정해주세요.

    💡 오버레이 프로젝트에는 원래 앱 패키지에 정의된 ID가 없습니다. 그러니 ID 참조가 원래 패키지를 가리키도록 수정해 주어야 정상적으로 리소스 오버레이 프로젝트의 컴파일이 가능합니다.
    <리소스 분류>/<리소스 ID> => @*<패키지명>:<리소스 분류>/<리소스 ID>
    e.g. integers/awesome => com.lvlz8.lovelinus:integers/awesome

  • 이제 레이아웃 파일에 원하는 수정 사항을 적용하세요 e.g. 뷰 요소 자리 바꾸기, (TextView 등 안드로이드 기본) 뷰 추가하기, ...

위험도: 높음 ❗
잘못된, 없는 ID 참조 시 예외로 앱이 강제종료됩니다. 시스템UI라면 상황은 더 심각합니다. 삼성 기기는 안전 모드로 재부팅하면 리소스 오버레이가 해제되는 듯 한데, LG 기기는 그렇지도 않아서 어려움이 있습니다.

aapt 툴 통해 프로젝트를 APK 파일로 생성, 서명 후 기기에 잘 설치

수정할 리소스 준비를 모두 마쳤다면, 프로젝트 폴더에서 명령줄 창을 띄우고 aapt 도구로 APK 컴파일이 가능합니다. aapt는 안드로이드 SDK에 들어있는 패키징 도구입니다.

아래 명령은 Pie OS 기기를 (API 28) 목표로 하고, 시스템 요소 중 SystemUI.apk 파일을 대상으로 준비해 본 명령입니다.

aapt package -f -M AndroidManifest.xml --min-sdk-version 26 -S res -I <android-28.jar> -I <SystemUI.apk> -I <framework-res.apk> -I <제조사 리소스 경로> -F <APK 출력 경로>
  • 최소 SDK 버전을 26으로 두었는데요, 최소 버전이 낮으면 예전 권한 시스템이 필요한 것으로 잘못 인식해 불필요한 권한이 부여됩니다.
  • -M, -S 플래그로 각각 매니페스트 파일, 리소스 폴더를 지정했습니다.
  • -I 플래그를 연속으로 나열하여 안드로이드 API 파일 (SDK에서 떼어왔습니다), 수정 대상 앱 (SystemUI.apk), 그리고 안드로이드·제조사 프레임워크 리소스 파일을 지정했습니다. 이렇게 하면 컴파일 중 필요한 리소스를 모두 정상적으로 참조할 수 있습니다.
  • -F 플래그로 APK 출력 경로를 적었습니다.
  • 뭔가 덜 한 것 같기는 하지만 여러분이 완성시켜주세요.

컴파일된 APK 파일을 테스트 인증서 또는 여러분의 인증서로 서명하면 준비 끝.

준비된 리소스 오버레이 파일을 모아... 아래와 같이 설치해볼 수 있겠네요.

  • 오레오 OS라면 그냥 설치?
  • Substratum 같은 리소스 오버레이 매니저를 쓰시거나
  • 루팅해서 설치하거나...
  • LG, 삼성 기기면 직접 테마로 만들든 툴을 쓰든 해서

...여기까지 해서 비록 Pie 버전에서는 설치가 막혔지만, 시스템 리소스 또는 사용자 앱 리소스를 수정해볼 수 있는 안드로이드 OS 내장 도구인 리소스 오버레이를 알아봤습니다.
읽으시느라 고생하셨습니다!

반응형

분석 결과 예시

LG 휴대폰 오레오/파이 OS에서 상태바 통신사 문구 및 VoLTE 로고 숨기기를 위해 찾아본 리소스 목록을 예시로 들어 보겠습니다.

굿락으로 챙겨주는 갤럭시 부럽다..

무슨 파일을 덮어써야 할까...

상태바 리소스이니 다음 파일을 찾아봅시다.

  • SystemUI.apk를 먼저 찾아보고,
  • 공통 리소스를 담는 framework-res.apk
  • 그리고 제조사 공통 리소소를 담는 lge-res.apk (삼성 폰도 비슷한 거 있음) 파일을 열심히 뒤져봐야겠습니다.

SystemUI.apk에 있는 상태바 관련 레이아웃 파일이나 소스코드를 참고해서 관련 리소스 이름을 알아내는 것도 좋습니다.

문구 표출 형태, 그리고 관련 리소스 찾기

리소스 디렉토리 구조 및 이름 짓는 법을 알고 계시면 편합니다. 이 부분은 공식 문서 포함해서 좋은 글 많으니 직접 찾아보시고요, 여기서는 실제 예시를 들어보겠습니다.

❌ 레이아웃(layout) 등 다른 ID를 참조하는 코드가 포함된 리소스는 수정하기 어렵습니다. 자세한 내용은 나중에..

  • 🧀 아이콘: KT, LGU+, 그리고 VoLTE 로고
    그림 형태이니까, 1px * 1px 크기 투명한 PNG 파일로 바꿔치기하면 되겠습니다.

수정 대상은 drawable 리소스 자체일 수 있고, drawable 리소스는 여러 mipmap 요소를 상황에 따라 사용하도록 하는 XML 파일이며 실제 이미지는 mipmap일 수 있습니다. 후자의 경우는 실제 그림 리소스인 mipmap을 고치는 게 좋겠습니다.

SystemUI.apk에만 있으면 좋았을텐데, lge-res.apk에도 VoLTE 로고를 포함해서 몇 개 있었습니다. 싹다 바꿔주면 되겠고요.
DPI 별로 베리에이션이 있으니 (xhdpi, xxhdpi, xxxhdpi 등) 모든 베리에이션에 대해 역시 싹다 바꿔줍니다.

  • 🅰 문구: SKT
    그냥 T로고가 예쁜데..

이 경우는 문구에 해당하는 폰트 크기를 0으로 만드는 방법이 먹혔습니다. 아래와 같은 방법으로 수정 대상 값을 찾을 수 있는데요.

  • res/values/dimens.xml 파일의 dimen(sion) 값 중 적절해보이는 이름을 찾을 수 있고, (머리가 귀찮으면 몸이 고생)
  • 상태바 관련 레이아웃에서 통신사 문구를 나타내는 요소를 찾아 확인할 수 있습니다.
    • 폰트 크기 속성이 직접 지정되어 있으면 그 값을 봅니다.
      res/layout/status_bar.xml:
      <TextView ... android:textSize="@dimen/operator_text_font_size" />
      <TextView ... android:textSize="12sp" /> <!-- 단, 이렇게 값이 직접 씌어있다면 수정하기 난감합니다. -->
    • (폰트 크기 속성을 포함하는) 스타일을 지정한 경우, res/values/styles.xml을 확인해야 합니다.
      <!-- res/layout/status_bar.xml: -->
      <TextView ... android:textAppearance="@style/TextAppearance.StatusBar.Clock" />
      <!-- res/values/styles.xml: -->
      <style android:name="TextAppearance.StatusBar.Clock">
      ...
      <item name="android:textSize">10sp</item>  <!-- 값으로 직접 지정되어 있으면 styles.xml을 고치면 되고 -->
      <item name="android:textSize">@dimen/clock_text_font_size</item> <!-- 또다른 값을 참조하면 해당 값을 고쳐줍니다 -->
      ...
      </style>
      • 레이아웃 대신 소스코드에서 뷰 요소의 폰트 크기를 지정할 수도 있겠죠. 이 경우는 잘... 확인하시면 되겠습니다.
  • 역시 화면 밀도, 가로보기 여부, 야간 모드 등의 베리에이션이 있을 수 있으니 싹 다 찾아서 바꿔주세요.

다행히 폰트 크기가 레이아웃에 하드코딩되어있지 않으며 dimen 값을 참조하게 되어 있었고, 해당하는 값은 다른 상태바 요소와 공유되지 않는 별도 값이었습니다. (통신사 문구 폰트 크기를 0으로 만들었더니 시계 문구도 안 보이면 난감하니까요.)

마지막 (4)편에서...

반응형
* 번거로운 작업 없이도 SD카드에 파일을 넣기만 하면 되는 아래 방법을 대신 사용해보세요:
https://limerainne.tistory.com/56

샤오미(산하 브랜드?)에서 만든 블랙박스 70mai Pro는 저렴한 가격에 고화질 영상을 자랑하는 가성비 블랙박스입니다. 국내 제품에서는 고가 제품에만 들어가는 고급 이미지 센서를(소니 IMX335) 7만원짜리 저가형 제품에서 누릴 수 있는 것이 큰 장점입니다.

70mai pro

(이미지 출처: 다나와 상품정보)

러시아산 커스텀 펌웨어: 시스템 파일, 즉 안내 음성 파일 수정 가능

여기에 러시아 형누님들의 SSH 지원 커스텀 펌웨어를 설치하면, 보안성이 조금 더 떨어지는 대신 (러시아 형누님들이 무슨 일을 했을 지..), SSH 연결을 통해 시스템 파일 접근 및 수정이 가능합니다. 그러면.. 안내 음성 파일 교체도 할 수 있죠!

4pda 포럼 70mai Pro 게시글에 가시면 최신 커스텀 펌웨어를 찾을 수 있습니다.

네이버 클로바: 매우 자연스러운 음성 합성 서비스

클로바 보이스 체험 페이지에 가시면 딥러닝 기술로 만든 음성 합성 기능을 체험해 볼 수 있습니다. 게다가 개인 콘텐츠 제작 목적에 한하여 무료 사용이 가능하다고 하네요! (*추가: "출처(Clova Voice)를 명시"하시라는군요..)

그렇다면?

네이버 클로바로 만든 70mai Pro 한국어 안내 음성

  • 위 포럼 링크에서 러시아 형누님들이 준비한 음성 파일 및 대본 엑셀 파일을 받았습니다.
  • 열심히 번역하고, 네이버 클로바 음성합성 데모 페이지에서 손수 하나하나 음성 WAV 파일을 받았습니다. 스크립트 만들면 편했겠지만, 몇 개 없으니..
  • 적절한 형식의 AAC 파일이 필요하므로 음악 파일 변환기로 WAV를 AAC로 바꿔줍니다.

70mai_NaverAraVoice.zip
2.90MB

해서 한국어 안내 음성 파일을 준비했습니다. 이제 기기의 음성 파일을 교체하기 위해 역시 러시아 포럼의 가이드 글을 따랐습니다:

  • SSH 연결을 지원하는 커스텀 펌웨어를 설치한 블랙박스 기기를 준비합니다.
    • 저는 v1.0.6na 버전을 설치했는데, 지금 보니 1.0.7 버전이 공개되어 있네요. 1.0.8 버전도 나왔대요.
  • 기기와 안드로이드 폰을 와이파이로 연결한 뒤 (70mai 앱 통해 연결), "AndFTP" 앱으로 기기에 SCP 접속했습니다.
    • SFTP가 아니라 SCP라 위 가이드에 있던 AndFTP 외에 마땅한 앱을 모르겠더라고요.
    • 러시아산 SSH 펌웨어 기본 계정은 "root", 비밀번호는 없음.
  • 루트 디렉토리로 올라가서 "/liteos/voiceplay/" 디렉토리로 들어가세요. 상황별 디렉토리(adas, record, storage, system) 밑에 음성 파일들이 있습니다. 요걸 한국어 음성 파일로 교체하면 끝!
    • 펌웨어 버전 v1.0.8부터는 다국어 지원 기능이 추가되면서 (일본어 추가됐다던데 일본 진출하나봅니다. 한국은?) 이제부터는 "/liteos/voiceplay/english/"로 들어가셔야 합니다.

위와 같은 안드로이드폰(또는 PC) 및 번거로운 작업이 싫으신 분은 아래 글을 참고해보세요:
샤오미 70mai Pro 한국어 안내 음성 with 네이버 클로바 보이스 #2

문제점
버전 차이인지 다른 음성 디렉토리가 또 있는지, 대본 파일에 있던 일부 음성 항목은 1.0.6버전 기기에 없었습니다. 총 17개 항목이 없었는데, "음성 녹음" 상태 안내 빼고는 펌웨어 업데이트 등 주요 기능에는 무관한 종류라 뭐 일단은 괜찮은 것 같습니다.

UI 언어 교체, 즉 화면 문구를 한국어로 바꾸는 일, 그리고 음성 인식 언어 교체 모두 현재는 불가합니다.
- UI 언어 교체는 얼마 전에 4pda 포럼 글을 살펴 보니 변경 시도가 이루어지는 것 같아요.
- 음성 인식 언어는.. 아직까지는 개인이 만들기는 어렵고 정식 발매되어 한국어 지원도 정식으로 되기를 기다려 봐야겠습니다.

참고

음성 소스가 "네이버 클로바"임을 꼭 명시하시기 바랍니다.

변환 설정

  • Pazera Free Audio Extractor
  • AAC, 48000Hz 샘플레이트, 모노 채널, CBR 비트레이트, AAC-LC 프로필, 그리고 64kbps 비트레이트

 

 

반응형

1단계: 수정 대상 앱 또는 시스템 프레임워크 확보

특정 앱을 수정하고 싶다면,

  • 앱 파일 획득

    • 기기에서 직접 추출
      요즘 파일 탐색기에 다들 앱 추출 기능이 있죠. 아니면 Send Anywhere 앱도 있고요.

    • APK 공유 사이트 이용
      저는 APKMirrorAPKPure를 이용하는데요, APKPure 사이트는 왜인지 모르게 중국 풍이 느껴지기는 하지만 한국 개발사 포함해서 각종 앱이 다 있습니다. 혹 변조되었을지 두려운 마음 조금 있지만, 디지털 서명을 믿을 수밖에요. (서명 v1은 뚫릴 수 있다던데... 설마...)

한편 휴대폰 UI를 바꾸고 싶다면,

  • 시스템 프레임워크 파일 획득: 보통 아래 파일을 획득하면 되는 것 같습니다.
    • /system/framework/framework-res.apk : 안드로이드 OS 공통
    • /system/priv-apps/SystemUI.apk : 상태바, 내비게이션 버튼 등 시스템 요소 그리는 애
    • 또한 /system/framework 디렉토리에 제조사별 커스텀 리소스도 있으니 함께 확보해야 합니다.

그 외에 프레임워크 소스 파일, Settings.apk도 확보하셔서 소스를 들여다보면 이 리소스가 어디에 쓰이는지 아는 데 도움이 됩니다. 문제는 안드로이드 OS 5.0부터 ART를 도입하고, 그 후로 갖가지 최적화가 적용되면서 소스 파일이 점점 숨어들어갔습니다. 디컴파일해서 소스 들여다보기가 쉽지는 않습니다.

2단계: 역컴파일 툴 통해 코드 및 리소스 분석

최근 저는 JADX 즐겨 쓰고 있습니다. 별다른 복잡한 조작 없이 달빅 바이트코드를 자바 코드로 바꿔 보여주고, 전체 코드 검색 가능한 점이 좋습니다. 리소스 대상으로는 키워드 검색 안 되는 점은 아쉬워요.

JADX 개발자님들께 이 자리를 빌어 감사드립니다. 물론 의미없죠..
JADX 프로젝트 홈페이지

팁은... 잘 모르겠고... 감으로 때려맞추고 이래저래 하다보면 답이 나옵니다...

한편, 시스템 프레임워크를 수정한다면 제조사 또는 서드파티 테마 파일을 열어보는 것도 좋습니다. 어디를 건드려야 할 지 가이드가 되거든요.

 

(3)편에서 계속...

반응형

RRO 또는 runtime resource overlay

개념

안드로이드 앱 패키지의 주요 구성 요소는 코드, 리소스라 할 수 있습니다. 이 중 리소스에는 색상, 문자열 같은 단순 값부터, 그림 (drawable), 그리고 (XML) 레이아웃 등이 들어있는데요.

(런타임) 리소스 오버레이는 앱 패키지를 불러올 때, 원래 앱의 리소스를 사용자가 설치한 오버레이 패키지의 리소스로 덮어씌워주는 도구입니다.

소니 모바일 팀에서 개발했고 6.0 - 7.0 시절에 AOSP에 기부했다고 하고요, 8.0부터 본격적으로 도입이 된 것 같아요. 삼성, LG를 포함해서 많은 제조사들이 이 리소스 오버레이 기능을 바탕으로, 휴대폰 테마 기능을 개발하여 적용했습니다. substratum 테마 앱도 리소스 오버레이를 기반으로 작동하죠.

한계

  • 소스 코드를 고칠 수 없습니다. 당연하죠.
  • 새 버전이 나올 때마다 변경 사항이 있는지 보고 반영해야 합니다. 못할 짓이죠.
  • 앱 오류, 또는 시스템 전체에 오류를 일으킬 수 있습니다
    • 단순한 값 변경도 경우에 따라 코드에서 부작용을 일으킬 수 있습니다
    • 레이아웃처럼, 코드에서 리소스 내부 구성요소에 (예. 레이아웃 내 뷰) ID로 접근한다던지 하면, 없어진 요소를 프로그램에서 찾지 못하고 NullPointerException 예외를 일으킬 수 있는 등 예기치 못할 위험이 있습니다
  • Android 10부터는 `assets/`를 덮어쓸 수 없도록 개선(?)되었습니다. 'assets' 디렉토리에 실행가능한 바이너리 등을 두는 등 (아니 딴 데다 둬야지 왜 거기다가) 동작이 변조될 위험이 있는 앱이 있다고 합니다. 따라서 리소스 오버레이가 'assets' 디렉토리를 수정하는 일이 나쁜 목적으로 쓰일 수 있으므로 기능을 삭제했다고 하네요.

    * 참조: https://source.android.com/devices/architecture/rros#installing-rros 중,
    "Android 10에는 애셋을 오버레이 가능으로 지정하는 메커니즘이 포함되어 있지 않으므로 오버레이가 더 이상 애셋을 오버레이할 수 없습니다."

 

안드로이드 9.0 파이 제한 사항 및 우회 방안

다만 안드로이드 9.0부터는 구글이 보안 개선을 이유로 제조사 서명이 되지 않은 리소스 오버레이는 설치할 수 없도록 제한했습니다. 보안 위협 시나리오가 어떤 것이 있을지..., 소설을 써 보겠습니다만 실현 가능성이 없어 보이네요.

  • 사악한 DLC 앱: 스토어에는 검수하기에 멀쩡해 보이는 앱을 게시하고, 사용자를 속여 리소스 오버레이가 설치되면 오버레이에 정의된 플래그 값을 읽어서 숨겨진 사악한 기능이 활성화되는 앱. 구글 검수를 퍽이나 통과할 수 있겠죠?
  • 앱 기능 망치기: 특정 상수값을 고쳐서 앱 동작을 방해한 뒤, 기밀 정보를 빼돌리기. 참 쉽겠죠?

삼성 기기의 경우 S9/노트9 빼고는 설치 제한이 걸린 것으로 보입니다. 만, 플레이스토어에 파이 OS용 리소스 오버레이 기반 테마 제작 앱이 있습니다. *Synergy
LG 기기는 사용자 테마를 만들고 그 안에 직접 만든 리소스 오버레이를 넣어둘 수 있습니다. 언젠가는 막히겠지만 안 막히기를 바랍니다.

제작 절차

  • 수정하려는 앱 및 시스템 프레임워크 파일을 획득
  • 역컴파일 툴 (JADX 추천) 통해 코드 및 리소스 분석
  • 리소스 오버레이 프로젝트 폴더 준비: AndroidManifest.xml 파일, res 폴더
  • 수정할 리소스 만들어 넣기
  • aapt 툴 통해 프로젝트를 APK 파일로 생성, 서명 후 기기에 설치

 

(2)편에서 계속...

반응형

국내 스마트폰은 플레이스토어, 제조사 스토어에서 원하는 폰트를 유·무료로 구입해서 쓸 수 있습니다.

아이폰에는 탈옥 않으면 절대 바랄 수 없고, 외산폰은 루트 권한 있어야 하는 것 같은데...

그런데 스토어에 없는 (수정도 허용되는 라이선스 또는 개인 소장) 폰트를 쓰고 싶으면.. 폰트 패키지를 직접 만들어 써야겠죠. 국내 두 제조사는 각기 다른 패키지 형식을 쓰는데요.

"플립폰트"를 쓰는 듯한 (아직 그대로죠?) 삼성폰은 방법이 여러가지 마련되어 있죠.
예를 들어 파이 OS 즈음부터 (S9/노트9 빼고) 사용자 폰트 추가가 차단된 듯 하나, 테마 무료체험 기능을 오용하는 꼼수가 마련되어 있습니다. https://tali.kr/378
역시 사용자 많고 봐야 합니다. 저도 끼워주세요... 곧 아이폰11 나오니 기대하겠습니다.. 지금도 잘 팔리니 그런 일은 없을 것..
지금은 없는... 팬텍도 플립폰트였네요.

 

그럼 LG는...? "HY폰트"라는데 한양에서 만든 폰트 시스템인가봐요.
문제는 이 집은 해독하기 어려운 폰트 패키지 형식을 쓰는 바람에 사용자가 비공식 폰트 패키지를 만들기가 쉽지 않습니다. 조금 더 자세히는 패키지 내의 "/asset/font.dat" 바이너리 파일이 도대체 무슨 형식인지 알 수가 없습니다.

HY폰트 형식이 나온지가 매우 오래되었기 때문에 비공식 패키지 제작법이 나올만 한데, 저는 아직까지는 찾지 못했습니다. 미국, 아니면 베트남 커뮤니티에서 뚫어줄 만 한데, 굳이 고생할 유인이 없는 듯 합니다. 폰트 판매자 입장에서는 다행이지요.

LG폰 사용자 폰트 패키지 글을 참고하세요.

 

다만 방법이 완전히 없는 건 아니라고 합니다. 다른 폰트 패키지를 하나 받아서, 패키지 내의 폰트 파일을 원하는 폰트로 바꿔치기하는 방식이 있다고 하네요. "font.dat"를 수정하지 않겠다는 의도가 되겠습니다. 이미 수 년 된 방법입니다.

폰트 파일만 교체
폰트 수정 (FontCreator 사용) + 파일 교체

  1. 쓰지는 않을 폰트 패키지를 하나 다운로드하거나 구입하고, PC로 추출해 둡니다.
  2. 패키지를 반디집 같은 압축 프로그램으로 열어서, 폰트 이름, 폰트 파일 이름을 확인합니다.
  3. (필수 아님?) 사용하려는 폰트를 FontForge(무료!) 같은 프로그램으로 열어서 폰트 이름을 패키지의 것으로 바꿉니다.
  4. 사용하려는 폰트 파일 이름을 패키지의 폰트 파일 이름으로 바꾼 뒤, 패키지에 넣어줍시다.
  5. 패키지를 다시 서명한 후, 기기에 설치하고 적용합시다. 기기에서 폰트 이름은 패키지의 것으로 표시되겠습니다.

...뭔가 자괴감 드는 방법이죠.. 그냥 스토어에 있는 폰트 씁시다.

반응형

참고: Enable Windows 10 Fluent Design Acrylic Effects in VirtualBox

Windows 10 1903 (19년 5월 업데이트) 즈음부터 'Fluent Design', 특히 아크릴 느낌 배경 효과가 본격 적용되고 있습니다.

그러나 PC 성능이 부족하다던지 하면 알아서 효과가 해제되는데요, 설정 앱에서 "투명 효과"를 아무리 껐다켰다 해도 소용이 없습니다.

위 링크를 보시면 가상PC인 경우 안 되는 모양이고요, 저 같은 경우에는 사용중인 PC가 저성능 PC로 인식이 되었습니다.

이 경우 위의 링크를 참조해서 레지스트리 편집 하나 해 주면 고급 꾸밈 효과를 강제로 동작시킬 수 있다고 합니다.

  • 레지스트리 편집기를 열고
  • HKEY_LOCAL_MACHINE\SOFTWARE\Microsoft\Windows\Dwm까지 찾아 들어가서
  • 새 32-bit DWORD 값을 만들고, 이름은 ForceEffectMode로, 값은 2로 지어주세요. (왜 2일까...)
  • PC를 재부팅하면 끝!

한편, 창이 비활성화될 때 아크릴 효과가 꺼지고 단색 배경으로 대체되는 건 원래 그렇게 설계된 거랍니다. 안 쓰는 창에도 아크릴 배경을 그리는 불필요한 연산을 줄이기 위해서래요. 고성능 PC에서는 다 돌려도 괜찮은데 말이죠.

레딧에 항상 아크릴 배경을 그리게 만든 분이 계시네요. 링크

반응형

+ Recent posts