単項マイナスと構文解析
単項マイナスとは 単項マイナスと括弧 括弧なし単項マイナスを許容する場合のBNF calcの場合
今回はGTK 3とGTK 4をRubyで使うライブラリについて書きたいと思います。
GTKはオープンソースのGUIライブラリです。 オリジナルはCで書かれており、Linuxで開発されました。 その後Windowsでも動くようになり、また言語もPythonやPerlなどで使えるようになりました。
GTKの現在の安定版はGTK 4(バージョン4.8.2)です。 以前の版がGTK 3でその最新版は3.24.35です。 GTK 4がリリースされて2年以上経ちますので、これから使うとしたらGTK 4になると思います。 ですが、今回はGTK 3とGTK 4の両方を試してみました。
RubyでGTKを動かすプロジジェクトはGitHub上で開発が進んでいます。
このGitHubの中でgtk3とgtk4の両方のgemが開発されています。 gemのバージョンは4.0.3となっています。 ドキュメントが少ないため、どの程度まで開発が進んでいるのかは良くわかりませんでした。
とにかく、試すしかないか、という感じです。
gemコマンドでインストールします。 gem名は「gtk3」と「gtk4」です。 どちらか一方をインストールすれば十分ですが、私は両方を試してみることにしましたので、2つともインストールします。
gtk4のインストール
$ gem install gtk4
Fetching red-colors-0.3.0.gem
... ... ...
Fetching gtk4-4.0.3.gem
... ... ...
... ... ...
13 gems installed
gtk3のインストール
$ gem install gtk3
Fetching gtk3-4.0.3.gem
... ... ...
... ... ...
2 gems installed
Ruby/GTK3とRuby/GTK4をまとめてRuby/GTKと書くことにします。 Ruby/GTKはGTKの機能をRubyで実現しようとするもので、GTKへの理解が前提となります。 そこで、GTKのAPIリファランスのリンクを示しておきます。
GTKに馴染みのない方はこのリファランスの「Additional documentation」にある「Getting started with GTK」をまず読んでください。 ここがGTKのすべての出発点になります(それも難しいかもしれませんが)。 GTK 4については、GitHubレポジトリGtk4-tutorial、 またはそのHTML版もあります。
Ruby/GTK自体のドキュメントはありますが、完全ではありません。
例えばGTKのWindowオブジェクトにはそのデフォルトサイズを指定するgtk_window_set_default_size
という関数があります。
これはRubyのインスタンス・メソッドに相当するものです。
Ruby/GTK4にはこの記載はありませんが、set_default_size
というRubyメソッドで使うことができます。
想像ですが、Ruby/GTKはこのメソッドをプログラムによる自動生成で作っているのではないでしょうか。 そうであれば、通常のメソッド定義の構文は使っていないことになります。 そして、ドキュメント自体もプログラム(おそらくRDoc)による自動生成ならば、メソッドをドキュメントに拾い出すことはできないでしょう。 この点については、ソースコードを確認できていないのであくまで推測です。
いずれにせよ、ドキュメントにないGTKのメソッドがRubyで使えるかどうかは、実際に試してみるしかありません。
手始めはいつも「Hello world」です。 このプログラムではGTK 3とGTK 4を両方試せるようになっていて、引数に3を入れるとGTK 3にそれ以外ではGTK 4を使うようになっています。
$ ruby hello.rb 3 #=> GTK 3を使う
$ ruby hello.rb #=> GTK 4を使う
先にコードを示して、その後説明します。
@gtk_version = (ARGV.size == 1 && ARGV[0] == "3") ? 3 : 4
require "gtk#{@gtk_version}"
print "Require GTK #{@gtk_version}\n"
application = Gtk::Application.new("com.github.toshiocp.hello", :default_flags)
application.signal_connect "activate" do |app|
window = Gtk::ApplicationWindow.new(app)
window.set_default_size 400,300
window.title = "Hello"
label = Gtk::Label.new("Hello World")
if @gtk_version == 4
window.child = label
window.show
else
window.add(label)
window.show_all
end
end
application.run
require "gtk3"
またはrequire "gtk4"
を実行。これによりRuby/GTKが使えるようになるGTK 3の使い方は徐々に変わってきて、gtk_application
を使うのが良い方法になってきました。
GTK 4も同じ使う方をしますので、ここではそれにならってプログラムしています。
abc@example.com
というメールアドレスを自分が持っているとする。
アプリケーションIDはcom.example.abc.hello
とすれば良い。
(注:だれでも同じ文字列を使ってアプリケーションを作れるから、このIDが世界中でユニークだということを保証することはできない。
しかしGTKのシステムがアプリケーションIDがユニークであることを前提に作られているので、アプリ制作者がIDの付け方に十分注意を払う必要がある)。
上記のhello.rb
ではGitHubのIDを用いている:default_flags
は「アプリケーションのデフォルト動作」ということ。
このフラグはGApplication ver2.74から使うようになった。
(GIO ドキュメント)
以前は:flags_none
を用いていた。
Cでプログラムする場合は「引数がファイル名である(:handles_open
)」「引数が任意の文字列である(:handles_command_line
)」などの定数もあるが、Rubyではあまり必要ないかもしれない(Rubyで引数処理できるから・・・正確ではないかも・・・):default_flags
で)アクティブになるとactivate
シグナルが発せられる。
そのシグナルを受けて動作するプログラムを「ハンドラ」という。
シグナルとハンドラをつなぎ合わせるメソッドがsignal_connect
メソッド。
そのメソッドのブロックにハンドラを記述する。window.show
を使うとウィンドウだけが表示され、ラベルが見えなくなってしまう。
window.show_all
を使うとウィンドウとその子孫ウィジェットが表示できる。
なお、window.show
に加えてlabel.show
とすればウィンドウ・ラベルともに表示できるが、show_all
を使うほうが簡単さて、「Hello world」を表示するだけにもかかわらず、説明がこんなに長くなってしまいました。 考えてみると、その多くの部分はGTKの説明です。 このことはRuby/GTKを使えるようになるには、GTKの理解が重要だということを示しています。 残念ながらGTK 4の日本語の解説資料はほとんどありません。 英語で最も頼りになるのはGTKのドキュメントです。 ですが、それも分かりやすいわけではありません。 GTKは学習コストが高いなあ、と思います。 ただ、それは非常に大きな首尾一貫したシステムだからで、GTKをマスターすればソフトウェアのスキルが格段に上がることは疑いないと思います。
簡単なGUIのプログラムである電卓を作ってみます。
lib_calc.rb
はGlimmerと同じものを使います。
calc.rb
だけをGTK対応に変更します。
まず、プログラムを示しましょう。
@gtk_version = (ARGV.size == 1 && ARGV[0] == "3") ? 3 : 4
require "gtk#{@gtk_version}"
print "Require GTK #{@gtk_version}\n"
require_relative "lib_calc.rb"
def get_answer a
if a.instance_of?(Float) && a.to_i == a
a.to_i.to_s
else
a.to_s
end
end
application = Gtk::Application.new("com.github.toshiocp.calc", :default_flags)
application.signal_connect "activate" do |app|
calc = Calc.new
window = Gtk::ApplicationWindow.new(app)
window.default_width = 400
if @gtk_version == 4
window.default_height = 120
else
window.default_height = 80
end
window.title = "Calc"
vbox = Gtk::Box.new(:vertical, 5)
window.child = vbox
hbox = Gtk::Box.new(:horizontal, 5)
label = Gtk::Label.new("")
if @gtk_version == 4
vbox.append(hbox)
vbox.append(label)
else
vbox.pack_start(hbox)
vbox.pack_start(label)
end
entry = Gtk::Entry.new
entry_buffer = entry.buffer
button_calc = Gtk::Button.new(label: "計算")
button_clear = Gtk::Button.new(label: "クリア")
button_quit = Gtk::Button.new(label: "終了")
if @gtk_version == 4
hbox.append(entry)
hbox.append(button_calc)
hbox.append(button_clear)
hbox.append(button_quit)
else
hbox.pack_start(entry)
hbox.pack_start(button_calc)
hbox.pack_start(button_clear)
hbox.pack_start(button_quit)
end
button_calc.signal_connect "clicked" do
label.text = get_answer(calc.run(entry_buffer.text))
end
button_clear.signal_connect "clicked" do
entry_buffer.text = ""
end
button_quit.signal_connect "clicked" do
if @gtk_version == 4
window.destroy
else
window.close
end
end
if @gtk_version == 4
window.show
else
window.show_all
end
end
application.run
最初のあたりはGTK 3/4両方に対応するための処理、それからget_answer
は以前と同じメソッドです。
GTKに関するプログラムはapplicationの定義以降です。
activateシグナルのハンドラがプログラムの大部分なのでそこを説明します。
:vertical
横に並べるときは:horizontal
を生成時に引数に渡す。
引数の2番めはオブジェクト間のスペースをピクセル単位で指定する。
ここでは縦に並べるボックスを、1番めがボックス、2番めをラベルにして定義。
内側のボックスは横に並べるボックスで、その中にエントリーと3つのボタンを含める。
GTK 4ではappend
メソッドを、GTK 3ではpack_start
メソッドを使ってオブジェクトをボックスに追加していくclicked
シグナルが発生する。
このシグナルに対するハンドラを定義してそれらをsignal_connect
メソッドで結びつける。
entry_buffer
の文字列を取り出し(2)calc.run
で計算し(3)get_answer
で文字列化し(4)ラベルのテキストに代入するentry_buffer
の文字列を空文字列にするdestroy
、GTK 3ではclose
メソッドを用いるプログラムが長くなった原因はウィジェットを並べるコマンドを長々と書かなければならなかったからです。 これを解決するためにGTKにはウィジェットを別ファイル(UIファイル)にXMLで書くことができます。 次のセクションではこのことについて述べます。
ウィジェットの入れ子になった構造をXMLで表したファイルをUIファイルといい、拡張子をui
にします。
これを用いると本体のRubyプログラムを簡潔にすることができます。
電卓のUIファイルは次のとおりです。
<?xml version="1.0" encoding="UTF-8"?>
<interface>
<object id="window" class="GtkWindow">
<property name="title">Calc</property>
<property name="default-width">400</property>
<property name="default-height">200</property>
<child>
<object id="vbox" class="GtkBox">
<property name="orientation">GTK_ORIENTATION_VERTICAL</property>
<child>
<object id="hbox" class="GtkBox">
<property name="orientation">GTK_ORIENTATION_HORIZONTAL</property>
<child>
<object id="entry" class="GtkEntry">
</object>
</child>
<child>
<object id="button_calc" class="GtkButton">
<property name="label">計算</property>
</object>
</child>
<child>
<object id="button_clear" class="GtkButton">
<property name="label">クリア</property>
</object>
</child>
<child>
<object id="button_quit" class="GtkButton">
<property name="label">終了</property>
</object>
</child>
</object>
</child>
<child>
<object id="label" class="GtkLabel">
</object>
</child>
</object>
</child>
</object>
</interface>
最初にXMLの定義を書き、次の行からウィジェットの定義を書きます。 一番外のタグはinterfaceです。 その中にobjectタグでウィジェットを表し、childタグでその親子関係を表します。 propertyタグではオブジェクトのプロパティを定義します。
label
プロパティがある。
GTK 4のドキュメントであれば「GtkButtonクラスのプロパティの説明」を参照するXMLファイルを見ると、ウィジェットの親子関係が反映されていることが分かると思います。
注意するのは、ボックス内に並べるウィジェットひとつひとつに<child>
タグが必要なことです。
まとめてひとつの<child>
タグにすることはできません。
UIファイルを読み込んでオブジェクトをメソッドが組み立ててくれるので、Ruby内では記述の必要なオブジェクト(たとえばシグナルを設定するオブジェクト)だけをUIファイルから取り出すだけですみます。 ボックスのようなものは取り出す必要がありません。 そのおかげでプログラムはかなりすっきりします。
@gtk_version = (ARGV.size == 1 && ARGV[0] == "3") ? 3 : 4
require "gtk#{@gtk_version}"
print "Require GTK #{@gtk_version}\n"
require_relative "lib_calc.rb"
def get_answer a
if a.instance_of?(Float) && a.to_i == a
a.to_i.to_s
else
a.to_s
end
end
application = Gtk::Application.new("com.github.toshiocp.calc", :default_flags)
application.signal_connect "activate" do |app|
calc = Calc.new
builder = Gtk::Builder.new(file: "calc.ui")
window = builder["window"]
entry = builder["entry"]
entry_buffer = entry.buffer
button_calc = builder["button_calc"]
button_clear = builder["button_clear"]
button_quit = builder["button_quit"]
label = builder["label"]
window.set_application(app)
if @gtk_version == 3
window.default_height = 80
end
button_calc.signal_connect "clicked" do
label.text = get_answer(calc.run(entry_buffer.text))
end
button_clear.signal_connect "clicked" do
entry_buffer.text = ""
end
button_quit.signal_connect "clicked" do
if @gtk_version == 4
window.destroy
else
window.close
end
end
if @gtk_version == 4
window.show
else
window.show_all
end
end
application.run
UIファイルの取り込みにはGtk::Bulderクラスを使います。
Builderクラスのオブジェクトを生成するときにUIファイル内のウィジェットも生成されます。
そこからウィジェット(オブジェクト)を取得するには[]
メソッドを使います。
ちょうどハッシュのキーを使って値を取り出すようにします。
このUIファイルではGtkWindowオブジェクトを用いています。
これは一般的なウィンドウ・オブジェクトです。
そのため、ウィンドウとアプリケーションを繋げなければなりません。
それをするのが、GtkWindowクラスのset_application
メソッドです。
UIファイルでGtkApplicationWindowを用いることもできます。
しかし、UIファイル内ではアプリケーションと結びつけることができません。
したがって、プログラム中でset_application
メソッドで結びつけることが必要になります。
それ以外はUIファイルを使わないcalcのプログラムと同じです。
UIファイルはウィジェットが多く複雑なときにとても便利です。 本体のプログラムが冗長になるのを防いでくれます。
Ruby/GTKがどこまでできるかはもっと使ってみないとわかりませんが、ここまでのところを見るとかなりGTKのつくりを反映していると思いました。 GObject IntrospectionというGTKのCライブラリなどを他の言語で使えるようにするバインディングソフトがあり、Ruby/GTKもそれを使っています。 ということは、ライブラリのRubyへの翻訳はプログラムを使って自動化されているということです。 おそらく、GTKでできることはRubyでもほとんどできるのかもしれません。
例えばGNOMEの標準エディタであるGEdit、あるいはお絵描きソフトのGimpはGTKで書かれています。 ということは相当完成度の高いアプリを書くことができるということです。 それらのソフトはCで書かれていますが、Rubyでも同程度のものの作成が期待できます。
GTKのドキュメントを見ると数多くのウィジェットが用意されています。 GTKの全体を理解するのにはとても時間がかかりますが、得られるものも大きいはずです。
あとはRuby/GTKのドキュメントの整備が課題だと思います。 GTKもドキュメントが分かりにくいと考える人が多いですが、Ruby/GTKの場合は更に深刻で、ほとんどのメソッドの記述が無い状態です。
これはマンパワーの問題が大きいのだと思います。 大企業ではプロジェクトを推進するのに十分な人材を充てることができますが、オープンソースではそうでないケースが多いです。 そうなると優れたソフトウェアほど開発に注力しなければならず、ドキュメントがなかなか充実しません。 特に、易しい記述のガイドが少なくなりがちで、ソフトウェアを使う人の裾野が広がらないのです。
プロジェクトの開発においては、ソフトウェア自体の開発以外にドキュメントの整備やソフトウェア普及の活動などトータルな計画が必要だと思います。
単項マイナスとは 単項マイナスと括弧 括弧なし単項マイナスを許容する場合のBNF calcの場合
パーサ・ジェネレータとは 少し複雑な文法 四則(加減乗除)計算のBNF Racc で実装 クラス定義、BNFの記述部分 ヘッダー、インナー、フッター コンパイルと実行 演算子の優先順位と結合における左右の優先順位 まとめ
StrScanライブラリのドキュメント 字句解析とは StrScanライブラリ StrScanライブラリを使った字句解析 実例
lbtというgemを作って公開してみた lbtはどんなgemか ファイルの配置 lbt.gemspec Rakefile gemのビルド RubyGems.orgへのアップロード 補足・・rake/gempackagetaskサブライブラリについて
文字列のエンコーディングに頭を悩ませることはほとんどなくなりました。 なぜなら、どのアプリ、システムもUTF-8を使うようになったからです。 Rubyでもエンコーディングの問題が起こることはまず無いでしょう。 ですが、今回はエンコーディングの考え方を整理してみたいと思います。
Fiberを書いたときから、次はスレッドを書こうと思っていましたが、時間がかかってしまいました。 その理由は、期待したとおりのスレッドの効果がなかったためです。 今回はそのことを書きますが、これはRubyのスレッドの抱えている問題なのか、自分のやり方が悪いのかははっきりしていません。
Fiberは「ノンプリエンプティブな軽量スレッド」とRubyのマニュアルに記載されています。
今回はRubyプログラムから自動的にドキュメントを作成するRDocについて書きたいと思います。 私はこのことについて、エキスパートではありません。 この記事も、初心者の体験談だと考えてください。
Ruby/Gtkの記事を先日書いたときに、「これはかなり使える」という手応えを感じたので、WordBook(Railsで作った単語帳プログラム)のGTK 4版を作りました。 プログラムは「徒然なるままにRuby」のGitHubレポジトリに置いてあります。 レポジトリをダウンロードし、ディレクトリ_example/...
今回はGTK 3とGTK 4をRubyで使うライブラリについて書きたいと思います。
今回もRubyとGUIのトピックです。 Glimmerを取り上げます。
Rubyはグラフィックについて弱い印象があります。 しかし、グラフィックはデバイスに関することなので、言語そのものには直接の関係はないはずで、あるとすればライブラリです。 今後グラフィック関係のgemが開発されることに期待しましょう。
Rails7におけるシステムテストについて書きます。
前回作ったWordbook(リソースフル)のテストを書いてみます。 RailsのテストはminitestをRails用に拡張したものです。
今回はRailsの慣例に沿った形でWordbookを作り直します。
今回はWordBookの検索と削除についてです。
今回はRailsにおけるデータの作成と保存、そして変更について説明します。 そのベースになるモデルとデータベースの話から始め、appendとchangeの動作について詳しく説明します。
一般に、HTMLは文書の構造を表し、CSSはその体裁(見栄え)を表します。 Railsは最終的にCSSを含むHTML文書を出力するので、この2つについての理解が必須です。 この記事ではとくにCSSの人気ライブラリであるBootstrapを紹介します。 BootstrapはJavascriptも含んでいます。
Rubyの最も人気のあるアプリケーションであるRuby on Railsを取り上げようと思い、書き始めました。 予想してはいましたが、相当な分量になってしまいました。 そのため、何回かに分けて記事にすることにします。 また、対象となる読者のレベルをどうしようかと考えましたが、「徒然Ruby」が基礎的な内容から始ま...
Rubyのライブラリ管理システムのRubygemsとコマンドgemおよびbundlerについて説明します。
minitestについて連続して2回書いてきました。 「minitestはドキュメントが少ない」という人がいますが、私も同感です。 例えば、モックとスタブの説明も少ないです。 そこで、今回はmock.rbのソースコードを参考に、モックの私的ドキュメントを書いてみました。 あくまで私個人の考えであり、minites...
今回もminitestの話です。 mockとstubに焦点をあて説明します。
アプリ作成の記事でminitestを使いました。 今回はminitestについて、また一般にテストについて、私の考えを書こうと思います。
今回はメソッドの呼び出し制限ついて説明します。 呼び出し制限にはpublic、private、protectedの3つがあります。
今回は特異メソッド、特異クラス定義、名前空間、モジュール関数について説明します。
2023/10/29 追記:この記事は新しく書き直しました。 古い記事で使っていたGitHubのCalcが大幅にアップデートされたためです。 そこで、この記事に合うようなプログラムsimple_calcを新たに作りました。 このプログラムは本レポジトリの_example/simple_calcにあります。
if〜elsif〜・・・〜else〜endは皆さん良く使うでしょうか? これは場合分けで良く使われる方法です。 これと同様の制御構造にcase文があります。 Cのswitch文に似ていますが、より強力な機能を持っています。 if-else-endよりも高い能力があるといえます。
Procオブジェクトを生成するメソッドlambdaについて説明します。
今回はブロックを一般化したオブジェクトProcを説明します。
ブロック付きメソッドの作り方を説明します。
モジュールには名前空間とミックスイン(Mix-in)の2つの機能があります。 ここではミックスインについて説明します。
クラスの親子関係
Rubyの演算子とその再定義について書きます。
今回からクラスとインスタンスを定義、生成する方法を説明します
Kernelモジュールのメソッドはどこでも使うことができます。 そのメソッドの中には便利で有用なものが多いです。
ここでは私が便利だと思ったメソッドを紹介します。
実数
今回はシンボルとハッシュについて説明します。
文字列は最も使うオブジェクトのひとつです。 特にウェブ・アプリケーションでは、コンテンツだけでなくHTMLのタグやCSSを含めすべてが文字列です。 Rubyは文字列オブジェクトのメソッドが充実しており、またパターンマッチのための正規表現も充実しています。
配列は、どのプログラミング言語にもあると思います。 複数の要素を一括して扱うことができるのが配列です。 Rubyの配列はメソッドが充実しているので、プログラムを効率的、機能的に書くのに役立ちます。
今回の目標はインスタンスです。 インスタンスを説明するために、ローカル変数と文字列オブジェクトを事前に扱います。
今回はメソッド定義です。 メソッド定義はRubyの核心ですが、今回はトップレベルに限って説明します。 この限定によって、内容はかなり易しくなっています。
ブロックはRubyの特長です。 ブロックのおかげで記述が非常にすっきりと分かりやすくなります。 今回はブロックをイテレータの本体として使う方法を説明します。
ここではRubyの最も基本的なオブジェクトである整数について説明します。
「徒然なるままに」をネットで調べてみると、「することもなく、手持無沙汰なのにまかせてという意味」とありました。 まさに、自分の現状を言い当てた言葉。 しかも、ブログに書くネタもなかなか思いつかない日々。