회원가입아이디/비번찾기
홈으로

안드로이드 앱 커스텀 글꼴 넣기
9년 전

안드로이드 앱 커스텀 글꼴 넣기


저는 안드로이드의 기본 글꼴 Droid Sans가 그다지 못생겼다고 생각하지는 않지만 글꼴은 앱의 인상에 상당히 큰 영향이 있는 것 같습니다. 특히 버튼 등 이미지로 만들어 놓은 UI와 동적으로 변하는 텍스트의 글꼴이 다르면 많이 어색합니다. 그래서 이 글에서는 앱 수준에서 전체적인 글꼴을 바꾸는 방법을 알아보겠습니다.

준비

재배포가 자유로운 글꼴을 준비합니다. 사실 그런 한글 글꼴은 나눔글꼴이 거의 유일하죠. 영문 글꼴은 선택의 폭이 훨씬 넓습니다.

참고로 TrueType 글꼴보다 OpenType 글꼴의 용량이 훨씬 적은 것 같으니, otf 파일을 사용하시면 앱 용량이 무식하게 커지는 것을 막을 수 있습니다. 요즘 나오는 기종은 괜찮은데, 저처럼 내장 메모리가 적은 디자이어 같은 폰을 쓰시는 분들에게는 용량이 크면 좀 치명적이거든요.

기본

글꼴 파일을 assets/ 디렉토리에 넣습니다. 그리고 글꼴을 불러온 뒤, 글꼴을 지정하기 원하는 뷰에 setTypeface 메소드를 호출하면 됩니다. 코드로 나타내면 다음과 같이 됩니다.
Typeface typeface = Typeface.createFromAsset(getAssets(), "font.ttf");
TextView textView = (TextView) findViewById(R.id.textView);
textView.setTypeface(typeface);

// 굵은 글꼴이 따로 있을 때 추가로 해줘야 하는 부분
Typeface boldTypeface = /* ... 위와 같은 방법 ... */;
textView.setTypeface(boldTypeface, Typeface.BOLD);


여기까지만 보면 상당히 쉬워보이지만 한가지 문제가 있습니다.

용량이 큰 글꼴 집어넣기

바로, assets 디렉토리에 넣은 글꼴 파일의 용량이 클 경우 읽어오지 못하는 문제가 발생합니다. 대부분의 한글 글꼴은 용량이 크기 때문에 일반적인 방법으로는 읽어올 수가 없습니다. 과연 어떻게 해야 읽어올 수 있을까요?

한참 구글링한 끝에 쪼개서 넣고 합치는 방법을 발견했지만 뭔가 깔끔하지가 않아서, 근본적인 문제(왜 용량이 큰 asset은 읽어오지 못할까)를 조사해봤습니다.

알고 보니 apk 파일을 만드는 과정에서 특정 확장자(jpg, png, mp3 등)가 아닌 파일은 압축이 되며, 압축이 되고 나면 압축 해제 시 용량이 1MB를 넘는 경우 런타임에 사용할 수 없어진다고 합니다. (출처, 진저브레드부터는 이 제한이 사라졌습니다. 아마도 최소 요구 메모리가 늘어났기 때문으로 보입니다.)

그래서 결론적으로, 글꼴 파일의 확장자를 mp3로 바꾸면 압축 대상에서 제외되며, 읽어올 수 있게 됩니다! 다소 황당하지만 이 방법을 쓰면 1MB 제한을 벗어날 수 있습니다.

레이아웃 전체에 적용하기

앞서 살펴본 코드에서는 한번에 뷰 하나씩 일일히 글꼴을 적용해 줘야 했는데요. 귀찮기도 할 뿐더러 실수할 확률이 높고 레이아웃이 바뀔 때마다 고쳐야 한다는 문제가 있습니다.

그러니 AndroidManifest.xml 같은 곳에 지정할 수 있는 방법이 있으면 좋겠지만 안타깝게도 그런 건 없습니다. 그래서 최상위 뷰를 찾아서 트리를 타고 내려가면서 글꼴을 적용한다는 아이디어를 생각해냈습니다. 대략 다음과 같이 하면 됩니다.
private Typeface mTypeface;

@Override
protected void onCreate(Bundle icicle) {
    super.onCreate(icicle);
    setContentView(R.layout.fontTest);
    mTypeface = Typeface.createFromAsset(getAssets(), "font.ttf.mp3");
    ViewGroup root = (ViewGroup) findViewById(android.R.id.content);
    setGlobalFont(root);
}

void setGlobalFont(ViewGroup root) {
    for (int i = 0; i < root.getChildCount(); i++) {
        View child = root.getChildAt(i);
        if (child instanceof TextView)
            ((TextView)child).setTypeface(mTypeface);
        else if (child instanceof ViewGroup)
            setGlobalFont((ViewGroup)child);
    }
}


일단 전체적인 구조는, 뷰 여러 개를 담고 있는 ViewGroup(LinearLayout 등)을 만나면 재귀적으로 호출해서 트리를 타고 내려가는 겁니다. 위의 코드에서 주의할 점 몇 가지를 살펴보겠습니다.
•findViewById(android.R.id.content)는 최상위 뷰를 찾기 위한 코드입니다. 안드로이드 내부에 의존하기 때문에 조금 불안하긴 하지만, 일단 잘 돌아가니까 놔둡시다.
•루트 뷰를 찾아서 setGlobalFont를 부르는 작업은 반드시 setContentView를 호출한 후에 해야 합니다.
•setGlobalFont에서는 현재 TextView의 글꼴만 바꿔주고 있는데 TextView의 서브클래스가 아닌 뷰의 글꼴을 바꾸시려면 직접 추가해주셔야 합니다. 참고로 Button이나 EditText 등 대부분의 텍스트를 가진 뷰들은 TextView를 상속하고 있습니다.

앱 전체에 적용하기

위에서 설명한 방법을 사용해도 여전히 각 액티비티마다 글꼴 지정 코드를 넣어야 합니다. 수고를 덜기 위해서 별도의 기반 클래스를 하나 만들어서 사용하면, 액티비티 클래스마다 따로 지정하지 않고 상속만 해서 글꼴을 적용할 수 있습니다.
public class BaseActivity extends Activity {
    private static Typeface mTypeface;

    @Override
    public void setContentView(int layoutResID) {
        super.setContentView(layoutResID);

        if (BaseActivity.mTypeface == null)
            BaseActivity.mTypeface = Typeface.createFromAsset(getAssets(), "font.ttf.mp3");

        ViewGroup root = (ViewGroup) findViewById(android.R.id.content);
        setGlobalFont(root);
    }

    // 아까랑 같은 setGlobalFont
}

public class MainActivity extends BaseActivity { ... }


setContentView를 오버라이드 했습니다. 별로 좋은 방법 같지는 않지만 onCreate를 오버라이드 하는 것보다는 간단하니 이렇게 하는게 낫겠습니다.

글꼴을 불필요하게 여러번 불러오지 않도록 하기 위해 static으로 선언해서 필요할 때만 불러오도록 하였습니다. 취향에 따라 싱글턴 패턴을 적용하시거나, Application 클래스를 이용하셔도 됩니다.
추천추천 : 446 추천 목록
번호 제목
2,885
input 입력 필드 앞뒤 공백 실시간 제거
2,884
Placeholder 포커스시 감추기
2,883
MySQL 중복된 데이터를 삭제
2,882
MySQL 중복 데이터 확인
2,881
sessionStorage.getItem 와 sessionStorage.setItem
2,880
제이쿼리 랜덤으로 배경색 변경
2,879
preg match에 관한 정규식
2,878
Stream an audio file with MediaPlayer 오디오 파일 스트리밍 하기
2,877
Audio Streaming PHP Code
2,876
PHP $ SERVER 환경 변수 정리
2,875
Vimeo (비메오) API 를 사용하여 플레이어 컨트롤하기
2,874
iframe 사용시 하단에 발생하는 공백 제거방법
2,873
아이프레임(iframe) 전체화면 가능하게 하기
2,872
부트스트랩(bootstrapk)에서 사용하는 class명 정리
2,871
부트스트랩 CSS
2,870
크롬에서 마진 조절
2,869
PHP 현재 페이지의 도메인명이나 url등의 정보 알아오기
2,868
PHP preg match all()
2,867
PHP 로 웹페이지 긁어오기 모든 방법 총정리!
2,866
[PHP] 원격지 파일 주소 노출 안하고 curl로 다운로드 받기
2,865
PHP 함수 정리
2,864
아이프레임(iframe) 비율 유지하면서 크기 조절하는 방법
2,863
PHP 배열에서 무작위로 하나 뽑아주는 array rand() 함수
2,862
PHP 정규식 정리
2,861
PHP 정규식을 활용한 태그 및 특정 문자열 제거 및 추출 방법
2,860
php 크롤링 또는 파싱 함수, 정규식 모음
2,859
제이쿼리 기본 명령어
2,858
웹페이지 가로 모드세로 모드 인식하기
2,857
모바일 웹 화면 강제 회전(가로모드 고정)
2,856
[HTML5]에서 frameset 대체 방법과 iframe 속성
목록
뮤직트로트 부산광역시 부산진구 가야동 ㅣ 개인정보취급방침
Copyright ⓒ musictrot All rights reserved.