data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
COBOLエディタを作成してみる。
- 2013/07/28
- uchida
EclipseプラグインとしてCOBOLエディタを作成してみます。
1.org.eclipse.jdt.uiのソースをインポート
Eclipseでエディタと言えばJavaEditorですので、これを参考に作ってみます。
JavaEditorはorg.eclipse.jdt.uiの中にあるので、
ここのソースをインポートします。
(1)インポートを選択します。
(2)Plug-in DevelopmentのPlug-ins and Fragmentsを選択します。
(3)Import AsのProjects with source foldersを選択します。
(4)IDに「org.eclipse.jdt.ui」を入力し、Addします。
あとはFinishすればソースがインポートされます。
2.COBOLEditor
COBOLEditorはシンプルに以下のように書きました。
JavaEditorは色々な機能が実装されていますが、ひとまず予約語の強調表示のみということで
このようにしています。
import org.eclipse.ui.editors.text.TextEditor;
public class COBOLEditor extends TextEditor {
private COBOLColorManager colorManager;
public COBOLEditor() {
super();
colorManager = new COBOLColorManager();
setSourceViewerConfiguration(new COBOLSourceViewerConfiguration(colorManager));
}
public void dispose() {
colorManager.dispose();
super.dispose();
}
}
3.COBOLSourceViewerConfiguration
SourceViewerConfigurationは次のように実装しています。
予約語と文字列に色をつけるところまでで、コメントについては処理はしていません。
public class COBOLSourceViewerConfiguration extends TextSourceViewerConfiguration {
private COBOLCodeScanner fCodeScanner = null;
private SingleTokenCOBOLScanner fStringScanner = null;
public COBOLSourceViewerConfiguration(COBOLColorManager colorManager){
fCodeScanner = new COBOLCodeScanner(colorManager);
fStringScanner= new SingleTokenCOBOLScanner(colorManager, ICOBOLColorConstants.COBOL_STRING);
}
@Override
public IPresentationReconciler getPresentationReconciler(ISourceViewer sourceViewer) {
PresentationReconciler reconciler = new PresentationReconciler();
DefaultDamagerRepairer dr = new DefaultDamagerRepairer(fCodeScanner);
reconciler.setDamager(dr, IDocument.DEFAULT_CONTENT_TYPE);
reconciler.setRepairer(dr, IDocument.DEFAULT_CONTENT_TYPE);
dr= new DefaultDamagerRepairer(fStringScanner);
reconciler.setDamager(dr, ICOBOLColorConstants.COBOL_STRING);
reconciler.setRepairer(dr, ICOBOLColorConstants.COBOL_STRING);
return reconciler;
}
}
4.強調表示
JavaEditorは設定画面(PreferenceStore)から何色にするかを取得しています。
以下はその部分
if (fColorManager != null && colorKey != null && fColorManager.getColor(colorKey) == null) {
RGB rgb= PreferenceConverter.getColor(fPreferenceStore, colorKey);
if (fColorManager instanceof IColorManagerExtension) {
IColorManagerExtension ext= (IColorManagerExtension) fColorManager;
ext.unbindColor(colorKey);
ext.bindColor(colorKey, rgb);
}
}
今回は、設定画面はあとで作るとしてソース上に直書きします。
if (fColorManager != null && colorKey != null && fColorManager.getColor(colorKey) == null) {
RGB rgb = null;
if(colorKey.equals(ICOBOLColorConstants.COBOL_KEYWORD)){
rgb = new RGB(139, 32, 82);
}else if(colorKey.equals(ICOBOLColorConstants.COBOL_STRING)){
rgb = new RGB(63, 44, 250);
}else if(colorKey.equals(ICOBOLColorConstants.COBOL_DEFAULT)){
rgb = new RGB(0, 0, 0);
}else if(colorKey.equals(ICOBOLColorConstants.COBOL_KEYWORD_RETURN)){
rgb = new RGB(139, 32, 82);
}else if(colorKey.equals(ICOBOLColorConstants.COBOL_OPERATOR)){
rgb = new RGB(0, 0, 0);
}else if(colorKey.equals(ICOBOLColorConstants.COBOL_BRACKET)){
rgb = new RGB(0, 0, 0);
}else{
rgb = new RGB(0, 0, 0);
}
if (fColorManager instanceof IColorManagerExtension) {
IColorManagerExtension ext= (IColorManagerExtension) fColorManager;
ext.unbindColor(colorKey);
ext.bindColor(colorKey, rgb);
}
}
5.動かしてみる。
ちゃんと予約語が強調表示されました。
文字列も青色になっていてます。
次回は色を設定画面から取得するようにしてみます。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/cd5a7/cd5a70c7e444c7536b188c1044d23b31fe45b5c8" alt=""
AutoLayoutをiOS5にbackportするRRAutoLayout
- 2013/07/23
- yoshida
iOS6で導入されたAutoLayout。正しく使えば3.5inchでも4inchでも怖くない!しかし当然iOS5以前では使えません。。。2013年7月現在でも、iOS5を使用されてるユーザは全体の1割程はいるようで、切るにも切れないこともあるかと思います。
そこでRRAutoLayout、素晴らしすぎます。
セットアップにはハマりませんでしたが、思ったような表示にならなかったので色々触っていると、次のようなことに注意すれば良いことが分かりました。
Storyboard上で、例えばViewの最上部に置きたい画像がある場合、通常ImageViewをドラッグして、Viewの最上部に置くと思いますが、この際自動でConstraintsが設定されるかと思います(SuperviewとのVertical Spaceが0など)。この自動でできるConstraintsだとRRAutoLayout は認識してくれないので、あくまでも手動でのConstraintsに変換してやります。(ImageViewのHeightなども含め)
まだ多くのViewを定義してみたわけではありませんが、これで効率よく開発できそうです。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/4cffa/4cffa47b54dbaf9aa28d86e40e2fba08a68ee8a7" alt=""
CentOS6でchef-soloを9手でインストールして実行する
- 2013/07/22
- aikawa
data:image/s3,"s3://crabby-images/38d70/38d7020b146eeaca2678ba01addca6c19bee2faa" alt=""
「低コスト自動音声案内サービス loco」の開発版で、サービス内容を確認してみる!
先日、弊社で発表いたしました新サービス 「低コスト自動音声案内サービス loco(ロコ)」 の
開発版が公開されました。
なんと無料!!でサービス内容の確認が行えます。
トップページの「今すぐ登録する」をクリックし、右上の ”ユーザ登録” から簡単に 登録できますので、
ぜひ試してみて下さい。
locoホームページはこちら >>> https://lo-co.jp/
[ユーザ登録」
必要なのはメールアドレスとパスワードだけなのであっという間です(´∀`)
登録したメールアドレス宛にメールが送られてくるので、メール本文の ”ここをクリックしてください。”を
クリックし、登録を完了させてください。
登録が済んだら、さっそくログインします 。
まずは電話番号の仮登録を行います。
「電話番号仮登録」
とりあえず電話番号の説明には「会社受付」という項目で登録しました。
「その他の設定」
必要であれば、業務時間の設定を行います。
[電話番号振り分け]
電話番号振り分けを設定します。
今回は試しに、私個人の携帯電話から仮登録電話番号に電話をすると、ランカードコム本社に
転送されるよう設定してみます。
以下設定方法です
左下の”行追加”ボタンで振り分けたい電話番号を追加し、振り分けを設定します。
項目の上にカーソルを合わせると、設定項目の説明が表示されます。
項目をクリック→設定内容を入力→”レ”ボタン(決定)で項目ごとに設定を行い、
右下の登録ボタンを押して登録完了です。
[発信元電話番号]
自分の携帯電話の番号を入力します。
※ここに設定した電話番号から発信された場合の転送先や着信拒否を設定することができます。
[拒否]
電話を転送したいのでそのままにしておきます。
※”する”にチェックを入れると設定した発信元電話番号からの着信を拒否します。
[着信時メッセージ]
応答メッセージを入力します。
※入力した内容を着信時に応答メッセージとして自動で読み上げます。
[転送先]
ランカードコム本社の電話番号を入力します。
※ここに設定した電話番号へ転送します。
電話番号を複数登録し、複数人に転送することができます。
[業務時間外転送]
業務時間外を設定していないのでそのままにしておきます。
※”する”にチェックを入れると、「その他設定」で設定した業務時間外のみ電話の転送を行います。
[ラベル]
ラベルを入力します。
※録音を再生する場合に区別する為、ラベルを付けます。
特に入力しなくてもOKです。
設定の登録が完了したら、発信元電話番号で設定した番号の電話から試しに電話をかけてみます。
まだ仮登録で正式な番号は発行されていませんので、画面最下部の050から始まる仮登録電話番号に
電話をかけ、音声案内に従ってIDをプッシュします。
設定したとおりに入力した音声案内が流れ、ランカードコムに転送されます。
電話番号ごとに転送先の変更や拒否設定ができるので非常に便利です。
転送先の番号を複数設定すると、一人が電話をとれなくても設定された他の番号に自動転送されるので
電話の取りこぼしを防ぐことができます。
お客様のご要望に応じて各種オプションや専用カスタマイズも行っておりますので、お気軽にご相談下さい。
お問い合わせはこちら >>> http://www.lancard.com/contact_loco/
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Roundcubeプラグイン フックを設置する
- 2013/05/21
- uchida
Roundcubeでフックを設置する方法を紹介します。
今回はsample_aプラグインにフックを設置し、それをsample_bから利用する例です。
sample_aは画面をもったプラグインでボタンをクリックしたら、フックを実行し
元の画面を再度表示するようにしています。
ソースは以下のようになります。
sample_a.php
rc = rcmail::get_instance();
//ローカライズ対応
$this->add_texts('localization/');
//タスクとしてsample_aを設定
$this->register_task('sample_a');
//タスクバーにボタンを追加
$this->add_button(array(
'command' => 'sample_a',
'label' => 'sample_a.sample',
), 'taskbar');
//sampleタスクの場合のみ、アクションのコールバックを登録
if($this->rc->task == 'sample_a'){
$this->register_action('index', array($this, 'action'));
$this->register_action('fook_test', array($this, 'action'));
}
}
public function action(){
if($this->rc->action === 'fook_test'){
$data = $this->rc->plugins->exec_hook('sample_hook', array('data_a' => 'hoge'));
error_log(print_r($data,true));
}
$this->rc->output->send('sample_a.index');
}
}
?>
index.html
sample_b.php
add_hook('sample_hook', array($this, 'set_data'));
}
public function set_data($args){
error_log(print_r($args,true));
$data['data_b'] = "foo";
return $data;
}
}
?>
①フックを設置するコードは以下の部分になります。
$data = $this->rc->plugins->exec_hook('sample_hook', array('data_a' => 'hoge'));
引数として配列を渡しています。
②フックを利用するコードは以下の部分になります。
function init(){
$this->add_hook('sample_hook', array($this, 'set_data'));
}
public function set_data($args){
error_log(print_r($args,true));
$data['data_b'] = "foo";
return $data;
}
このような配列が渡されてきます。
(
[data_a] => hoge
[abort] =>
)
渡された引数を処理して戻り値を返したりすることが出来ます。
戻り値はこのようになります。
(
[data_b] => foo
[data_a] => hoge
[abort] =>
)
exec_hookの定義が以下のようになっているため、戻り値が配列の場合は自動的に
渡された配列に足して返されます。
public function exec_hook($hook, $args = array())
{
if (!is_array($args))
$args = array('arg' => $args);
$args += array('abort' => false);
$this->active_hook = $hook;
foreach ((array)$this->handlers[$hook] as $callback) {
$ret = call_user_func($callback, $args);
if ($ret && is_array($ret))
$args = $ret + $args;
if ($args['abort'])
break;
}
$this->active_hook = false;
return $args;
}
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Roundcubeプラグイン ユーザー情報の取得方法
- 2013/05/20
- uchida
Roundcubeでのユーザー情報の取得方法について紹介します。
①ユーザー情報
ユーザー情報の取得方法は以下のようになります。
$rc = rcmail::get_instance();
$user_data = $rc->user->data;
error_log(print_r($user_data,true));
このような情報が取得できます。
(
[user_id] => 4
[username] => foo
[mail_host] => 192.168.15.117
[alias] =>
[created] => 2013-04-11 10:14:23
[last_login] => 2013-05-17 07:45:24
[language] => ja_JP
[preferences] => a:5:{s:16:"message_sort_col";s:7:"subject";s:18:"message_sort_order";s:3:"ASC";s:8:"timezone";s:4:"auto";s:4:"skin";s:5:"larry";s:17:"message_threading";a:0:{}}
)
②識別情報
Roundcubeでは設定画面から識別情報を登録出来ます。
以下のようにして識別情報を取得出来ます。
$rc = rcmail::get_instance();
$identity = $rc->user->get_identity();
error_log(print_r($identity,true));
(
[identity_id] => 4
[user_id] => 4
[changed] => 2013-05-17 12:28:06
[del] => 0
[standard] => 1
[name] => テスト
[organization] => bar
[email] => foo@roundcube.com
[reply-to] => hoge@roundcube.com
[bcc] => test@roundcube.com
[signature] => 署名
[html_signature] => 0
)
③特定のユーザーの情報を取得する方法。
$rc = rcmail::get_instance();
$identity = $rc->user->get_identity();
上記のようにユーザー情報を取得する場合、ログインユーザーの情報を取得します。
その理由は、rcmailクラスの188行目が以下のようになっているからです。
// create user object
$this->set_user(new rcube_user($_SESSION['user_id']));
ということで、Roundcubeのuser_idがわかれば、rcube_userクラスをインスタンス化し
ユーザー情報やユーザーの識別情報を取得出来ます。
例えば、次のようになります。
$user_id = 4;
$user = new rcube_user($user_id);
$user_data = $user->data;
$user_identity = $user->get_identity();
error_log(print_r($user_data,true));
error_log(print_r($user_identity,true));
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Roundcubeプラグイン設定ファイルの利用方法
- 2013/05/17
- uchida
Roundcubeプラグイン設定ファイルの利用方法を紹介します。
ソースは以下のようになります。
sample.php
rc = rcmail::get_instance();
//設定ファイルをロード
$this->load_config('config/config.inc.php');
}
function init() {
$name = $this->rc->config->get('sample.user', '');
error_log('name:'.$name);
}
}
config.inc.php
割と簡単に使えるようです。
rc->config->get('sample.user', '');
?>
ここのconfigの実態はrcube_configオブジェクトになります。
getメソッドの第二引数は値が取得出来なかった場合のデフォルト値になります。
注意点としては
コンストラクタの中で親クラスであるrcube_pluginのコンストラクタを
必ず呼び出す必要があります。
rcube_pluginクラスのコンストラクタは以下のようになっています。
ID = get_class($this);
$this->api = $api;
$this->home = $api->dir . $this->ID;
$this->urlbase = $api->url . $this->ID . '/';
}
?>
homeディレクトリなどの値を設定しています。
そして、rcube_pluginクラスのload_configメソッドは以下のようになっています。
home.'/'.$fname;
$rcmail = rcmail::get_instance();
if (is_file($fpath) && !$rcmail->config->load_from_file($fpath)) {
raise_error(array('code' => 527, 'type' => 'php',
'file' => __FILE__, 'line' => __LINE__,
'message' => "Failed to load config from $fpath"), true, false);
return false;
}
return true;
}
?>
ここで設定ファイルの場所を取得するためにhome変数を使用しているため
rcube_pluginクラスのコンストラクタを呼び出していないと設定ファイルを取得出来なくなります。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Roundcubeプラグインからメール送信
- 2013/05/16
- uchida
Roundcubeプラグインからメール送信出来たら、色々な可能性が広がるので調べてみました。
ソースは以下のようになります。
rc = rcmail::get_instance();
//ローカライズ対応
$this->add_texts('localization/');
//タスクとしてsampleを設定
$this->register_task('sample');
//タスクバーにボタンを追加
$this->add_button(array(
'command' => 'sample',
'label' => 'sample.sample',
), 'taskbar');
//sampleタスクの場合のみ、アクションのコールバックを登録
if($this->rc->task == 'sample'){
$this->register_action('index', array($this, 'index'));
$this->register_action('send_mail', array($this, 'send_mail'));
}
}
public function index(){
$this->rc->output->send('sample.index');
}
public function send_mail(){
global $CONFIG;
$message_charset = $this->rc->output->get_charset();//rcube_html_pageオブジェクトの持っている、charsetを取得します。
$flowed = $this->rc->config->get('send_format_flowed', true);
$from = 'foo@roundcube.com';
$to = 'bar@roundcube.com';
$headers = array(
'Date' => rcmail_user_date(),
'From' => $from,
'To' => $to,
'Message-ID' => rcmail_gen_message_id(),
'User-Agent' => $CONFIG['useragent'],
'Subject' => 'テストメール',
'X-Sender' => $to,
);
$mail_mime = new Mail_mime("\r\n");
$mail_mime->headers($headers);
//$mail_mime->setTXTBody('こんにちは');
$mail_mime->setHTMLBody('こんにちは');
//ASCII文字か判定
if (preg_match('/[^\x00-\x7F]/', $mail_mime->getTXTBody())){
$transfer_encoding = $this->rc->config->get('force_7bit') ? 'quoted-printable' : '8bit';
}else{
$transfer_encoding = '7bit';
}
//文字コードなどヘッダフィールドの設定
$mail_mime->setParam('text_encoding', $transfer_encoding);
$mail_mime->setParam('html_encoding', 'quoted-printable');
$mail_mime->setParam('head_encoding', 'quoted-printable');
$mail_mime->setParam('head_charset', $message_charset);
$mail_mime->setParam('html_charset', $message_charset);
$mail_mime->setParam('text_charset', $message_charset . ($flowed ? ";\r\n format=flowed" : ''));
//ヘッダ情報の配列をテキスト形式に変換します。
$smtp_headers = $mail_mime->txtHeaders($headers, true);
if (!is_object($this->rc->smtp)){
//rcmailオブジェクトにrcube_smtpクラスのオブジェクトがない場合、生成します。
$this->rc->smtp_init(true);
}
//メールを送信します。
$sent = $this->rc->smtp->send_mail($from, $to, $smtp_headers, $mail_mime->getMessageBody());
//必要な場合以下の情報で後処理をする。
//$smtp_response = $this->rc->smtp->get_response();
//$smtp_error = $this->rc->smtp->get_error();
//元の画面を再表示
$this->index();
}
}
/roundcuberoot/program/steps/mail/sendmail.incを参考に作成しています。
この内容でほとんどのメールは送信可能かと思います。
添付ファイルをつけたメール送信や画像を含んだメールなどはさらに処理が必要になります。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Roundcubeプラグインの作り方② ~画面のあるプラグイン編~
- 2013/05/15
- uchida
今回は、Roundcubeプラグインとして、画面のあるプラグインの作り方を紹介します。
まず、初めにフォルダ構成は以下のようになります。
ソースの内容は以下のようになります。
sample.php
rc = rcmail::get_instance();
//ローカライズ対応
$this->add_texts('localization/');
//タスクとしてsampleを設定
$this->register_task('sample');
//タスクバーにボタンを追加
$this->add_button(array(
'command' => 'sample',
'label' => 'sample.sample',
), 'taskbar');
//sampleタスクの場合のみ、アクションのコールバックを登録
if($this->rc->task == 'sample'){
$this->register_action('index', array($this, 'index'));
$this->register_action('test', array($this, 'test'));
}
}
public function index(){
$this->rc->output->send('sample.index');
}
public function test(){
$this->rc->output->send('sample.test');
}
}
index.html
初期画面
test.html
テスト画面
ja_JP.inc
この状態でこのように動きます。
初期画面
data:image/s3,"s3://crabby-images/b6556/b6556b16a810504b6aee2575f052877b586b4b73" alt="img5"
テスト画面
data:image/s3,"s3://crabby-images/c8e18/c8e18a31d934cca9362201733da5afbdbe8f7954" alt="img4"
ここからはソースの説明になります。
①全てのリクエストは、プラグインクラスが受け付けます。
プラグインクラスとは、rcube_pluginを継承したクラスになります。
今回の場合、sampleクラスになります。
②タスクとアクション
Roundcubeにはタスクとアクションの概念があります。
1つのプラグインの中に1つのタスクがあり、複数のアクションがあるイメージになります。
今回の場合は、sampleタスクの中にindexアクションとtestアクションがある状態になります。
例えば、「http://roundcube.com/?_task=sample&_action=test」のようにアクセスすると
タスクはsampleで、アクションはtestという指定で動作します。
「http://roundcube.com/?_task=sample」このようなURLの場合
アクションは何もついていませんので、indexというアクションが自動的に付加されます。
③rcmailオブジェクト
$this->rc = rcmail::get_instance();
Roundcubeの様々な情報を持っているオブジェクトになります。
ここにタスクとアクションの情報も入っていますので、メンバ変数に保持します。
④コールバックの登録
あるアクションが指定されたときに、どのメソッドがコールされるかを登録します。
$this->register_action('index', array($this, 'index'));
この場合はindexというアクションが指定されたら、
$thisの中にあるindexメソッドをコールするという登録方法になります。
⑤テンプレート表示
$this->rc->output->send('sample.index');
上記のように記述することで、sampleタスク内にあるindex.htmlを表示するようにします。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/03f15/03f15fba062f1b0ddb2469cfa3c038a31df94362" alt=""
Eclipse RCPの作り方
- 2013/05/06
- uchida
Eclipse RCPの作り方を紹介します。
(といっても、久しぶりに作ろうと思ったら忘れていたので
今度は忘れないように自分のためのログですが・・。)
①Eclipse RCP
Eclipse Rich Client Platformの略で、EclipseとJavaの資産が
使えることが利点になります。
さらに、一度ソースを書くとWindows Mac Linuxなど
様々なOS用にコンパイルすることが出来ます。
②プラグインプロジェクトの作成
Eclipseには内蔵ブラウザがあるので、ブラウザを使ったものを作ってみます。
1.プラグインプロジェクトを選択します。
2.リッチクライアントアプリケーションを作成の部分は「はい」を選択して下さい。
「いいえ」を選択するとEclipseプラグインになります。
3.特に機能はいらないので「Hello RCP」を選択します。
③とりあえず、起動する。
右下にあるテストの項目の、「Eclipseアプリケーションの起動」をクリックします。
何も無い画面だけが表示されます。
④ブラウザ表示用にViewを追加します。
1.plugin.xmlを開きます。
2.拡張タブを選択します。
3.追加ボタンをクリックします。
4.「org.eclipse.ui.views」を追加します。
5.追加された「org.eclipse.ui.views」を右クリックして
以下のようにviewを選択します。
6.追加されたviewを選択して以下のように入力します。
7.以下のように「com.lancard.mybrowser.view.MyBrowserView」クラスを作成します。
package com.lancard.mybrowser.view;
import org.eclipse.swt.SWT;
import org.eclipse.swt.browser.Browser;
import org.eclipse.swt.layout.RowData;
import org.eclipse.swt.layout.RowLayout;
import org.eclipse.swt.widgets.Composite;
import org.eclipse.ui.part.ViewPart;
public class MyBrowserView extends ViewPart {
public static final String ID = "com.lancard.mybrowser.view.MyBrowserView";
private Browser browser = null;
public void createPartControl(Composite parent) {
Composite top = new Composite(parent, SWT.NONE);
RowLayout layout = new RowLayout(SWT.VERTICAL);
layout.wrap = true;
top.setLayout(layout);
browser(top);
}
private void browser(Composite top){
Composite obj = new Composite(top, SWT.NONE);
RowLayout layout = new RowLayout(SWT.HORIZONTAL);
layout.wrap = true;
obj.setLayout(layout);
browser = new Browser(obj,SWT.BORDER);
browser.setUrl("https://www.google.co.jp");
browser.setLayoutData(new RowData(800,500));
}
public void setFocus() {
}
}
⑤ブラウザを表示するコマンドを追加します。
1.plugin.xmlを開きます。
2.拡張タブを選択します。
3.追加ボタンをクリックします。
4.「org.eclipse.ui.commands」を追加します。
5.追加された「org.eclipse.ui.commands」を右クリックして
以下のようにcommandを選択します。
6.追加されたcommandを選択して以下のように入力します。
7.以下のように「com.lancard.mybrowser.action.MyBrowserAction」クラスを作成します。
package com.lancard.mybrowser.action;
import org.eclipse.jface.action.Action;
import org.eclipse.jface.dialogs.MessageDialog;
import org.eclipse.ui.IWorkbenchPage;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.PartInitException;
import com.lancard.mybrowser.ICommandIds;
public class MyBrowserAction extends Action {
private final IWorkbenchWindow window;
private int instanceNum = 0;
private final String viewId;
public MyBrowserAction(IWorkbenchWindow window, String label, String viewId) {
this.window = window;
this.viewId = viewId;
setText(label);
setId(ICommandIds.CMD_MY_BROWSER);
}
public void run() {
if(window != null) {
try {
window.getActivePage().showView(viewId, Integer.toString(instanceNum++), IWorkbenchPage.VIEW_ACTIVATE);
} catch (PartInitException e) {
MessageDialog.openError(window.getShell(), "Error", "Error opening view:" + e.getMessage());
}
}
}
}
「com.lancard.mybrowser.ICommandIds」は以下のように記述します。
package com.lancard.mybrowser;
public interface ICommandIds {
public static final String CMD_MY_BROWSER = "com.lancard.mybrowser.mybrowser";
}
⑥メニューを追加してみます。
メニューを追加するには、自動的に作成されている
「com.lancard.mybrowser.ApplicationActionBarAdvisor」クラスに以下のように記述します。
package com.lancard.mybrowser;
import org.eclipse.jface.action.IMenuManager;
import org.eclipse.jface.action.MenuManager;
import org.eclipse.ui.IWorkbenchActionConstants;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.application.ActionBarAdvisor;
import org.eclipse.ui.application.IActionBarConfigurer;
import com.lancard.mybrowser.action.MyBrowserAction;
import com.lancard.mybrowser.view.MyBrowserView;
public class ApplicationActionBarAdvisor extends ActionBarAdvisor {
private MyBrowserAction myBrowserAction;
public ApplicationActionBarAdvisor(IActionBarConfigurer configurer) {
super(configurer);
}
protected void makeActions(IWorkbenchWindow window) {
myBrowserAction = new MyBrowserAction(window, "ブラウザ", MyBrowserView.ID);
register(myBrowserAction);
}
protected void fillMenuBar(IMenuManager menuBar) {
MenuManager fileMenu = new MenuManager("&メニュー", IWorkbenchActionConstants.M_FILE);
menuBar.add(fileMenu);
fileMenu.add(myBrowserAction);
}
}
⑦構成
この時点でプロジェクトの構成はこのようになっています。
⑧起動してみる。
今度は右上にメニューが出来ていて中にはブラウザメニューがあります。
ブラウザメニューを選択すると、作成したブラウザビューが表示されます。
⑨製品化する。
このままだと、Eclipseを起動しないと使えないので製品化する作業をします。
1.MyBrowserプロジェクトを選択して、右クリックします。
2.新規メニューから製品構成を選択します。
3.以下のように入力して、完了ボタンをクリックします。
「MyBrowser.product」というファイルが追加されます。
4.「MyBrowser.product」を開きます。
5.概要タブを選択し、以下のように記述します。
「新規」ボタンをクリックします。
次のように入力して完了をクリックします。
次のようになります。
6.依存関係タブを選択し、追加ボタンをクリック。
「MyBrowser」があると思うので選択して「OK」をクリック。
「必須プラグインの追加」ボタンをクリックします。
7.起動タブを選択し、以下のように入力します。
8.概要タブのエクスポートの項目にある
「Eclipse製品エクスポート・ウィザード」をクリックします。
ディレクトリーは適当な場所にして、完了をクリックします。
9.2つのフォルダが出来ます。
10.mybrowserの中にある「MyBrowser」を起動します。
11.ちゃんと動きました。
しかし、ソース上に書いた文字は文字化けします。
⑩国際化対応
文字がソース上にそのまま書いてあるため、文字化けしている状態になります。
国際化対応するための機構があるので国際化対応します。
1.「com.lancard.mybrowser.resource」パッケージを作成します。
2.「custom_ja.properties」というファイルを作成します。
ファイルの中は次のように記述します。
MENU=&メニュー
BROWSER=ブラウザ
3.「com.lancard.mybrowser.CustomString」クラスを作成します。
package com.lancard.mybrowser;
import java.util.Locale;
import java.util.ResourceBundle;
public class CustomString {
private static final String BUNDLE_NAME = "com.lancard.mybrowser.resource.custom";
private static ResourceBundle rb = null;
public static void setBundle(Locale locale) {
try {
rb = ResourceBundle.getBundle(BUNDLE_NAME, locale);
} catch (Exception e) {
rb = ResourceBundle.getBundle(BUNDLE_NAME, Locale.JAPANESE);
}
}
public static String getString(String key) {
try {
String keyValue = new String(rb.getString(key).getBytes("ISO-8859-1"), "UTF-8");
return keyValue;
} catch (Exception e) {
e.printStackTrace();
return key;
}
}
}
4.既に存在する「com.lancard.mybrowser.Application」クラスの
startメソッドに次の一行を追加します。
「CustomString.setBundle(Locale.getDefault());」
public Object start(IApplicationContext context) throws Exception {
CustomString.setBundle(Locale.getDefault());
Display display = PlatformUI.createDisplay();
try {
int returnCode = PlatformUI.createAndRunWorkbench(display, new ApplicationWorkbenchAdvisor());
if (returnCode == PlatformUI.RETURN_RESTART)
return IApplication.EXIT_RESTART;
else
return IApplication.EXIT_OK;
} finally {
display.dispose();
}
}
5.「com.lancard.mybrowser.ApplicationActionBarAdvisor」の
文字部分を次のように書き換えます。
package com.lancard.mybrowser;
import org.eclipse.jface.action.IMenuManager;
import org.eclipse.jface.action.MenuManager;
import org.eclipse.ui.IWorkbenchActionConstants;
import org.eclipse.ui.IWorkbenchWindow;
import org.eclipse.ui.application.ActionBarAdvisor;
import org.eclipse.ui.application.IActionBarConfigurer;
import com.lancard.mybrowser.action.MyBrowserAction;
import com.lancard.mybrowser.view.MyBrowserView;
public class ApplicationActionBarAdvisor extends ActionBarAdvisor {
private MyBrowserAction myBrowserAction;
public ApplicationActionBarAdvisor(IActionBarConfigurer configurer) {
super(configurer);
}
protected void makeActions(IWorkbenchWindow window) {
myBrowserAction = new MyBrowserAction(window, CustomString.getString("BROWSER"), MyBrowserView.ID);
register(myBrowserAction);
}
protected void fillMenuBar(IMenuManager menuBar) {
MenuManager fileMenu = new MenuManager(CustomString.getString("MENU"), IWorkbenchActionConstants.M_FILE);
menuBar.add(fileMenu);
fileMenu.add(myBrowserAction);
}
}
6.もう一度エクスポートします。
今度は文字化けしていた部分もきちんと日本語が表示されました。
data:image/s3,"s3://crabby-images/bb537/bb5379ffaf5d63b2d30aa6b7597d3102d375f76c" alt="Share on Facebook Facebook"
data:image/s3,"s3://crabby-images/42b8c/42b8ccfabf0c1e57dea2662192c614aa119ed4fe" alt="Share on Twitter twitter"
data:image/s3,"s3://crabby-images/6dc33/6dc33a932e99659a9a46212b1e72a593f0454883" alt="Share on Linkedin linkedin"
data:image/s3,"s3://crabby-images/42cd1/42cd1da67c50922c2f9c0730cceb7ff5f35d70e7" alt="Share on tumblr tumblr"
data:image/s3,"s3://crabby-images/04f31/04f31512948ea960f1cd1a04c3285dbb07eb32e7" alt="Share by email mail"