libqrencode を使用したサンプル QR コード作成ツール QRURL と静的ライブラリ libqrencode.a を公開。

OS X Mountain Lion が 7月25日(米国時間)に発売されました。最初のダウンロードは混むので、一日置いてから Mac App Store でダウンロードしました。何の問題もなく OS X Lion から OS X Mountain Lion アップデートできました。後、念のためクリーンインストールしようか、悩んでいます。

また、北米版プチコンmkII : Petit Computer が 2012年7月19日に発売されました。

Nintendo.com – Petit Computer – Game Info
http://www.nintendo.com/games/detail/_YpH-64pY4G4_tAy1wrKVecHp4pmWSzw
Top Page - Petit Computer
http://www.petitcomputer.com/
スマイルブーム.com
http://smileboom.com/blog/

上のブログによると QR コードも共通仕様なので、プチコンmkII 版のプログラムは、北米版 Petit Computer で遊べるみたいです。(一部、TALK 文などが対応していません。) 私の公開しているプログラムも英語の解説を付けたいのですが、英語が苦手なので無理ですね。ホームページのアクセス解析によると海外のアクセス数が増加しています。また、拙作プチコンmkII 版マインスイーパのリンクを紹介した海外サイトもあります。

話は変わって、以前から QR コードの作成プログラムに興味がありました。

ニンテンドーDSi ウェア ‘プチコンmkII’ は、ホームページ上で SDカードに保存したファイルを QR コード変換ツールを使って QR コードに変換します。

QR コード変換ツールを

PTCUtilities v0.7.3: ROBOMIC(ブログ)
http://micono.cocolog-nifty.com/blog/2012/05/ptcutilities--5.html
PTCUtilites
http://micutil.com/ptcutilities/top.html

上記 URL で公開した人がいます。

Mac 版と Windows 版が作成されていたので、どんなツールで作成されていたのか興味がありました。

実行ファイル PTCUtilities をマウスで右クリック、パッケージの内容を表示を選ぶと

PTCUtilitiesMac/PTCUtilities.app/Contents/MacOS/rbframework.dylib
PTCUtilitiesMac/PTCUtilities.app/Contents/Frameworks/MBS Real Studio Cocoa Plugin.rbx_16.dylib

と表示されます。rb の接頭語から、私が Mac OS 9 時代に使用していた REALbasic を思い出しました。Wikipedia で検索してみると

REALbasic - Wikipedia
http://ja.wikipedia.org/wiki/REALbasic
REALbasicの現状

今は REAL Studio という名称になって発売されているようです。

QR コード作成ツールを作ってみたいと思い、QR コードのライブラリを Google で検索してみました。

libqrencode
http://fukuchi.org/works/qrencode/index.html.ja

上記 URL で公開されている C のライブラリが良さそうです。開発環境をなるべくデフォルトの状態にしたいので、ライブラリのビルドは、Parallels DesktopOS X Lion 上でおこないました。

上記 URL から、ファイル qrencode-3.3.1.tar.gz をダウンロードし、ダブルクリックすると qrencode-3.3.1 という名前のフォルダが作成されます。ターミナルでそのフォルダに移動し、./configure を実行すると pkg-config のバージョンが古いというエラーが出ます。

それで Homebrew を使って

brew install pkg-config

最新の pkg-config をインストールしました。

./configure
make

で問題なくビルドに成功しました。しかし、静的ライブラリ libqrencode.a を作ってくれません。また、ビルドして作成される実行ファイル qrencode は、png ライブラリを使用しています。しかし、libqrencode ライブラリ自体は、どのライブラリにも依存していませんでした。

それで Xcode で無理やり、静的ライブラリを作成することにしました。png ライブラリを使用している qrenc.c 以外の拡張子が C のファイルと config.h 含むヘッダファイルをプロジェクトに追加し、C のファイルのヘッダ読み込み

#if HAVE_CONFIG_H
# include "config.h"
#endif

の箇所から

#if HAVE_CONFIG_H
#endif

を削除し、ビルドしてみました。警告が表示されますが、ビルドに成功しました。

簡単な QR コード作成ツール QRURL と静的ライブラリ libqrencode.a を一緒に公開します。

QRURL
libqrencode.a
http://kyoshiaki.sakura.ne.jp/osx/Sample/qrcode.zip

ここで、振り出しに戻るのですが、

リンク集 - プチコンまとめWiki
http://wiki.hosiken.jp/petc/?Link
PTC2QR

上記 URL で Ruby のソース付きQR コード変換ツール PTC2QR が公開されています。

ocra-1.2.0 Documentation
http://ocra.rubyforge.org/

初めて知ったのですが、Ruby のソースファイルから Windows 実行ファイルを作るプログラム (ocra 上記 URL 参照) があるんですね!

PTC2QR のソースを Ruby から Objective-C に書き換えてみました。何とか動作するものができました。著作権上、公開は無理なので、スクリーンショットだけ載せておきます。

WWDC、E3、Objective-C のブロックと変数

WWDC 2012 基調講演が、11日午前10時 (日本時間、翌12日午前2時) に開催されました。Mac Book Air / Pro、iOS 6、OS X Mountain Lion の発表がありましたが、一番の関心は、Retina ディスプレイ MacBook Pro の発表です。iPadiPhone アプリ開発のために、iMac 27 インチを使用してきましたが、Retina ディスプレイ MacBook Pro でも問題なさそうです。所有してる iMac (Mid 2010) 27 インチは、Thunderbolt、USB 3.0 に対応していないので、購入時期も迫ってきています。Retina ディスプレイ iMac が発表されていたら、間違いなく購入していました。もう一年、我慢しようか、悩みどころです。

基調講演は、Apple のサイト

Apple - Apple Events - Apple Special Event March 2012
http://events.apple.com.edgesuite.net/123pibhargjknawdconwecown/event/index.html

で見れます。Podcast でも配信されています。

iTunes - Podcast - Apple Inc.「Apple Keynotes」
http://itunes.apple.com/jp/podcast/apple-keynotes/id275834665
iTunes - Podcast - Apple Inc.「Apple Keynotes (1080p)」
http://itunes.apple.com/jp/podcast/apple-keynotes-1080p/id509310064
iTunes - Podcast - Apple Inc.「Apple Keynotes (HD)」
http://itunes.apple.com/jp/podcast/apple-keynotes-hd/id470664050

1080p、HD 版も配信されています。

E3 も開催されていました。

E3 2012 - GameSpot.com
http://e3.gamespot.com/

上のサイト、左上の赤の背景に白色の文字で E3 2012 と表示されている部分をクリックし、Press Conferences を選ぶとそれぞれの基調講演を見ることができます。また、任天堂の基調講演は

任天堂 E3 2012情報 | E3レポート
http://www.nintendo.co.jp/n10/e3_2012/report/index.html

でも見れます。

以前、[id:KYoshiaki:20120429] iCloud プログラミングで書いた

Creating iOS 5 Apps: Develop and Design

Creating iOS 5 Apps: Develop and Design

[Kindle 版] と復習のために買った

Objective-C Programming: The Big Nerd Ranch Guide (Big Nerd Ranch Guides)

Objective-C Programming: The Big Nerd Ranch Guide (Big Nerd Ranch Guides)

[Kindle 版] を読み終えました。

Blocks とスコープ外の変数について、理解が深まりました。

typedef void (^ArrayEnumerationBlock)(id, NSUInteger, BOOL *);
ArrayEnumerationBlock devowelizer;
NSMutableArray *newStrings = [NSMutableArray array];
devowelizer = ^(id string, NSUInteger i, BOOL *stop) {
NSMutableString *newString = [NSMutableString stringWithString:string];
for (NSString *s in vowels) {
NSRange fullRange = NSMakeRange(0, [newString length]);
[newString replaceOccurrencesOfString:s withString:@"" options:NSCaseInsensitiveSearch range:fullRange];
}
[newStrings addObject:newString];
};
[oldStrings enumerateObjectsUsingBlock:devowelizer];

上のように、block 内で

[newStrings addObject:newString];

とスコープ外の変数 newStrings の内容を変更することができます。これは、メソッドを呼んでいるからです。プロパティで代入する場合も、同じです。

typedef void (^SampleBlock)();
__block NSString *two = [NSString stringWithString:@"two"];
SampleBlock sampleBlock;
FooClass *fooClass;
fooClass = [[FooClass alloc] init];
sampleBlock = ^() {
NSString *three = [NSString stringWithString:@"three"];
fooClass.foo = [NSString stringWithString:@"Orange"];
two = three;
changeValue(8);
};
sampleBlock();

ブロック内で

fooClass.foo = [NSString stringWithString:@"Orange"];

と代入することができます。ここでは、説明しやすいように

self.foo = bar;

とします。

実際には、ドット演算子を使ったプロパティは

The Objective-C Programming Language: Declared Properties
https://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/ObjectiveC/Chapters/ocProperties.html#//apple_ref/doc/uid/TP30001163-CH17-SW1
▼ Declared Properties
▼ Property Declaration and Implementation
Property Declaration
Key-Value Coding Programming Guide: Key-Value Coding Fundamentals
https://developer.apple.com/library/ios/#documentation/Cocoa/Conceptual/KeyValueCoding/Articles/BasicPrinciples.html#//apple_ref/doc/uid/20002170-BA
▼Key-Value Coding Fundamentals
Dot Syntax and Key-Value Coding

を利用しているので

[self setFoo:bar];

または

[self setValue:bar forKey:@"foo"];

と同一です。故にメソッドを使っているので、block 内で代入できます。

foo = bar;

は、block 内で使えないので注意してください。foo が Objective-C のオブジェクト、または int、double などのスカラー型いずれの場合でも、__block を宣言しない限り、ブロック内では代入できません。

sample(bar);

上のような C の関数は、ブロック内で使えます。

結局、Cの関数、Objective-C の method は、ブロック内で使うことができます。また、プロパティを使ったドット表記も、実際には method を使っているので、ブロック内で使えます。ブロック内でスコープ外の変数に = (イコール) を使った代入は、__block を宣言しないと使えません。

サンプルとして zip ファイルを公開します。

VowelMovement.zip
http://kyoshiaki.sakura.ne.jp/osx/Sample/VowelMovement.zip
Creating iOS 5 Apps: Develop and Design

Creating iOS 5 Apps: Develop and Design

[Kindle 版] で見つけた面白い Blocks の使い方のサンプルも下記 URL で公開します。

Weights.zip
http://kyoshiaki.sakura.ne.jp/osx/Sample/Weights.zip

サンプルは、[ Creating iOS 5 Apps: Develop and Design ] [Kindle 版] のソースから、面白い使い方の箇所を抜き出して修正したものです。

//
//	Weights.m
//
- (void)processWeightEntryUsingBlock:(void (^)(NSNumber*)) block
{
NSLog(@"Weights.entries: %@", self.entries);
for (NSNumber* entry in self.entries) {
block(entry);
}
}
//
//	States.m
//
-(void)doStates
{
NSMutableArray *array;
array = [NSMutableArray array];
[self.weights processWeightEntryUsingBlock:^(NSNumber *number) {
int value;
NSNumber *num;
value = [number intValue];
num = [NSNumber numberWithInt:value*2];
[array addObject:num];
}];
NSLog(@"processWeightEntryUsingBlock: %@", array);
}

Homebrew をちょこっと解析、Cocoa Emacs インストール。

前回書いたように、パッケージシステムとして MacPorts から Homebrew に変更してみました。

Installation - GitHub
https://github.com/mxcl/homebrew/wiki/installation

Homebrew をインストールするには、上記ページの Installation で表示されているスクリプト

/usr/bin/ruby -e "$(curl -fsSL https://raw.github.com/gist/323731)"

をコピーし、ターミナル上でペーストして実行するだけです。ただし、上記に記載されている通り、事前に Xcode をインストールしておく必要があります。

Safari で直接、https://raw.github.com/gist/323731にアクセスすれば、Homebrew をインストールする Ruby スクリプトを見ることができます。 (curl コマンドでもダウンロードできます。)

curl は、サーバから、あるいはサーバへ、データ伝送を行うツールです。プロトコル HTTP、HTTPSFTP などに対応してます。上のスクリプトの場合、指定した URL [ https://raw.github.com/gist/323731 ] が返す Ruby スクリプトをダウンロードします。各引数は、ターミナルで man curl を使って確認してください。(man curl | col -bfx > curl.txt でファイルに変換して、open curl.txt で中身を確認すると便利です。)

その後、/usr/bin/ruby -e コマンドで Ruby スクリプトを実行します。

スクリプトの内容は、簡単に説明すると

/usr/local/.git ディレクトリの中身が空じゃない時、スクリプトを終了。

/usr/local/
bin/
Cellar/
etc/
include/
lib/
Library/
sbin/
share/
var/
.git/

のそれぞれのディレクトリが存在しない時、/bin/mkdir で新規作成、存在する時は属性を変更。

ディレクトリ /usr/local に移動し、https://github.com/mxcl/homebrew/tarball/master から mxcl-homebrew-xxxxxxxx.tar.gz ファイルをダウンロードして展開する。

これで終了です。

Homebrew のディレクトリのおおざっぱな中身は

/usr/local/
bin/
brew
Cellar/
Library/
Aliases
Contributions
brew_bash_completion.sh
brew_fish_completion.fish
brew_zsh_completion.zsh
examples/
brew-bottle.rb
brew-depstree.rb
brew-dirty.rb
brew-graph
brew-grep
brew-leaves.rb
brew-linkapps.rb
brew-man
brew-missing.rb
brew-pull.rb
brew-readall.rb
brew-server
brew-switch.rb
brew-unpack.rb
brew-upgrade.rb
brew-which.rb
manpages/
Formula/
emacs.rb
wget.rb
.
.
Homebrew/
blacklist.rb
cleaner.rb
compatibility.rb
download_strategy.rb
exceptions.rb
formula.rb
formula_installer.rb
global.rb
hardware.rb
install.rb
keg.rb
keg_fix_install_names.rb
utils.rb
cmd/
---cache.rb
--cellar.rb
--config.rb
--env.rb
--prefix.rb
--repository.rb
audit.rb
cat.rb
cleanup.rb
create.rb
deps.rb
diy.rb
doctor.rb
edit.rb
fetch.rb
help.rb
home.rb
info.rb
install.rb
link.rb
list.rb
log.rb
options.rb
outdated.rb
prune.rb
search.rb
test.rb
uninstall.rb
unlink.rb
update.rb
uses.rb
extend/
ARGV.rb
ENV.rb
pathname.rb
string.rb
test/
share/

です。

また、キャッシュとして

~/Library/Caches/Homebrew 

フォルダが利用されます。Library フォルダは不可視になっていますが、ターミナル上で

~ $ open Library

で開くことができます。

brew コマンドのヘルプは、

man brew
brew help

で確認できます。

ここでは、wget のパッケージをインストールするコマンド

brew install wget

の動作を簡単に説明したいと思います。間違っている個所もあるかもしれません。

実行ファイル /usr/bin/brew は、rubyスクリプトです。

#--- /usr/local/ibin/brew ---
cmd = ARGV.shift # => 'install'
cmd = aliases[cmd] if aliases[cmd] # cmd = :install

cmd に :install が代入され

#--- /usr/local/ibin/brew ---
elsif require? HOMEBREW_REPOSITORY/"Library/Homebrew/cmd"/cmd
Homebrew.send cmd.to_s.gsub('-', '_')

/usr/local/Library/Homebrew/cmd/install.rb が読み込まれ、module Homebrew は /usr/local/Library/Homebrew/cmd/install.rb に宣言されています。

send (Object) - Rubyリファレンス
http://ref.xaio.jp/ruby/classes/object/send

Homebrew.send は Homebrew モジュールの install メソッドを呼ぶことになります。

#--- /usr/local/Library/Homebrew/cmd/install.rb ---
module Homebrew extend self
def install
ARGV.named.each do |name|
msg = blacklisted? name
raise "No available formula for #{name}\n#{msg}" if msg
end unless ARGV.force?
install_formulae ARGV.formulae
end

ARGV は

#--- /usr/local/bin/brew ---
require 'global'

#--- /usr/local/Library/Homebrew/global.rb ---
require 'extend/pathname'
require 'extend/ARGV'
require 'extend/string'
require 'utils'
require 'exceptions'
ARGV.extend(HomebrewArgvExtension)

から /usr/local/Library/Homebrew/extend/ARGV.rb で拡張されています。

#--- /usr/local/iLibrary/Homebrew/cmd/install.rb ---
module Homebrew extend self
def install
ARGV.named.each do |name|
msg = blacklisted? name
raise "No available formula for #{name}\n#{msg}" if msg
end unless ARGV.force?
install_formulae ARGV.formulae
end

/usr/local/Library/Homebrew/blacklist.rb の blacklisted? でインストールできないパッケージを判断。

#--- /usr/local/iLibrary/Homebrew/cmd/install.rb ---
install_formulae ARGV.formulae

ARGV.formulae は Arrray 型で、インストールする Foumla が入っています。今回は Class Wget が1つ入っています。Class Wget は、/usr/local/Library/Formula/wget.rb で宣言されています。

#--- /usr/local/iLibrary/Homebrew/formula_installer.rb ---
def install_formulae formulae
formulae = [formulae].flatten.compact
return if formulae.empty?
check_ppc
check_writable_install_location
check_cc
check_macports
formulae.each do |f|
begin
installer = FormulaInstaller.new f
installer.ignore_deps = ARGV.include? '--ignore-dependencies'
installer.go
rescue FormulaAlreadyInstalledError => e
opoo e.message
end
end
end

f は、Formula クラスを継承した Wget クラスが入っています。FormulaInstaller の @f にも格納されています。

#--- /usr/local/iLibrary/Homebrew/formula_installer.rb ---
def go
if @f.installed? and not ARGV.force?
raise FormulaAlreadyInstalledError, @f
end
unless ignore_deps
needed_deps = @f.recursive_deps.reject {|d| d.installed?}
unless needed_deps.empty?
puts "Also installing dependencies: "+needed_deps*", "
needed_deps.each do |dep|
FormulaInstaller.install_formula dep
end
end
begin
FormulaInstaller.check_external_deps @f
rescue UnsatisfiedExternalDependencyError => e
onoe e.message
exit! 1
end
end
FormulaInstaller.install_formula @f
end

FormulaInstaller.install_formula @f を呼び出すことになります。

#--- /usr/local/iLibrary/Homebrew/formula_installer.rb ---
def self.install_formula f
@attempted ||= Set.new
raise FormulaInstallationAlreadyAttemptedError, f if @attempted.include? f
@attempted << f
ENV['HOMEBREW_ERROR_PIPE'] = write.to_i.to_s
fork do
begin
read.close
exec '/usr/bin/nice',
'/usr/bin/ruby',
'-I', Pathname.new(__FILE__).dirname,
'-rinstall',
'--',
f.path,
*ARGV.options_only
rescue Exception => e
Marshal.dump(e, write)
write.close
exit! 1
end
end

fork、nice を使用していますが、簡単に説明すると

/usr/bin/ruby -I /usr/local/Library/Homebrew -rinstall -- /usr/local/Library/Formula/wget.rb

上記コマンド実行します。

上のスクリプトを実行する時に読み込んでいるライブラリ /usr/local/Library/Homebrew/install.rb の下のソースからプログラムが開始されます。

#--- /usr/local/iLibrary/Homebrew/installer.rb ---
at_exit do
begin
raise $! if $! # an exception was already thrown when parsing the formula
require 'extend/ENV'
require 'fileutils'
require 'hardware'
require 'keg'
require 'compatibility'
ENV.extend(HomebrewEnvExtension)
ENV.setup_build_environment
puts "-----"
puts $0
install(Formula.factory($0))
rescue Exception => e
if ENV['HOMEBREW_ERROR_PIPE']
pipe = IO.new(ENV['HOMEBREW_ERROR_PIPE'].to_i, 'w')
Marshal.dump(e, pipe)
pipe.close
exit! 1
else
onoe e
puts e.backtrace
exit! 2
end
end
end

install(Formula.factory($0)) が実行している箇所です。$0 は、/usr/local/Library/Formula/wget.rb です。

#--- /usr/local/iLibrary/Homebrew/installer.rb ---
def install f
# ・
# ・
# ・
# ・
else
f.prefix.mkpath
beginning=Time.now
f.install if not f.pourable?
FORMULA_META_FILES.each do |filename|
# ・
# ・
# ・
end

f.install の部分が wget.rb の Class Wget で install メソッドを呼び出して実際にインストールしている箇所です。

#--- /usr/local/iLibrary/Formula/wget.rb ---
def install
args = ["--disable-debug",
"--disable-dependency-tracking",
"--with-ssl=openssl",
"--prefix=#{prefix}"]
args << "--disable-iri" unless ARGV.include? "--enable-iri"
system "./configure", *args
system "make install"
end

これが、brew を使って wget をインストールするおおまかな流れです。

Cocoa Emacs も Homebrew を使ってインストールしてみました。Cocoa Emacs の Formula は、あらかじめインストールされているものより

Homebrew Emacs for OSX Lion with native full-screen ― Gist
https://gist.github.com/1109223

で公開されているものから、Forks されている最新版 Formula (emacs.rb : 入手先、インストール方法などが記載されています。)

Homebrew Emacs for OSX Lion with native full-screen ― Gist
https://gist.github.com/1128997

を利用しました。

Formula の URL は、ソースの右上 ‘raw’ 文字をクリックして確認できます。また、Option で –use-git-head を使わない場合、あらかじめ bazaar をインストールしておく必要があります。(残念ながら、git を使って最新版をダウンロードすると patch をあてることができませんでした。)

~ $ brew install bazaar
~ $ brew install https://raw.github.com/gist/1128997/347bb0d97155a294d9c50b6418c9130fadec8a5b/emacs.rb --cocoa --lion
~ $ cp -pR /usr/local/Cellar/emacs/23.3a/Emacs.app /Applications

日本語入力にも対応しています。command + space で日本語入に力切り替えることができます。.emacs ファイルに

;;------- .emacs -------
;; shell
(global-set-key [f5] 'shell)
;; IM
(setq default-input-method "MacOSX")

IM を追加してください。ついでに F5 キーでシェルを起動するようにしてみました。

Emacs を学び直してみようと思い、iTunes App Store

Learning GNU Emacs, Third Edition
カテゴリ/ブック
\600
http://itunes.apple.com/jp/app/learning-gnu-emacs-third-edition/id333188827?mt=8

を購入しました。

404 Blog Not Found:perl - O'ReillyのiPhoneアプリ本からepubをぶっこぬく
http://blog.livedoor.jp/dankogai/archives/51484907.html

上記スクリプトを使って epub に変換し、iPad で読んでいます。たった 600 円でオライリー本を入手できるので、iPad を所有している人には、ぜひお勧めです。