Подтвердить что ты не робот

Android: предпочтение от размера шрифта в приложении

Можно ли установить для всей ширины шрифта всю ширину шрифта для всех видов, отображающих текст? Я хотел бы предоставить предпочтение пользователю, который должен позволить масштабировать весь текст в приложении.

Android явно позволяет использовать блок измерения "sp" для масштабируемого текста, однако нет фактического способа установить "пользовательский размер шрифта" "глобально".

Итерация по всем представлениям экземпляра Activity не является вариантом; -)

4b9b3361

Ответ 1

Вот он, как я сделал это для своего приложения. В нескольких словах - в Activity.onCreate() вы получаете идентификатор ресурса стиля с определенным набором размеров шрифтов и применяете этот стиль к теме активности. Затем с помощью настроек предпочтений вы можете переключаться между этими наборами.

В первую очередь в значениях /attrs.xml объявляются атрибуты для набора размеров шрифтов:

<declare-styleable name="FontStyle">
    <attr name="font_small" format="dimension" />
    <attr name="font_medium" format="dimension" />
    <attr name="font_large" format="dimension" />
    <attr name="font_xlarge" format="dimension" />
</declare-styleable>

Затем в значениях /styles.xml объявляются несколько наборов размеров шрифтов:

<style name="FontStyle">
</style>

<style name="FontStyle.Small">
    <item name="font_small">14sp</item>
    <item name="font_medium">16sp</item>
    <item name="font_large">18sp</item>
    <item name="font_xlarge">20sp</item>
</style>

<style name="FontStyle.Medium">
    <item name="font_small">18sp</item>
    <item name="font_medium">20sp</item>
    <item name="font_large">22sp</item>
    <item name="font_xlarge">24sp</item>
</style>

<style name="FontStyle.Large">
    <item name="font_small">26sp</item>
    <item name="font_medium">28sp</item>
    <item name="font_large">30sp</item>
    <item name="font_xlarge">32sp</item>
</style>

Затем в методе onCreate() каждого действия добавьте:

getTheme().applyStyle(new Preferences(this).getFontStyle().getResId(), true);

где Preferences - это фасад объекта SharedPreferences:

public class Preferences {
    private final static String FONT_STYLE = "FONT_STYLE";

    private final Context context;

    public Preferences(Context context) {
        this.context = context;
    }

    protected SharedPreferences open() {
        return context.getSharedPreferences("prefs", Context.MODE_PRIVATE);
    }

    protected Editor edit() {
        return open().edit();
    }

    public FontStyle getFontStyle() {
        return FontStyle.valueOf(open().getString(FONT_STYLE,
            FontStyle.Medium.name()));
    }

    public void setFontStyle(FontStyle style) {
        edit().putString(FONT_STYLE, style.name()).commit();
    }
}

и FontStyle:

public enum FontStyle {
    Small(R.style.FontStyle_Small, "Small"), 
    Medium(R.style.FontStyle_Medium, "Medium"), 
    Large(R.style.FontStyle_Large, "Large");

    private int resId;
    private String title;

    public int getResId() {
        return resId;
    }

    public String getTitle() {
        return title;
    }

    FontStyle(int resId, String title) {
        this.resId = resId;
        this.title = title;
    }
}

И FontStyle.values() используется как элемент для Spinner в вашем PreferencesActivity. Вот как выглядит моя:

public class PreferencesActivity extends Activity {
@Override
protected void onCreate(Bundle savedInstanceState) {
    getTheme().applyStyle(new Preferences(this).getFontStyle().getResId(), true);
    super.onCreate(savedInstanceState);

    setContentView(R.layout.preferences);

    Preferences prefs = new Preferences(this);

    Spinner fontStylesView = (Spinner) findViewById(R.id.font_styles);
    FontStylesAdapter adapter = new FontStylesAdapter(this,
            R.layout.font_styles_row, FontStyle.values());
    fontStylesView.setAdapter(adapter);

    fontStylesView.setSelection(prefs.getFontStyle().ordinal());
}

@Override
public boolean onCreateOptionsMenu(Menu menu) {
    MenuInflater inflater = getSupportMenuInflater();
    inflater.inflate(R.menu.preferences, menu);
    return super.onCreateOptionsMenu(menu);
}

@Override
public boolean onOptionsItemSelected(MenuItem item) {
    switch (item.getItemId()) {
    case R.id.menu_done:
        onMenuDone();
        finish();
        return true;
    case R.id.menu_cancel:
        finish();
        return true;
    default:
        return false;
    }
}

private void onMenuDone() {
    Preferences prefs = new Preferences(this);

    Spinner fontStylesView = (Spinner) findViewById(R.id.font_styles);
    prefs.setFontStyle((FontStyle) fontStylesView.getSelectedItem());
}
}

И, наконец, вы можете использовать свои настройки размера шрифта:

<TextView android:textSize="?attr/font_large" />

Или я предпочитаю использовать стили в значениях /styles.xml add:

<style name="Label" parent="@android:style/Widget.TextView">
    <item name="android:textSize">?attr/font_medium</item>
    <item name="android:layout_width">wrap_content</item>
    <item name="android:layout_height">wrap_content</item>
</style>

<style name="Label.XLarge">
    <item name="android:textSize">?attr/font_xlarge</item>
</style>

И вы можете использовать его таким образом:

<TextView style="@style/Label.XLarge" />

Надеюсь, мой ответ поможет вам.

Ответ 2

Да, это возможно. Для этого вам необходимо:

  • Объявите свой собственный класс TextView
  • Использовать во всех ваших диалогах/мероприятиях только он

Как

public class SimpleTextView extends TextView
{
    private static final float DEFAULT_TEXT_SIZE=12.0;
    private static float textSize=DEFAULT_TEXT_SIZE;

    public SimpleTextView(Context context)
    {
        super(context);
        this.setTextSize(textSize);
    }

    public SimpleTextView(Context context, AttributeSet attrs)
    {
        super(context, attrs);
        this.setTextSize(textSize);
    }

    public SimpleTextView(Context context, AttributeSet attrs, int defStyle)
    {
        super(context, attrs, defStyle);
        this.setTextSize(textSize);
    }

    public static void setGlobalSize(float size)
    {
        textSize=size;
    }

    public static float getGlobalSize()
    {
        return textSize;
    }
}

И теперь, когда бы вы ни были, вы можете глобально изменить все размеры текста до 20 во всех текстовых просмотров, которые вызывают только:

SimpleTextView.setGlobalTextSize(20);

Ответ 3

Съемка с идеи хип здесь, чтобы рассмотреть (не требуется специальная реализация TextView)

  • Объявить свойство что-то вроде UNIVERSAL_FONT_SIZE с идеей, что его можно изменить из настроек, но будет сохранено между приложениями-приложениями
  • В onCreate метод каждого из ваших действий получает значение этого свойства и сохраняет как поле
  • Сделайте свой код используемым для каждого компонента, изменяющего текст
  • Ничто не остановит вас от создания нескольких свойств, таких как BUTTONS_TXT_SIZE, TEXT_SIZE, LIST_TXT_SIZE и т.д., а затем будет иметь логику, которая, например, занимает процент увеличения текста и рассчитывает правильные размеры для каждого типа управления (поскольку у вас могут быть разные размеры для разных элементов управления)

В то же время, скажем, вы хотите заставить это работать динамически? Создайте простой класс (например, TextSetter), который содержит внутренний список и имеет 3 метода: add, remove и setSize

  • В Activity#onCreate укажите каждый элемент управления, который вы хотите настроить, и используйте TextSetter#set, чтобы добавить его в список
  • Когда пользователь хочет увеличить/уменьшить размер шрифта, возможно, из меню, когда вы справляетесь с этим, просто выполните TextSetter # setSize, в котором вы будете прокручивать список элементов управления, обнаруживать, какой тип он есть, и соответственно корректировать размер текста.

Ответ 4

Для тех, у кого есть проблемы с раздуванием просмотров с пользовательскими атрибутами из @mixels ответа.

Если ваши представления в фрагменте, вам необходимо применить FontStyle также в фрагменте onCreateView() или установить значения этих атрибутов в теме приложения.

Подробнее см. мой aswer здесь.