らじおぱ for Android バージョン2.0.0リリース

in Ladiopa

有償Androidアプリらじおぱ for Androidを開発しバージョン2.0.0としてGoogle Playにリリースしました。 これはiOS版らじおぱバージョン2の仕様を元にAndroidアプリとしてスクラッチから開発したものです。 そしてkawauso.comとしては初のAndroidアプリとなります。

らじおぱ for Android

iOS版との違いが重要だと思いますのでその観点でポイントを列挙します。

  • 設定でプレイヤーのバッファーサイズを変更する機能は省きました。
  • お気に入りの共有機能は(以前から通知機能との兼ね合いが難しかったこともあり)省きました。
  • スワイプ動作による画面遷移は(Androidにはバックボタンがあること、スワイプジェスチャーの認識に難があったことにより)省きました。
  • お気に入りサウンドのファイル自体を入れ替える機能は採用せず、Androidシステムに登録されている通知サウンドを自由に選択できるようにしました。
  • iOSのAirDropに類する機能は省きました。
  • アーカイブが途中で終了してもその時点までのアーカイブは残るようになりました。
  • 内蔵プレイヤーがOgg Vorbisを再生できるようになりました。
  • 内蔵プレイヤーがAACおよびその派生フォーマットを再生できるか否かはデバイスに依存します。
  • 携帯電話網での再生ビットレート上限値(128kbps)の制限を(Androidは特に制約が無いようですので)はずしました。
  • 携帯電話網での自動アーカイブ抑制処理をはずしました。
基本的な目標としてiOS版の機能と使用感をなるべく再現したいというのがありましたが、それについては(ずいぶん苦労はしましたが)まずまず出来たのではないかと思います。

今回の開発でAndroid SDK(ソフトウェア開発キット)やGoogle PlayがiOSと比べてどのような点が優れていてどのような点が劣っているかいろいろと知ることができました。 これについてはまた機会を改めてエントリーを書ければと思います。

このソフトウェアの動作や仕様について、プログのコメントやメールにてぜひお知らせください。 ではでは。


らじおぱバージョン2.0.4リリース

in Ladiopa

もう一ヶ月前の話なんですがiOSアプリらじおぱを更新しバージョン2.0.4としてリリースしました。 このリリースでの変更点は以下のとおりです。

  • アーカイブファイルの表示を修正しました。

らじおぱのアーカイブ機能が利用しているiOS 7バックグラウンドトランスファー機能はまだ非公開の制限や動作仕様が多く、仕様リクエストは出しているものの広範な状況に対応する機能として改善されるためには少し時間がかかりそうです。

そしてそれがらじおぱ for Androidを作らせる直接の動機になったりしたんですが^^;。


Emacs入門 #2 WindowsとMac OS X

in Emacs

前回はGNU EmacsのMac OS X上でのインストール方法について書きました。 Emacsの設定の話に入る前にWindows上でのインストール方法についても書いておきましょう。

これもまたいろいろな方法があると思いますが、最近だとgnu packというパッケージを使うのが簡単だと思います。 現在の最新はemacs-24.2-20121208.exeとなっています。 早速ダウンロードして実行してみましょう。

\Users\kawauso\Downloads\emacs-24.2-20121208.exe

展開されたフォルダにあるEmacs起動プログラムを実行してみます。

\Users\kawauso\Downloads\emacs-24.2-20121208\bin\runemacs.exe
Screen Shot 2014-11-20 at 16.11.49

OKですね。

さらにLinux上でのインストール方法について書いてもよかったのですけれど、各ディストリビューション標準の方法を用いればLinuxユーザーにとっては特に難しい話はないと思いますので割愛します。

Emacsは日本語および日本語入力との長い戦いの歴史を持っていますが(^^;)ここまで紹介した方法でインストールしたものであれば特に意識する必要はないかなと思います。 ただしMacPorts版のデフォルトの設定では日本語が以下のように表示されることと思います。

Screen Shot 2014-11-21 at 08.10.07

これでも特に問題は無いのですが、日本語(いわゆる全角文字)が英数(いわゆる半角文字)の倍の幅にそろえられている方が都合がよい場合も多いでしょう。(ちなみにWindowsのgnu pack版の方はデフォルトでそろえられています。)

その設定を行うためにはEmacsのフォント設定についての少し難しい知識が必要になりますが、最も簡単に行うには以下の設定を行うことだと思っています。

  (set-fontset-font t 'japanese-jisx0213-1 (font-spec :family "Hiragino Kaku Gothic ProN"))
  (set-fontset-font t 'japanese-jisx0213-2 (font-spec :family "Hiragino Kaku Gothic ProN"))
  (set-fontset-font t 'katakana-jisx0201 (font-spec :family "Hiragino Kaku Gothic ProN"))
  (add-to-list 'face-font-rescale-alist '("Hiragino " . 1.2))

1行目と2行目でEmacsが用いるデフォルトの全角日本語フォントを、3行目で半角片仮名フォントを全てHiragino Kaku Gothic ProNに置き換えています。 4行目はその全角フォントの幅が英数フォントに対してちょうど2倍に当たるように調整しています。

この設定をして表示がどうなるか見てみましょう。

Screen Shot 2014-11-21 at 08.26.21

無事幅がそろいましたね。

でもちょっと待ってください、どうやって設定を反映させたのでしょうか? 次回はやっとEmacsの設定の話になります(^^)。


Can I Swift? その4 – 変わらないプロパティ

in Swift

別のテーマに行こうと思いましたがもう少しだけ変わらない変数をテーマにしたいと思います。

変わらないインスタンス変数 in Objective-C

ローカル変数ほど頻度は高くはありませんが、インスタンス変数も初期値を設定した後インスタンスが消滅するまで値が変わらない場合があります。 ではObjective-Cのインスタンス変数にconstをつけるとどうなるでしょうか。

@implementation Book {
    NSString * const title = @"Default Title";
}

Parse Issue: Expected ';' at end of declaration list 値を設定することができません。 Objective-Cでは変わらないインスタンス変数は実現できないようです。 どうしても実現したい場合はそれに近い方法として@propertyで宣言しreadonly属性をつけるなどとして工夫するしかなさそうです。

変わらないプロパティ in Swift

Swiftのletではどうでしょうか。

class Book {
    let title:String = "Default Title"
}

インスタンス変数に相当するものはSwiftではプロパティとなりますがletをつけても値を設定することが可能です。さらに

class Book {
    let title:String

    init() {
        self.title = "Default Title"
    }
}

と値をinit()(イニシャライザ、初期化子)の中で設定することも可能です。 Swiftのletの方が遥かに柔軟に値を設定することができるようですね。

依存関係をともなう初期化

Swiftで依存関係をともなうような複雑な初期化でもプロパティにletをつけることが可能かどうか。具体的なコードで見てみましょう。

class Author {
    let bestsellerTitle:String = "Bestseller Title"
    let book:Book

    init() {
        self.book = Book(author:self)
    }
}

class Book {
    let title:String = "Default Title"

    init(author:Author) {
        self.title = author.bestsellerTitle
    }
}

クラスAuthorはクラスBookのインスタンスをプロパティとして保持し、init()の中で値を設定しています。 一方クラスBookも自分を生成したクラスAuthorのインスタンスをinit()の引数で受け取っています。

これをコンパイルしようとするとクラスAuthorinit()でエラーになります。 Swift Compiler Error: Variable 'self.book' used before being initialized init()self(インスタンス自身)を参照しているのですが、プロパティbookがまだ初期化されていないためエラーになったようです。

この初期化問題の解決に役立ちそうな記述がドキュメントThe Swift Programming LanguageInitializationの節Optional Property Typesにありました。

SwiftにはOptional Type(オプショナルな型)という非常に重要な考え方があります。 オプショナルな型は通常の型と違い、値がnilをとることを許します。 逆にいえばSwiftでは通常の型で宣言した場合、値がnilをとることを許していません。 これはObjective-Cでは無かった区別で、プログラムがより安全になる反面、その選択には注意が必要になります。 表記方法は簡単で通常の型に対してその後ろに?をつければ(対応する)オプショナルな型になります。

オプショナルな型を使って先ほどのコードを書き直してみましょう。

class Author {
    let bestsellerTitle:String = "Bestseller Title"
    let book:Book?

    init() {
        self.book = Book(author:self)
    }
}

class Book {
    let title:String = "Default Title"

    init(author:Author) {
        self.title = author.bestsellerTitle
    }
}

書き直すといってもlet book:Booklet book:Book?にしただけですが、これによりクラスAuthorinit()でのコンパイルエラーが無くなりました。 何もしなくてもプロパティbookにはnilが設定されるようになったからです。

しかしこのままだと少し不便なことがあります。 プロパティbookをオプショナルな型で宣言してしまったがために、その後bookを参照する度に値がnilかどうか条件判定するはめになります。 bookletがついていてかつinit()で値を設定していますのでnilにはなることは無く、条件判定は全く不要です。

Swiftはこのように文法上はオプショナルな型であることを必要とするものの、処理上はオプショナルな型であることを不要とする場合、?の代わりに!を用いてオプショナルな型を宣言することができます(この型は無条件で開封されるオプショナルな型(Implicitly Unwrapped Optional Type)と呼ぶようです)。 これでわずらわしい条件判定を意識しないで参照できるようになります。 念のため再度書き直してみましょう。

class Author {
    let bestsellerTitle:String = "Bestseller Title"
    let book:Book!

    init() {
        self.book = Book(author:self)
    }
}

class Book {
    let title:String = "Default Title"

    init(author:Author) {
        self.title = author.bestsellerTitle
    }
}

これでOKです。

変わらない変数の話は終わりにして次回はやっと別のテーマに移ろうと思います。 ではでは。


Can I Swift? その3 – 値の設定の手法

in Swift

Objective-Cはconstをつけることによって、Swiftはletによって変わらない変数を表現できることがわかりました。 しかし変わらない変数は変えることができないが故に1回で値を設定しなければならないという大きな制約があります。 その制約が具体的にどのようなものかコードで見てみましょう。

条件つき値の設定 in Objective-C

通常のローカル変数を使った以下のようなコードがあったとします。

        NSString * title;

        title = getTitle();
        if (title == nil) {
            title = @"Default Title";
        }

変数titleにまず関数getTitle()で取得した値を設定し、その値がnilだった場合には@"Default Title"に設定し直しています。 このような設定処理は実際のプログラムによく現れますが、もし変数titleの値がその後変更されるものでなければ、変わらない変数として宣言して処理を行いたいところです。 書き直してみましょう。

    NSString * const tempTitle = getTitle();
    NSString * const title = (tempTitle != nil ? tempTitle : @"Default Title");

一時変数を別に用意し、三項演算子(?)を用いることによってなんとか変数titleに1回で値を設定することができました。

実はこのような設定処理は気のきいたプログラミング言語においては例えば以下のように簡潔に記述することが可能です。

    String title = (getTitle() || "Default Title")

これが可能になるのはOR演算子(||)が論理値(真OR偽)を返すのではなく、真となった値(この場合は"Default Title")を返すタイプのプログラミング言語の場合です。 残念ながらCを親に持つObjective-Cではこの記述はできません。

条件つき値の設定 in Swift

新言語Swiftでは気のきいた記述が可能になっているかどうか早速テストしてみましょう。

    let title:String = (getTitle() || "Default Title")

Swift Compiler Error: Type 'String' does not conform to protocol 'LogicValue' 残念ながらSwiftでもOR演算子は論理値を返し、Objective-Cから変わりないようです。

クロージャを用いた値の設定

Swiftでの値の設定についてドキュメントThe Swift Programming LanguageInitializationの章にこのような例が載っています。

    let someProperty: SomeType = {
        // create a default value for someProperty inside this closure
        // someValue must be of the same type as SomeType
        return someValue
    }()

値を生成するための複雑な処理をクロージャで記述し、即実行することで値を取得し設定しています。 クロージャはこのような使い方もできて便利ですね。 これが書ければSwiftで値を設定するのに困ることはなさそうです。

しかしちょっと思い出してください。 Objective-Cにもクロージャに相当するBlocksという仕組みがMac OS X 10.6以降、iOS 4以降使えるようになりました。 Blocksを用いてObjective-C版を書き直してみましょう。

    NSString * const title = ^ NSString * {
        NSString * const tempTitle = getTitle();
        if (tempTitle != nil) {
            return tempTitle;
        }
        return @"Default Title";
    }();

見事に値を設定することができました。 この程度の複雑さですと大した恩恵はありませんが、処理が複雑になればなるほどBlocksを利用した値の設定の手法は有用になっていくでしょう。

変わらない変数についての話はこれぐらいとして、次回はまた別のテーマでSwiftを見ていきたいと思います。 ではでは。