4
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Unistyles 3.0を使ったらスタイルの定義がよりスッキリ書けそうな話

Last updated at Posted at 2025-04-21

はじめに

こんにちは、menu株式会社フロントエンドエンジニアの内田です。
menuのモバイルアプリの多くはReact Nativeを使っています。

突然ですが、みなさんはUnistylesをご存知ですか?
2025年現在、3.0がベータ版として公開されています。
すごく便利だなと思ったので、今回はUnistylesについて紹介します。

Unistylesってなに?

Unistylesは、React NativeのStyleSheetを拡張したライブラリです。JavaScriptにおけるTypeScriptのような関係と言えます。従来のStyleSheetの利点を継承しつつ、より強力で柔軟なスタイリングを提供します。まだベータ版ではありますが、Expoなどのフレームワークからも注目されており、React Nativeにおけるスタイリングの新たな選択肢として期待されています。

ドキュメントにも記載されている通り、Unistylesは特にテーマの適用やvariantによるスタイルの切り替えを必要とするUIコンポーネントに有効です。また、アプリ全体の再レンダリングを抑制し、パフォーマンス向上にも貢献します(New Architecture最適化)。

一方で、シンプルなアプリなど、高度なスタイリング機能が不要な場合には、Unistylesの恩恵は限定的かもしれません。また、現時点ではNew Architectureにのみ対応しており、Expo Goでは実行できないなどの制約もあります。

Unistylesの基本的な使い方

従来のスタイリングの辛いところ

まず、従来のReact Nativeでのスタイリングの問題点を確認します。以下のコードは、variant、size、disabledといったpropsに基づいてスタイルを切り替えるボタンコンポーネントの例です。

ボタンコンポーネント実装例
export const ContainedButton: React.FC<ContainedButtonProps> = ({
  label,
  variant = 'primary',
  size = 'medium',
  disabled = false,
  style,
  onPress,
}) => {
  const styles = useMemo(
    () =>
      StyleSheet.create({
        button: {
          alignItems: 'center',
          backgroundColor: disabled ? colors.button.disabled : colors.button[variant],
          borderRadius: 4,
          justifyContent: 'center',
          opacity: disabled ? 0.5 : 1,
          ...getSizeStyles(size),
        },
        text: {
          color: disabled
            ? colors.text.onDisabled
            : variant === 'inverse'
              ? colors.text.onInverse
              : colors.text.onPrimary,
          fontFamily: 'Poppins-Regular',
          fontSize: 16,
        },
      }),
    [variant, size, disabled]
  );

  return (
    <Pressable
      style={({ pressed }) => [
        styles.button,
        {
          opacity: pressed ? 0.8 : 1,
        },
        style as StyleProp<ViewStyle>,
      ]}
      onPress={onPress}
      disabled={disabled}
      accessibilityRole="button"
      accessibilityState={{ disabled }}
    >
      <Text style={styles.text}>{label}</Text>
    </Pressable>
  );
};

function getSizeStyles(size: ButtonSize) {
  switch (size) {
    case 'small':
      return {
        paddingVertical: 2,
        paddingHorizontal: 4,
      };
    case 'large':
      return {
        paddingVertical: 6,
        paddingHorizontal: 12,
      };
    case 'medium':
    default:
      return {
        paddingVertical: 4,
        paddingHorizontal: 8,
      };
  }
}

このコンポーネントでは、propsの値に応じてスタイルを切り替えるために,render処理内に条件分岐を記述する必要があり、コードが複雑になりがちです。また、これに加えてテーマなども注入するとなると、色々ごちゃごちゃしてきそうです。

Unistylesによるスタイリング

ではUnistylesを使用するとどうなるのでしょう。ありがたいことに、Unistylesは、React Native提供のStyleSheetと同じシンタックスで記述できます。

variantsキーを使用することで、動的に変更するスタイルプロパティを宣言的に記述できます。これにより、スタイルの定義をコンポーネントの外部に分離でき、コードの見通しが良くなります。

スタイル実装例
const styles = StyleSheet.create(() => ({
  button: {
    alignItems: 'center',
    borderRadius: 4,
    justifyContent: 'center',

    variants: {
      color: {
        primary: {
          backgroundColor: colors.button.primary,
        },
        secondary: {
          backgroundColor: colors.button.secondary,
        },
        tertiary: {
          backgroundColor: colors.button.tertiary,
        },
        inverse: {
          backgroundColor: colors.button.inverse,
        },
      },
      disabled: {
        true: {
          backgroundColor: colors.button.disabled,
          opacity: 0.5,
        },
        false: {},
      },
      size: {
        small: {
          paddingVertical: 2,
          paddingHorizontal: 4,
        },
        medium: {
          paddingVertical: 4,
          paddingHorizontal: 8,
        },
        large: {
          paddingVertical: 6,
          paddingHorizontal: 12,
        },
      },
    },
  },
  text: {
    fontFamily: 'Poppins-Regular',
    fontSize: 16,
    variants: {
      color: {
        primary: {
          color: colors.text.onPrimary,
        },
        secondary: {
          color: colors.text.onSecondary,
        },
        tertiary: {
          color: colors.text.onTertiary,
        },
        inverse: {
          color: colors.text.onInverse,
        },
      },
    },
  },
}));

そして、コンポーネント内では、useVariantsフックを使用して、適用するvariantを指定します。

ボタンコンポーネント with Unistyles 実装例
export const ContainedButton: React.FC<ContainedButtonProps> = ({
  label,
  variant = 'primary',
  size = 'medium',
  disabled = false,
  style,
  onPress,
}) => {
  styles.useVariants({
    color: variant,
    disabled,
    size,
  });

  return (
    <Pressable
      style={({ pressed }) => [
        styles.button,
        {
          opacity: pressed ? 0.8 : 1,
        },
        style as StyleProp<ViewStyle>,
      ]}
      onPress={onPress}
      disabled={disabled}
      accessibilityRole="button"
      accessibilityState={{ disabled }}
    >
      <Text style={styles.text}>{label}</Text>
    </Pressable>
  );
};

コンポーネント関数からスタイルの処理を分離することができたのでスッキリしましたね。今回は行いませんでしたが、UnistylesのStyleSheetはテーマの注入にも対応しているので、ダークモード・ライトモードの切り替えができるアプリでも非常に重宝しそうです。

まとめ

Unistylesは、React Nativeのスタイリングをより柔軟かつ効率的に行うための強力なライブラリです。variantによるスタイルの切り替えだけでなく、テーマ機能など、さまざまな便利な機能を提供します。今後のメジャーリリースに期待しましょう。

今回は、Unistylesの基本的な使い方として、variantによるスタイルの切り替えを紹介しました。Unistylesのより詳細な情報については、公式ドキュメントを参照してください。

カジュアル面談

モバイルアプリ開発について興味がある・menuの開発チームに興味があるなど、カジュアルに情報交換をしたい方を積極的に募集中です。下記のリンクからお気軽にご連絡ください。


▼採用情報

レアゾン・ホールディングスは、「世界一の企業へ」というビジョンを掲げ、「新しい"当たり前"を作り続ける」というミッションを推進しています。

現在、エンジニア採用を積極的に行っておりますので、ご興味をお持ちいただけましたら、ぜひ下記リンクからご応募ください。

4
1
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
4
1

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?