from __future__ import annotations
from abc import ABC, abstractmethod
from typing import Optional, TypedDict
from .color import Color
from .position import Positions
[ドキュメント]class M5ComDict(TypedDict):
    """
    M5から取得する情報を格納するDict。
    :param din0: ヘッドのdin0ピンのデジタル入力。HiがTrue、LoがFalse。デフォルトHi。
    :param din1: ヘッドのdin1ピンのデジタル入力。HiがTrue、LoがFalse。デフォルトHi。
    :param ain0: ヘッドのain0ピンのアナログ入力。0-3.3Vを0-4095の4096段階にする。
    :param dout0: ヘッドのdout0ピンのデジタル出力。TrueでHi(3.3V)、FalseでLo(0V)。
    :param dout1: ヘッドのdout1ピンのデジタル出力。TrueでHi(3.3V)、FalseでLo(0V)。
    :param pwmout0: ヘッドのpwmout0ピンのPWM出力。0-255の256段階で0-3.3Vを出力可能。
    :param general0: 任意値を入力可能なエリア。通常時は使用しない。M5のソフトを書き換えることで自由に使用可能。
    :param general1: 任意値を入力可能なエリア。通常時は使用しない。M5のソフトを書き換えることで自由に使用可能。
    :param button_a: M5の前面のボタンAの入力。ボタンを押すとTrue、押していないとFalse。
    :param button_b: M5の前面のボタンBの入力。ボタンを押すとTrue、押していないとFalse。
    :param button_c: M5の前面のボタンCの入力。ボタンを押すとTrue、押していないとFalse。
    :param temperature: AKARI側面の環境センサモジュールの温度計測値。単位は℃。
    :param pressure: AKARI側面の環境センサモジュールの観測絶対圧力。単位はPa。
    :param brightness: AKARI側面の光センサユニットの観測明るさを4096段階で出力。0が最も明るい。
    :param time: データ取得した時間。M5のソフト起動時からの経過秒数を用いている。
    :param is_responce: akari_client()の各コマンド実行完了時のレスポンス。実行完了するとTrueを返す。
    """
    din0: bool
    din1: bool
    ain0: int
    dout0: bool
    dout1: bool
    pwmout0: int
    general0: float
    general1: float
    button_a: bool
    button_b: bool
    button_c: bool
    temperature: float
    pressure: float
    brightness: int
    time: float
    is_response: bool 
[ドキュメント]class M5StackClient(ABC):
[ドキュメント]    @abstractmethod
    def set_dout(self, pin_id: int, value: bool, sync: bool = True) -> None:
        """ヘッド部GPIOピンのデジタル出力を設定する。
        Args:
            pin_id: pin番号。0でdout0、1でdout1を指定する。
            value: デジタル出力の値。Falseで0V,Trueで3.3V出力。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>     m5 = akari.m5stack
            >>>     m5.set_dout(0, True)
            # dout0がTrueになる。
        """
        ... 
[ドキュメント]    @abstractmethod
    def set_pwmout(self, pin_id: int, value: int, sync: bool = True) -> None:
        """ヘッド部GPIOピンのPWM出力を設定する。
        Args:
            pin_id: pin番号。デフォルトではpwmout0しかないため、0を指定すること。
            value: PWM出力の値。 0-255で指定し、0で0V、255で3.3Vを出力する。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>     m5 = akari.m5stack
            >>>     m5.set_pwmout(0, 200)
            # pwmout0が200になる。
        """
        ... 
[ドキュメント]    @abstractmethod
    def set_allout(
        self,
        *,
        dout0: Optional[bool] = None,
        dout1: Optional[bool] = None,
        pwmout0: Optional[int] = None,
        sync: bool = True,
    ) -> None:
        """ヘッド部GPIOピンの出力をまとめて設定する。
        Args:
            dout0: dout0の出力値。Falseで0V,Trueで3.3V出力。
            dout1: dout1の出力値。Falseで0V,Trueで3.3V出力。
            pwmout0: pwmout0の出力値。0-255で指定し、0で0V、255で3.3Vを出力する。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>     m5 = akari.m5stack
            >>>     m5.set_allout(dout0=False, dout1=True, pwmout0=100)
            # dout0がFalse, dout1がTrue, pwmout0が100になる。
        """
        ... 
[ドキュメント]    @abstractmethod
    def reset_allout(self, sync: bool = True) -> None:
        """ヘッド部GPIOピンの出力を初期値にリセットする。
        dout0、dout1はFalse、pwmout0は0となる。
        Args:
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>     m5 = akari.m5stack
            >>>     m5.reset_allout()
            # dout0がFalse, dout1がFalse, pwmout0が0になる。
        """
        ... 
[ドキュメント]    @abstractmethod
    def set_display_color(
        self,
        color: Color,
        sync: bool = True,
    ) -> None:
        """ボディー部M5のディスプレイ背景色を変更する。
        Args:
            color: 背景色を指定。色は ``color.Colors`` から色名を引用する、もしくはRGBの数値指定も可能。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>     m5 = akari.m5stack
            >>>     color = Colors.WHITE
            >>>     m5.set_display_color(color)
            # 画面色が白になる。
            >>>     color = Color(red=0, green=100, blue=200)
            >>>     m5.set_display_color(color)
            # 画面色が指定したRGB値になる。
        """
        ... 
[ドキュメント]    @abstractmethod
    def set_display_text(
        self,
        text: str,
        pos_x: int = Positions.CENTER,
        pos_y: int = Positions.CENTER,
        size: int = 5,
        text_color: Optional[Color] = None,
        back_color: Optional[Color] = None,
        refresh: bool = True,
        sync: bool = True,
    ) -> None:
        """ボディー部M5のディスプレイに文字を表示する。
        Args:
            text: 表示する文字列。
            pos_x: x方向の描画位置ピクセルを0-320で指定。左端が0。 ``position.Positions`` を用いた位置指定も可能。デフォルト値は中央揃え。
            pos_y: y方向の描画位置ピクセルを0-240で指定。上端が0。 ``position.Positions`` を用いた位置指定も可能。デフォルト値は中央揃え。
            size: 文字サイズを1-11の11段階で指定。デフォルト値は5。
            text_color: 文字色を指定。色は ``color.Colors`` から色名を引用する、もしくはRGBの数値指定も可能。値を指定しない場合、前回値を引き継ぐ。
            back_color: 背景色を指定。色は ``color.Colors`` から色名を引用する、もしくはRGBの数値指定も可能。値を指定しない場合、画面全体の背景色に合わせる。
            refresh: trueの場合画面全体を現在の背景色で更新する。falseの場合は現在の表示を維持しつつ、文字を描画する範囲のみ更新する。デフォルト値は背景更新あり。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> from akari_client.color import Colors
            >>> from akari_client.position import Positions
            >>> with AkariClient() as akari:
            >>>    text = "AKARI"
            >>>    pos_x = Positions.LEFT
            >>>    pos_y = Positions.TOP
            >>>    size = 7
            >>>    text_color = Colors.WHITE
            >>>    back_color = Colors.BLACK
            >>>    refresh = True
            >>>    m5.set_display_text(
            ...        text, pos_x, pos_y, size, text_color, back_color, refresh
            ...    )
            # 画面に「AKARI」と表示される。
        """
        ... 
[ドキュメント]    @abstractmethod
    def set_display_image(
        self,
        filepath: str,
        pos_x: int = Positions.CENTER,
        pos_y: int = Positions.CENTER,
        scale: float = -1.0,
        sync: bool = True,
    ) -> None:
        """ボディー部M5のディスプレイにM5のSDカード内の画像を表示する。
        Args:
            filepath: M5のSDカード内のファイルパス。(例;"image/hoge.jpg")
            pos_x: x方向の描画位置ピクセルを0-320で指定。左端が0。 ``position.Positions`` を用いた位置指定も可能。デフォルト値は中央揃え。
            pos_y: y方向の描画位置ピクセルを0-240で指定。上端が0。 ``position.Positions`` を用いた位置指定も可能。デフォルト値は中央揃え。
            scale: 画像の拡大縮小倍率を指定。1.0で等倍表示。マイナスの値を入れた場合、画面サイズに合わせて自動でサイズ調整される。
                   デフォルト値は自動サイズ調整となっている。
            sync: 同期実行の指定。Trueの場合M5側で実行完了するまで関数の終了待ちを行う。
        Example:
            >>> from akari_client import AkariClient
            >>> from akari_client.position import Positions
            >>> with AkariClient() as akari:
            >>>    filepath = "/jpg/logo320.jpg"
            >>>    pos_x = Positions.LEFT
            >>>    pos_y = Positions.TOP
            >>>    scale = 0.75
            >>>    m5.set_display_image(filepath, pos_x, pos_y, scale)
            # 画面に"/jpg/logo320.jpg"の画像が表示される。
        """
        ... 
[ドキュメント]    @abstractmethod
    def reset_m5(self) -> None:
        """M5をリセットして再起動する。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>    m5.reset_m5()
            # M5が再起動する。
        """
        ... 
[ドキュメント]    @abstractmethod
    def get(self) -> M5ComDict:
        """M5から環境センサ、ヘッドGPIOの入力値、M5のボタンの状態を取得する。
        Returns:
            M5から取得した各データを格納したDict。
        Example:
            >>> from akari_client import AkariClient
            >>> with AkariClient() as akari:
            >>>    data = m5.get()
            >>>    print(data["temperature"])
            31.675
        """
        ...