2011/10/28

twtr2src⇛twilog乗り換え

「twtr2src」の自動投稿機能を使って「taku1974のTweet日記」を作成していたが、自動投稿が全然配信されなくなって来たので、
に乗り換え。
これに伴い、「taku1974のTweet日記」は削除、、、twilogをそのまま使い「taku1974(@taku1974_guger)のtwilog」として公開することに。
twilogの設定は簡単で、「twilog」にアクセスしてtwitterのアカウントで新規登録・ログインするだけ。
後は、管理画面でツイートの並び順を「古→新」(朝→夜)に変更。
「taku1974のTweet日記」では、リプライを表示していなかったけど「twilog」では表示されてしまう。
「Replyを除く」をデフォルトに設定できればいいのだが、、、
とりあえず、リンクを貼るときには「Replyを除く」オプションを付けて「http://twilog.org/taku1974_guger/norep」をリンク先とすればよさそう。



2011/05/16

GmailのContentProviderから添付ファイル取得

GmailのContentProviderから添付ファイルを取得しようとしたが、イマイチうまくいかない。
とりあえず、画像の添付ファイルは取得できたのでその方法。
getAttachmentUriの第4,5引数にGmail.AttachmentRendition.SIMPLE, falseと指定すると画像の添付ファイル(jpegやgif)は取得できたが、エクセルファイルやPDFファイルは取得できない。
誰か分かる人教えてください、、、
MessageCursor mc = gmail.getMessageCursorForMessageId(account, messageId);
if(mc.next()){
    ArrayList<Attachment> attInfos = mc.getAttachmentInfos();
    if(!attInfos.isEmpty()){        // 添付があるなら
        for(Iterator<Attachment> ir = attInfos.iterator(); ir.hasNext();){
            Attachment attment = ir.next();
            Log.d(LOGTAG, "attment=" + attment);
            Uri uri = Gmail.getAttachmentUri(account, messageId, attment,Gmail.AttachmentRendition.SIMPLE, false);
            try {
                // 画像を読み込む
                InputStream in = getContentResolver().openInputStream(uri);
                BitmapFactory.Options sBitmapOptions = new BitmapFactory.Options();
                Bitmap bm = BitmapFactory.decodeStream(in, null, sBitmapOptions);
                Log.d(LOGTAG, "Bitmap size=" + bm.getWidth() + " x " + bm.getHeight());    // 画像サイズ
                int dispWidth = this.getWindowManager().getDefaultDisplay().getWidth();    // 画面サイズ(幅)
                if(bm.getWidth()> dispWidth){    // 画面幅より大きい時、縮小する
                    bm = Bitmap.createScaledBitmap(bm, dispWidth, bm.getHeight()*dispWidth/bm.getWidth(), false);
                }
                // 画像をファイルに書き込む
                String sdPash = Environment.getExternalStorageDirectory().getPath();    // SDカードのパス
                File sdDir = new File(sdPash + "/android/data/" + getPackageName() + "/cache/");
                sdDir.mkdirs();    //書き込みディレクトリ作成
                String fname = sdDir.getPath() + "/" + mc.getMessageId() + "_" + attment.name;    // 書き込みファイル名
                Log.d(LOGTAG, "fname=" + fname);
                FileOutputStream out = new FileOutputStream(fname);
                bm.compress(CompressFormat.JPEG, 100, out);
                out.flush();
                out.close();
            } catch (FileNotFoundException e) {    // InputStreamエラー処理
                Log.d(LOGTAG, "InputStream ERR! " + e.getMessage());
            } catch (IOException e) {            // FileOutputStreamエラー処理
                Log.d(LOGTAG, "FileOutputStream ERR! " + e.getMessage());
            }
        }
    }
}

2011/04/30

GmailResender Ver0.2.0 公開


・「カレンダーに登録」機能を追加しました。
 メール一覧のロングクリックで、メールの件名をタイトルに本文を内容に設定して、カレンダーアプリを起動します。
・設定画面を実装しました。
 起動時に表示するラベルを「受信トレイ」「送信済みメール」「前回終了時のラベル」「起動時に選択」から設定できます。

更新情報・アプリのダウンロードは、こちらです。

2011/04/29

Intentでカレンダーアプリを呼び出す方法

Intentで「タイトル」や「内容」、「場所」、「開始日時」、「終了日時」、「終日」を設定してカレンダーアプリを起動する方法。
ここここを参考にした。
Intent intent = new Intent(Intent.ACTION_EDIT);
intent.setType("vnd.android.cursor.item/event");
intent.putExtra("title", "!タイトル!");        // タイトル
intent.putExtra("description", "!内容!");    // 内容
intent.putExtra("eventLocation", "!場所!");  //場所
Calendar cal = Calendar.getInstance();
cal.set(2011, 4, 29, 20, 0);
intent.putExtra("beginTime", cal.getTimeInMillis());  // 開始日時
cal.set(2011, 4, 29, 23, 0);
intent.putExtra("endTime", cal.getTimeInMillis());    // 終了日時
intent.putExtra("allDay", true);                      // 終日
startActivity(intent);

2011/04/28

BloggerにFaviconを設定する方法

1.16x16サイズのアイコンを準備する。
2.Bloggerはicoファイルをアップロードできないので、googleサイトを利用して、ファイルキャビネットのページを作成し、そこにicoファイルをアップロードする。
3.アップロードしたファイルのリンクアドレスをコピーする。
4.Bloggerのダッシュボードから「デザイン」-「HTMLの編集」でテンプレートを編集
テンプレートの編集内容:
</head>タグの前に
<link href='hogehoge' rel='shortcut icon' type='image/x-icon'/>
を挿入する。hogehoge部分には、アップロードしたファイルのリンクアドレス(例:https://sites.google.com/site/hoge/favicon.ico)を書く
5.テンプレートを保存
以上。
※1</head>タグの直前に書くのが吉。なぜだか<head>タグの直後に書くとうまくいかない。
※2アップロードしたファイルのリンクアドレスの最後に付く「?attredirects=0」はあっても無くても問題なさそう。
今回設定したfaviconは、こんな感じ。
左から、BlogHome用、デジタル機器漬け用、Tweet日記用。

2011/04/23

PreferenceActivityの呼び出しでエラー

PreferenceActivityの呼び出しで上のようなエラーが出でて困ったので、調査方法をメモ。

Intent intentPref = new Intent(this,SettingActivity.class); // インテントへのインスタンス生成
startActivity(intentPref); // 設定画面(インテント)の起動


この、startActivity実行時にエラーになっていたのだが、理由が分からなかった。
分かってしまえばショウモナイ事なのだが調べる方法をメモしておく。
エラーをキャッチしてログに残せばいい」のだ。
上のコードを下のコードに変えてLogCatを見れば一発で解決!

Intent intentPref = new Intent(this,SettingActivity.class);    // インテントへのインスタンス生成
try {
    startActivity(intentPref); // 設定画面(インテント)の起動          
} catch (ActivityNotFoundException e) {
    Log.d(LOGTAG, "intentPrefErr=" + e.getMessage());
}

LogCatに「intentPrefErr=Unable to find explicit activity class {net.taku1974.GmailResender/net.taku1974.GmailResender.SettingActivity}; have you declared this activity in your AndroidManifest.xml?」ってでた。「AndroidManifest.xmlに宣言してる?」って。そう、マニフェストに宣言するのを忘れてただけってオチ、、、

2011/04/20

GoogleChromeで複数アカウントを使用する

複数のGoogleアカウント切り替える場合、いちいちログアウトして再ログインしないといけないが、GoogleChromeで複数のGoogleアカウント環境を同時に起動する方法がある。


設定方法:
1.ディスクトップにある「GoogleChrme」のショートカットをコピーする
2.2つ目のアカウント用にデータフォルダを作成しておく(今回はC:\hoge\hogehogeとする)
3.プロパティを開いて、「リンク先」の欄の後ろに_--user-data-dir="C:\hoge\hogehoge"と入力する(_は半角スペース)
4.作業フォルダも一応"C:\hoge\hogehoge"に変更して「OK」

これで、元々のショートカットからの起動したChromeとコピーしたショートカットから起動したChromeで全く別環境を構築できる。

家族で同じPCを使う場合や、一人で複数のGoogleアカウントを同時に利用したい時に便利。




結果がこんな感じ

2011/04/19

GmailResender Ver0.1.1 公開


アプリ紹介:
Android1.6~2.1の標準Gmailアプリの補助ソフトで標準Gmailアプリと併用する事を前提としてます。
このアプリで出来ること
・再送信(送信済みメールを送信前の状態にしてメーラーを起動します)
・返信(受信メールを引用無しの返信状態にしてメーラーを起動します)
・件名のコピー(件名をクリップボードにコピーします)
・本文のコピー(本文をクリップボードにコピーします)
・メールアドレスをアドレス帳の登録名で表示します
※メールの送受信は標準Gmailアプリにさせるので、このアプリにパスワードを入力する必要がありません。

使い方:
・メール一覧の「クリック」でメール内容をプレビュー表示
・メール一覧の「ロングクリック」でメール操作一覧を表示
・メニューの「ラベル選択」でメール一覧に表示するラベルを選択
・メニューの「アカウント」でメール一覧に表示するアカウントを選択
・メニューの「更新」でメール一覧を最新の状態に更新
・メニューの「設定」は、何もしません(現在未実装)
※標準Gmailアプリ側の「設定」-「ラベル」-「同期する日数」を長めに設定して、各ラベルを同期する設定にしてください。このアプリは同期されたメールしかメール一覧に表示しません。

未対応(今後対応するかもしれない内容):
・添付ファイルの表示
・絵文字を含む送信済みメールの再送信(絵文字部分が[OBJ]表示になってしまう)
・キャッシュのSDカードへの保存(現在は本体メモリに保存される)
・メールの削除、アーカイブ、既読にする、ラベル操作

動作確認機種:
・SO-01B Xperia X10 (2.1-update1)

ダウンロード&インストール方法:
下記リンク又はQRコードから、スマートフォンでアクセスしてください。
ダウンロードが完了したら、GmailResender011.apkを実行してください(ステータスバーから実行出来ると思います)
実行すると注意事項が表示され同意するとインストールが始まります。
「インストールはブロックされました」とメッセージが出る場合は、「設定」から提供元不明のアプリにチェックを入れてサードパーティアプリケーションのインストールを許可してください。
後は通常のアプリと同様にアイコンをクリックして起動してください。
GmailResender011.apk

2011/04/17

GmailResender Ver0.1.0 test公開

開発中のAndroidアプリのGmailResenderが、そこそこ完成したので野良アプリとしての公開方法をテスト。
下のリンクからダウンロード出来るかな?
アンドロイド端末から直接ダウンロード-インストール出来るかな?
これから実験してみるところ、、、
上手くできたら、インストールの仕方とかアプリの使い方とか投稿します。
とりあえず今回はテスト。。。

リンクからダウンロード
GmailResender010.apk

QRコードからダウンロード

カスタムタイトルバー

いつもお世話になっているここを参考にカスタムタイトルバーを使ってみた。
やりたかったことは、タイトルバーに右寄せで選択中のアカウントを表示する。選択中ラベルと重なる時は、選択中ラベルは省略せずにそのまま表示して、アカウントの方は最後を省略(...)で表示する。
できた結果がこれ↓
意外に苦労したのでソースを公開

custom_title.xmlを作成して、
<?xml version="1.0" encoding="utf-8"?>
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:layout_width="fill_parent"
    android:layout_height="fill_parent"
    >
    <LinearLayout
      android:orientation="horizontal"
      android:layout_width="fill_parent"
      android:layout_height="fill_parent"
     >
        <ImageView
            android:id="@+id/title_left_image"
            android:layout_width="wrap_content"
            android:layout_height="wrap_content"
         />
        <TextView 
            android:id="@+id/title_left_text"
            android:layout_width="wrap_content"
            android:layout_height="fill_parent"
            android:ellipsize="none"
            android:singleLine="true"
            style="?android:attr/windowTitleStyle"
            android:gravity="center_vertical"
         />
        <TextView 
           android:id="@+id/title_right_text"
           android:layout_width="fill_parent"
           android:layout_height="fill_parent"
           android:ellipsize="end"
           android:singleLine="true"
           android:textColor="#ffffff"
           android:textStyle="italic"
           android:gravity="center_vertical"
         />
    </LinearLayout>
</RelativeLayout>

ActivityのonCreateで、
public class GmailResenderActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
    requestWindowFeature(Window.FEATURE_CUSTOM_TITLE); // タイトルバーをカスタム宣言
    setContentView(R.layout.main);
    getWindow().setFeatureInt(Window.FEATURE_CUSTOM_TITLE, R.layout.custom_title); // カスタムタイトル登録
タイトルバーを更新したいところで、
// タイトルバー表示
ImageView iv = (ImageView) findViewById(R.id.title_left_image); // 左側イメージアイコン
iv.setImageResource(R.drawable.icon);
TextView tv;
tv = (TextView) findViewById(R.id.title_left_text); // 左側タイトル
tv.setText(selectedLabel);       // ラベル名表示
tv = (TextView) findViewById(R.id.title_right_text); // 右側タイトル
SpannableStringBuilder spannable = new SpannableStringBuilder();
spannable.append(account);
// 右揃えの SPAN インスタンスを生成。
AlignmentSpan.Standard right_span = new AlignmentSpan.Standard(Layout.Alignment.ALIGN_OPPOSITE);    
// SPAN を SpannableStringBuilder に組み込む。
spannable.setSpan(right_span, 0, spannable.length(), Spanned.SPAN_EXCLUSIVE_EXCLUSIVE);
tv.setText(spannable);
で出来上がり。

2011/04/16

LogCatが表示されない時

EclipseのLogCatが突然、
「logcat read: Invalid argument」
というエラー表示以外、表示されなくなった。
対処方法:
Android SDKをインストールしたフォルダ配下の
「platform-tools」の「adb.exe」にパラメータ「logcat -c」を付けて実行することで直るらしい。
私の環境の場合、
「スタート」-「ファイル名を指定して実行(R)」から、
D:\android\android-sdk\platform-tools\adb.exe logcat -c
として、Eclipseを再起動で直りました。

2011/04/13

ListViewにコンテキストメニュー追加とクリップボードにコピー

ここ>を参考にして、ListViewにコンテキストメニュー(ロングクリックで出てくるメニュー)を追加した。

1.ListViewにロングクリックリスナーを設定していると実行時エラーになるのでロングクリックリスナーは削除しておく。
2.コンテキストメニューで何を選択されたかは、MenuItem#getTitle()で取得。
3.ListViewのどの位置で押されたかは、MenuItem#getMenuInfo()をAdapterContextMenuInfoにキャストして、AdapterContextMenuInfo#positionで取得。
4.クリップボードへのコピーはClipboardManagerを使うだけで超簡単。

//  コンテキストメニュー表示
@Override
public void onCreateContextMenu(ContextMenu menu, View view, ContextMenuInfo menuInfo) {
    super.onCreateContextMenu(menu, view, menuInfo);
    AdapterContextMenuInfo adapterInfo = (AdapterContextMenuInfo) menuInfo;
    ListView listView = (ListView) view;
    Map map = (Map) listView.getItemAtPosition(adapterInfo.position);
    Log.d(LOGTAG, "DBG:onCreateContextMenu position=" + adapterInfo.position);
    menu.setHeaderTitle(map.get("name").toString() + map.get("subject").toString());
    menu.add(getString(R.string.ContextMenuResend));
    menu.add(getString(R.string.ContextMenuCopySubject));
    menu.add(getString(R.string.ContextMenuCopyBody));
}
// コンテキストメニュー処理部
@Override
public boolean onContextItemSelected(MenuItem item) {
    AdapterContextMenuInfo adapterInfo = (AdapterContextMenuInfo) item.getMenuInfo();
    Log.d(LOGTAG, "DBG:onContextItemSelected adapterInfo.position=" + adapterInfo.position);
    String strKindOfMenu = (String)item.getTitle();
    Log.d(LOGTAG, "DBG:onContextItemSelected getTitle=" + strKindOfMenu);
    if(strKindOfMenu.equals(getString(R.string.ContextMenuResend))){ // 再送信/返信の時
        setResendMessage(adapterInfo.position);
    }else{        // クリップボードにコピーの時
        copySubjectBody(strKindOfMenu, adapterInfo.position);
    }
        return super.onContextItemSelected(item);
}
// 件名・本文をクリップボードへコピー
protected void copySubjectBody(String strKindOfCopy, int position) {
    Map<String, Object> map = (Map<String, Object>) listview.getItemAtPosition(position);
    long messageId = (Long) map.get("messageId");
    MessageCursor mc = gmail.getMessageCursorForMessageId(account, messageId);
    if(mc.next()){
        ClipboardManager cm = (ClipboardManager)getSystemService(CLIPBOARD_SERVICE); // クリップボードマネージャ
        if(strKindOfCopy.equals(getString(R.string.ContextMenuCopySubject))){ // 件名をコピー
            cm.setText(mc.getSubject());  // クリップボードへ件名をコピー
        }else if(strKindOfCopy.equals(getString(R.string.ContextMenuCopyBody))){ // 本文をコピー
            cm.setText(Html.fromHtml(mc.getBody())); // クリップボードへ本文をコピー
        }
        Toast.makeText(this, strKindOfCopy + getString(R.string.msgFinish), Toast.LENGTH_SHORT).show(); // トーストで完了を表示
    }
}

2011/04/04

アイコン作成

ここを参考に、現在作製中のアプリ「GmailResender」用のアイコンを作ってみた。
Gmailアイコンを元に変形して、「Re」を追加した。こんな感じになった。
今回使用したアプリは「GIMP2」フリーでここまで出来るとは、、、すばらしい。

2011/04/01

アプリにメニューを追加する


ここここを参考に追加してみたが、メニューアイコンでつまずいたのでメモっておく。
設定アイコン(ic_menu_preferences)の方は、下図の①のようにmenu.xmlに記述すれば上手く表示されたのだがアーカイブアイコン(ic_menu_archive)の方が上手くいかない。
「android:icon="@android:drawable/ic_menu_archive"」と記述すれば、「エラー: Resource is not public.」と怒られる。
「android:icon="@*android:drawable/ic_menu_archive"」と記述すれば、エラーは無くなり実行できるが何故かではなくが表示された、、、意味不明!

仕方ないので下図の②のように記述して、③に「ic_menu_archive.png」をコピーすれば上手くいった。

2011/03/22

端末に設定されたGmailAccountを取得

以下のコードで端末に設定されているGmailAccountを取得できた。
※マニフェストには、
<uses-permission android:name="android.permission.GET_ACCOUNTS" />
を追加する。

// 端末に設定されているGmailAccountの取得
private String[] getGmailAccounts() {
    ArrayList<String> accountsInfo = new ArrayList<String>();
    Account[] accounts = AccountManager.get(this).getAccounts();
    for (Account account : accounts) {
        String name = account.name;
        String type = account.type;
        Log.d(LOGTAG, "DBG:" + account.toString());
        if(type.equals("com.google")){   // googleアカウントタイプなら
          accountsInfo.add(name);
        }
    }
    String[] result = new String[accountsInfo.size()];
    accountsInfo.toArray(result);
    return result;
}

2011/03/20

Gmailの送信済みメール取得とメーラ一覧の起動

Gmailの送信済みメールを再送信するAndroidアプリの作成を始めて約10日。
やりたい事が出来る目処がついたので、ここまでの道程とソースを公開。
アプリとしては、
・Android標準のGmailにContentProviderで繋げて、データを取得。
・Intentでメーラを起動して、取得したデータを渡す。
これだけ。


まず、「Gmail.java」と関連する「Lists.java」、「Maps.java」、「Sets.java」、「Regex.java」、「BaseColumns.java」、「Uri.java」をここからダウンロードしてアプリケーションのプロジェクトにインポート。
※このファイルたちを探すのに苦労した、、、最新のAndroidソースには含まれていないようだ。

次に、ここを参考にすれば簡単に受信メールを取得できるようになった。(実機でないと動作しないので注意)
だが、送信済みメールを取得する方法が分からず途方にくれる、、、gmail.javaのコード読んでqueryに「label:<label>」をセットすればラベルのついたメールを抽出できそうな事に気付く。

Intentでメーラを起動の方は、ここを参考にすれば直ぐ出来た。

で、出来たコードが下記。

今後の課題、
・Gmailアカウントを設定出来るようにする。
・見た目の改善。
・UIの改善

ソース
package net.taku1974.GmailResender;

import android.app.Activity;
import android.app.AlertDialog;
import android.os.Bundle;
import android.widget.ArrayAdapter;
import android.widget.AdapterView;
import android.text.Html;
import android.util.Log;
import android.view.View;
import android.widget.ListView;
import android.widget.TextView;
import android.content.ActivityNotFoundException;
import android.content.ContentResolver;
import android.content.DialogInterface;
import android.content.Intent;
import android.database.ContentObserver;
import android.os.Handler;
import android.provider.Gmail;
import android.provider.Gmail.BecomeActiveNetworkCursor;
import android.provider.Gmail.ConversationCursor;
import android.provider.Gmail.LabelMap;
import android.provider.Gmail.MessageCursor;


public class GmailResenderActivity extends Activity {
 
 private static String account = "hoge.android.test@gmail.com"; // GMailアカウント
 private static String LOGTAG = "GmailResenderLog";

    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        // Gmailからメッセージ取得
        ContentResolver contentResolver = this.getContentResolver();
        Gmail gmail = new Gmail(contentResolver); // Gmailクラス生成
        ContentObserver gmailObserver = new ContentObserver(new Handler()){}; // Gmail用コンテンツオブザーバ生成
        contentResolver.registerContentObserver(Gmail.BASE_URI, true, gmailObserver); // Gmail用コンテンツオブザーバを登録
     LabelMap lm = gmail.getLabelMap(account);  // ラベルマップ取得
     long labelId=lm.getLabelIdSent();    // 送信済みラベルID
     String query = "label:"+lm.getName(labelId); // ラベル:送信済みを抽出
        Log.d(LOGTAG, "DBG:" + query);
        ConversationCursor cc =  gmail.getConversationCursorForQuery(account, query, BecomeActiveNetworkCursor.YES); // 送信済みラベルを含む会話を取得
        ArrayAdapter adapterM = new ArrayAdapter(this,android.R.layout.simple_list_item_1); // リストビュー用アダプタ
        for (int i=0; i<10; i++){ // とりあえず10件取得
         if (cc.next()){
          MessageCursor mc = gmail.getMessageCursorForConversationId(account, cc.getConversationId());
          while (mc.next()) {
           if(mc.getLabelIds().contains(labelId)){  // 送信済みラベルIDを含むなら
                  Log.d(LOGTAG, "DBG:"+ mc.getSubject());
             adapterM.add(new GmailMessage(mc.getToAddresses() , mc.getSubject(), mc.getBody())); // アダプタに追加
           }
          }
         }else{
          break;
         }
     }
        // メッセージ一覧
        final ListView listview = (ListView) findViewById(R.id.listViewMail); // リストビュー生成
        listview.setAdapter(adapterM);  // アダブターセット
        listview.setOnItemClickListener( // クリックリスナー登録
         new AdapterView.OnItemClickListener() {
          public void onItemClick(AdapterView parent,View view, int position, long id) {
           setSelectedMessage(parent, position);
           }
         }
        );
        listview.setOnItemLongClickListener( // ロングクリックリスナー登録
         new AdapterView.OnItemLongClickListener() {
             public boolean onItemLongClick(AdapterView parent,View view, int position, long id) {
              setResendMessage(parent, position);
              return false;
             }   
   }
        );
    }
    // メッセージ一覧からクリックしたときの処理
 protected void setSelectedMessage(AdapterView parent, int position) {
  GmailMessage gmess = (GmailMessage) parent.getAdapter().getItem(position);
  final TextView selecteditem = (TextView) findViewById(R.id.textViewBody);
  CharSequence csBody = Html.fromHtml(gmess.getBody()); // メール本文(HTML)をCharSequenceに変換
  selecteditem.setText("Body:" + "\n" + csBody);
 }
 
 // メッセージ一覧からロングクリックしたときの処理
 protected void setResendMessage(AdapterView parent, int position) {
  GmailMessage gmess = (GmailMessage) parent.getAdapter().getItem(position);
  // メーラー起動インテントの作成
  Intent mailIntent = new Intent(Intent.ACTION_SEND);
  mailIntent.putExtra(Intent.EXTRA_EMAIL, gmess.getTo()); // 宛先(To)
  // mailIntent.putExtra(Intent.EXTRA_CC, new String[] {"bbb@xxxx.co.jp", "bbb@xxxx.co.jp"});
  mailIntent.putExtra(Intent.EXTRA_SUBJECT, gmess.getSubject()); // 件名
  mailIntent.putExtra(Intent.EXTRA_TEXT, Html.fromHtml(gmess.getBody())); // 本文
  // mailIntent.setType("image/png"); // 添付ファイル
  // Uri attachments = Uri.parse("file://" + selectedImageFileName); //<-- 画像へのパス
  // mailIntent.putExtra(Intent.EXTRA_STREAM, attachments);
  mailIntent.setType("message/rfc822"); // 送信データタイプ
  try {
   // システムにインストールされているメーラーを起動
   startActivity(Intent.createChooser(mailIntent, getString(R.string.app_name)));
  } catch (ActivityNotFoundException e ) {
    Log.e(LOGTAG, e.getMessage());
    String errorMsg = String.format("%s\n(%s)", getString(R.string.app_name), e.getMessage());
    // アラートダイアログ 
    AlertDialog.Builder errorDialog = new AlertDialog.Builder(this);
    errorDialog.setTitle("GmailResenderErr");
    errorDialog.setMessage(errorMsg);
    errorDialog.setPositiveButton(R.string.button_close, new DialogInterface.OnClickListener() {
     public void onClick(DialogInterface dialog, int which) {
      // ダイアログを閉じるだけ
     }
    });
    errorDialog.show();
  }

 }


 private class GmailMessage {
  String[] strTo;
  String strSubject;
  String strBody;
  GmailMessage(String[] To, String Subject, String Body) {
   strTo = To;
   strSubject = Subject;
   strBody = Body;
  }
  public String toString() {
   return "TO:" + strTo[0] + "\n" + "Subject:" + strSubject;
  }
  public String[] getTo() {
   return strTo;
  }
  public String getSubject() {
   return strSubject;
  }
  public String getBody() {
   return strBody;
  }
 }
}
マニフェスト
<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
      package="net.taku1974.GmailResender"
      android:versionCode="1"
      android:versionName="1.0">


    <application android:icon="@drawable/icon" android:label="@string/app_name" android:debuggable="true">
        <activity android:name=".GmailResenderActivity"
                  android:label="@string/app_name">
            <intent-filter>
                <action android:name="android.intent.action.MAIN" />
                <category android:name="android.intent.category.LAUNCHER" />
            </intent-filter>
        </activity>

    </application>
    <!-- GmailProviderアクセス -->
    <uses-permission android:name="com.google.android.providers.gmail.permission.READ_GMAIL" />
</manifest>

2011/03/13

FaziBear's Blogger Widgets

プログラムのソースコードをBloggerにアップする時、ハイライト表示してくれるツール。
http://www.kuribo.info/2008/06/blogger-syntax-highlighter.html」を参考に
FaziBear's Blogger Widgetsへアクセスしたが、google chromeでは、blogger.htmlをダウンロードしてしまう。また、ダウンロードしたファイルを開いても「Add to Blogger」ボタンが表示されずにコードそのものが表示されてしまっている。
仕方ないので、chromeの拡張機能のIE Tabを使って開き直すと「Add to Blogger」ボタンが表示され実行出来る。
但し、今度はBlogger側で「リクエストを処理できませんでした」と言うエラーがでる、、、何度かチャレンジすると上手くいったが、謎だ。。。

使い方は、HTML編集で
<pre class="java" name="code">・・・コード・・・</pre>
と書くだけ。結果がこれ↓

class hoge : public hoge{
pulic:
// コンストラクタ
hoge();
// デストラクタ
~hoge();
// 初期化
virtual const bool init() = 0;
};

※結局、BloggerのデザインにHTML/JavaScriptガジェットが追加されるだけなので、手動でガジェットを追加して中身に以下のコードを書けば使えると思われる。

<style>
.dp-highlighter { font-family: "Consolas", "Courier New", Courier, mono, serif; font-size: 12px; background-color: #E7E5DC; width: 99%; overflow: auto; margin: 18px 0 18px 0 !important; padding: 1px 0 0 0 !important; }
.dp-highlighter ol, .dp-highlighter ol li, .dp-highlighter ol li span { margin: 0; padding: 0; border: none; }
.dp-highlighter a, .dp-highlighter a:hover { background: none; border: none; padding: 0; margin: 0; }
.dp-highlighter .bar { padding: 0 0 0 45px; }
.dp-highlighter.collapsed .bar, .dp-highlighter.nogutter .bar { padding-left: 0px; }
.dp-highlighter ol { list-style: decimal; background-color: #fff; margin: 0px 0px 1px 45px !important; padding: 0px;color: #5C5C5C; }
.dp-highlighter.nogutter ol, .dp-highlighter.nogutter ol li { list-style: none !important; margin-left: 0px !important; }
.dp-highlighter ol li, .dp-highlighter .columns div { background: none; list-style: decimal-leading-zero; list-style-position: outside !important; border-left: 3px solid #6CE26C; background-color: #F8F8F8; color: #5C5C5C; padding: 0 3px 0 10px !important; margin: 0 !important; line-height: 14px; }
.dp-highlighter.nogutter ol li, .dp-highlighter.nogutter .columns div { border: 0; }
.dp-highlighter .columns { background-color: #F8F8F8; color: gray; overflow: hidden; width: 100%; }
.dp-highlighter .columns div { padding-bottom: 5px; }
.dp-highlighter ol li.alt { background-color: #FFF; color: inherit; }
.dp-highlighter ol li span { color: black; background-color: inherit; }
.dp-highlighter.collapsed ol { margin: 0px; }
.dp-highlighter.collapsed ol li { display: none; }
.dp-highlighter.printing { border: none; }
.dp-highlighter.printing .tools { display: none !important; }
.dp-highlighter.printing li { display: list-item !important; }
.dp-highlighter .tools { padding: 3px 8px 3px 10px; font: 9px Verdana, Geneva, Arial, Helvetica, sans-serif; color: silver; background-color: #f8f8f8; padding-bottom: 10px; border-left: 3px solid #6CE26C; }
.dp-highlighter.nogutter .tools { border-left: 0; }
.dp-highlighter.collapsed .tools { border-bottom: 0; }
.dp-highlighter .tools a { font-size: 9px; color: #a0a0a0; background-color: inherit; text-decoration: none; margin-right: 10px; }
.dp-highlighter .tools a:hover { color: red; background-color: inherit; text-decoration: underline; }
.dp-about { background-color: #fff; color: #333; margin: 0px; padding: 0px; }
.dp-about table { width: 100%; height: 100%; font-size: 11px; font-family: Tahoma, Verdana, Arial, sans-serif !important; }
.dp-about td { padding: 10px; vertical-align: top; }
.dp-about .copy { border-bottom: 1px solid #ACA899; height: 95%; }
.dp-about .title { color: red; background-color: inherit; font-weight: bold; }
.dp-about .para { margin: 0 0 4px 0; }
.dp-about .footer { background-color: #ECEADB; color: #333; border-top: 1px solid #fff; text-align: right; }
.dp-about .close { font-size: 11px; font-family: Tahoma, Verdana, Arial, sans-serif !important; background-color: #ECEADB; color: #333; width: 60px; height: 22px; }
.dp-highlighter .comment, .dp-highlighter .comments { color: #008200; background-color: inherit; }
.dp-highlighter .string { color: blue; background-color: inherit; }
.dp-highlighter .keyword { color: #069; font-weight: bold; background-color: inherit; }
.dp-highlighter .preprocessor { color: gray; background-color: inherit; }
</style>
<script type="text/javascript"><!--
// 1.5.1
var dp={sh:{Toolbar:{},Utils:{},RegexLib:{},Brushes:{},Strings:{AboutDialog:'<html><head><title>About...</title></head><body class="dp-about"><table cellspacing="0"><tr><td class="copy"><p class="title">dp.SyntaxHighlighter</div><div class="para">Version: {V}</p><p><a href="http://www.dreamprojections.com/syntaxhighlighter/?ref=about" target="_blank">http://www.dreamprojections.com/syntaxhighlighter</a></p>&copy;2004-2007 Alex Gorbatchev.</td></tr><tr><td class="footer"><input type="button" class="close" value="OK" onClick="window.close()"/></td></tr></table></body></html>'},ClipboardSwf:null,Version:'1.5.1'}};dp.SyntaxHighlighter=dp.sh;dp.sh.Toolbar.Commands={ExpandSource:{label:'+ expand source',check:function(highlighter){return highlighter.collapse;},func:function(sender,highlighter)
{sender.parentNode.removeChild(sender);highlighter.div.className=highlighter.div.className.replace('collapsed','');}},ViewSource:{label:'view plain',func:function(sender,highlighter)
{var code=dp.sh.Utils.FixForBlogger(highlighter.originalCode).replace(/</g,'&lt;');var wnd=window.open('','_blank','width=750, height=400, location=0, resizable=1, menubar=0, scrollbars=0');wnd.document.write('<textarea style="width:99%;height:99%">'+code+'</textarea>');wnd.document.close();}},CopyToClipboard:{label:'copy to clipboard',check:function(){return window.clipboardData!=null||dp.sh.ClipboardSwf!=null;},func:function(sender,highlighter)
{var code=dp.sh.Utils.FixForBlogger(highlighter.originalCode).replace(/&lt;/g,'<').replace(/&gt;/g,'>').replace(/&amp;/g,'&');if(window.clipboardData)
{window.clipboardData.setData('text',code);}
else if(dp.sh.ClipboardSwf!=null)
{var flashcopier=highlighter.flashCopier;if(flashcopier==null)
{flashcopier=document.createElement('div');highlighter.flashCopier=flashcopier;highlighter.div.appendChild(flashcopier);}
flashcopier.innerHTML='<embed src="'+dp.sh.ClipboardSwf+'" FlashVars="clipboard='+encodeURIComponent(code)+'" width="0" height="0" type="application/x-shockwave-flash"></embed>';}
alert('The code is in your clipboard now');}},PrintSource:{label:'print',func:function(sender,highlighter)
{var iframe=document.createElement('IFRAME');var doc=null;iframe.style.cssText='position:absolute;width:0px;height:0px;left:-500px;top:-500px;';document.body.appendChild(iframe);doc=iframe.contentWindow.document;dp.sh.Utils.CopyStyles(doc,window.document);doc.write('<div class="'+highlighter.div.className.replace('collapsed','')+' printing">'+highlighter.div.innerHTML+'</div>');doc.close();iframe.contentWindow.focus();iframe.contentWindow.print();alert('Printing...');document.body.removeChild(iframe);}},About:{label:'?',func:function(highlighter)
{var wnd=window.open('','_blank','dialog,width=300,height=150,scrollbars=0');var doc=wnd.document;dp.sh.Utils.CopyStyles(doc,window.document);doc.write(dp.sh.Strings.AboutDialog.replace('{V}',dp.sh.Version));doc.close();wnd.focus();}}};dp.sh.Toolbar.Create=function(highlighter)
{var div=document.createElement('DIV');div.className='tools';for(var name in dp.sh.Toolbar.Commands)
{var cmd=dp.sh.Toolbar.Commands[name];if(cmd.check!=null&&!cmd.check(highlighter))
continue;div.innerHTML+='<a href="#" onclick="dp.sh.Toolbar.Command(\''+name+'\',this);return false;">'+cmd.label+'</a>';}
return div;}
dp.sh.Toolbar.Command=function(name,sender)
{var n=sender;while(n!=null&&n.className.indexOf('dp-highlighter')==-1)
n=n.parentNode;if(n!=null)
dp.sh.Toolbar.Commands[name].func(sender,n.highlighter);}
dp.sh.Utils.CopyStyles=function(destDoc,sourceDoc)
{var links=sourceDoc.getElementsByTagName('link');for(var i=0;i<links.length;i++)
if(links[i].rel.toLowerCase()=='stylesheet')
destDoc.write('<link type="text/css" rel="stylesheet" href="'+links[i].href+'"></link>');}
dp.sh.Utils.FixForBlogger=function(str)
{return(dp.sh.isBloggerMode==true)?str.replace(/<br\s*\/?>|&lt;br\s*\/?&gt;/gi,'\n'):str;}
dp.sh.RegexLib={MultiLineCComments:new RegExp('/\\*[\\s\\S]*?\\*/','gm'),SingleLineCComments:new RegExp('//.*$','gm'),SingleLinePerlComments:new RegExp('#.*$','gm'),DoubleQuotedString:new RegExp('"(?:\\.|(\\\\\\")|[^\\""\\n])*"','g'),SingleQuotedString:new RegExp("'(?:\\.|(\\\\\\')|[^\\''\\n])*'",'g')};dp.sh.Match=function(value,index,css)
{this.value=value;this.index=index;this.length=value.length;this.css=css;}
dp.sh.Highlighter=function()
{this.noGutter=false;this.addControls=true;this.collapse=false;this.tabsToSpaces=true;this.wrapColumn=80;this.showColumns=true;}
dp.sh.Highlighter.SortCallback=function(m1,m2)
{if(m1.index<m2.index)
return-1;else if(m1.index>m2.index)
return 1;else
{if(m1.length<m2.length)
return-1;else if(m1.length>m2.length)
return 1;}
return 0;}
dp.sh.Highlighter.prototype.CreateElement=function(name)
{var result=document.createElement(name);result.highlighter=this;return result;}
dp.sh.Highlighter.prototype.GetMatches=function(regex,css)
{var index=0;var match=null;while((match=regex.exec(this.code))!=null)
this.matches[this.matches.length]=new dp.sh.Match(match[0],match.index,css);}
dp.sh.Highlighter.prototype.AddBit=function(str,css)
{if(str==null||str.length==0)
return;var span=this.CreateElement('SPAN');str=str.replace(/ /g,'&nbsp;');str=str.replace(/</g,'&lt;');str=str.replace(/\n/gm,'&nbsp;<br>');if(css!=null)
{if((/br/gi).test(str))
{var lines=str.split('&nbsp;<br>');for(var i=0;i<lines.length;i++)
{span=this.CreateElement('SPAN');span.className=css;span.innerHTML=lines[i];this.div.appendChild(span);if(i+1<lines.length)
this.div.appendChild(this.CreateElement('BR'));}}
else
{span.className=css;span.innerHTML=str;this.div.appendChild(span);}}
else
{span.innerHTML=str;this.div.appendChild(span);}}
dp.sh.Highlighter.prototype.IsInside=function(match)
{if(match==null||match.length==0)
return false;for(var i=0;i<this.matches.length;i++)
{var c=this.matches[i];if(c==null)
continue;if((match.index>c.index)&&(match.index<c.index+c.length))
return true;}
return false;}
dp.sh.Highlighter.prototype.ProcessRegexList=function()
{for(var i=0;i<this.regexList.length;i++)
this.GetMatches(this.regexList[i].regex,this.regexList[i].css);}
dp.sh.Highlighter.prototype.ProcessSmartTabs=function(code)
{var lines=code.split('\n');var result='';var tabSize=4;var tab='\t';function InsertSpaces(line,pos,count)
{var left=line.substr(0,pos);var right=line.substr(pos+1,line.length);var spaces='';for(var i=0;i<count;i++)
spaces+=' ';return left+spaces+right;}
function ProcessLine(line,tabSize)
{if(line.indexOf(tab)==-1)
return line;var pos=0;while((pos=line.indexOf(tab))!=-1)
{var spaces=tabSize-pos%tabSize;line=InsertSpaces(line,pos,spaces);}
return line;}
for(var i=0;i<lines.length;i++)
result+=ProcessLine(lines[i],tabSize)+'\n';return result;}
dp.sh.Highlighter.prototype.SwitchToList=function()
{var html=this.div.innerHTML.replace(/<(br)\/?>/gi,'\n');var lines=html.split('\n');if(this.addControls==true)
this.bar.appendChild(dp.sh.Toolbar.Create(this));if(this.showColumns)
{var div=this.CreateElement('div');var columns=this.CreateElement('div');var showEvery=10;var i=1;while(i<=150)
{if(i%showEvery==0)
{div.innerHTML+=i;i+=(i+'').length;}
else
{div.innerHTML+='&middot;';i++;}}
columns.className='columns';columns.appendChild(div);this.bar.appendChild(columns);}
for(var i=0,lineIndex=this.firstLine;i<lines.length-1;i++,lineIndex++)
{var li=this.CreateElement('LI');var span=this.CreateElement('SPAN');li.className=(i%2==0)?'alt':'';span.innerHTML=lines[i]+'&nbsp;';li.appendChild(span);this.ol.appendChild(li);}
this.div.innerHTML='';}
dp.sh.Highlighter.prototype.Highlight=function(code)
{function Trim(str)
{return str.replace(/^\s*(.*?)[\s\n]*$/g,'$1');}
function Chop(str)
{return str.replace(/\n*$/,'').replace(/^\n*/,'');}
function Unindent(str)
{var lines=dp.sh.Utils.FixForBlogger(str).split('\n');var indents=new Array();var regex=new RegExp('^\\s*','g');var min=1000;for(var i=0;i<lines.length&&min>0;i++)
{if(Trim(lines[i]).length==0)
continue;var matches=regex.exec(lines[i]);if(matches!=null&&matches.length>0)
min=Math.min(matches[0].length,min);}
if(min>0)
for(var i=0;i<lines.length;i++)
lines[i]=lines[i].substr(min);return lines.join('\n');}
function Copy(string,pos1,pos2)
{return string.substr(pos1,pos2-pos1);}
var pos=0;if(code==null)
code='';this.originalCode=code;this.code=Chop(Unindent(code));this.div=this.CreateElement('DIV');this.bar=this.CreateElement('DIV');this.ol=this.CreateElement('OL');this.matches=new Array();this.div.className='dp-highlighter';this.div.highlighter=this;this.bar.className='bar';this.ol.start=this.firstLine;if(this.CssClass!=null)
this.ol.className=this.CssClass;if(this.collapse)
this.div.className+=' collapsed';if(this.noGutter)
this.div.className+=' nogutter';if(this.tabsToSpaces==true)
this.code=this.ProcessSmartTabs(this.code);this.ProcessRegexList();if(this.matches.length==0)
{this.AddBit(this.code,null);this.SwitchToList();this.div.appendChild(this.bar);this.div.appendChild(this.ol);return;}
this.matches=this.matches.sort(dp.sh.Highlighter.SortCallback);for(var i=0;i<this.matches.length;i++)
if(this.IsInside(this.matches[i]))
this.matches[i]=null;for(var i=0;i<this.matches.length;i++)
{var match=this.matches[i];if(match==null||match.length==0)
continue;this.AddBit(Copy(this.code,pos,match.index),null);this.AddBit(match.value,match.css);pos=match.index+match.length;}
this.AddBit(this.code.substr(pos),null);this.SwitchToList();this.div.appendChild(this.bar);this.div.appendChild(this.ol);}
dp.sh.Highlighter.prototype.GetKeywords=function(str)
{return'\\b'+str.replace(/ /g,'\\b|\\b')+'\\b';}
dp.sh.BloggerMode=function()
{dp.sh.isBloggerMode=true;}
dp.sh.HighlightAll=function(name,showGutter,showControls,collapseAll,firstLine,showColumns)
{function FindValue()
{var a=arguments;for(var i=0;i<a.length;i++)
{if(a[i]==null)
continue;if(typeof(a[i])=='string'&&a[i]!='')
return a[i]+'';if(typeof(a[i])=='object'&&a[i].value!='')
return a[i].value+'';}
return null;}
function IsOptionSet(value,list)
{for(var i=0;i<list.length;i++)
if(list[i]==value)
return true;return false;}
function GetOptionValue(name,list,defaultValue)
{var regex=new RegExp('^'+name+'\\[(\\w+)\\]$','gi');var matches=null;for(var i=0;i<list.length;i++)
if((matches=regex.exec(list[i]))!=null)
return matches[1];return defaultValue;}
function FindTagsByName(list,name,tagName)
{var tags=document.getElementsByTagName(tagName);for(var i=0;i<tags.length;i++)
if(tags[i].getAttribute('name')==name)
list.push(tags[i]);}
var elements=[];var highlighter=null;var registered={};var propertyName='innerHTML';FindTagsByName(elements,name,'pre');FindTagsByName(elements,name,'textarea');if(elements.length==0)
return;for(var brush in dp.sh.Brushes)
{var aliases=dp.sh.Brushes[brush].Aliases;if(aliases==null)
continue;for(var i=0;i<aliases.length;i++)
registered[aliases[i]]=brush;}
for(var i=0;i<elements.length;i++)
{var element=elements[i];var options=FindValue(element.attributes['class'],element.className,element.attributes['language'],element.language);var language='';if(options==null)
continue;options=options.split(':');language=options[0].toLowerCase();if(registered[language]==null)
continue;highlighter=new dp.sh.Brushes[registered[language]]();element.style.display='none';highlighter.noGutter=(showGutter==null)?IsOptionSet('nogutter',options):!showGutter;highlighter.addControls=(showControls==null)?!IsOptionSet('nocontrols',options):showControls;highlighter.collapse=(collapseAll==null)?IsOptionSet('collapse',options):collapseAll;highlighter.showColumns=(showColumns==null)?IsOptionSet('showcolumns',options):showColumns;var headNode=document.getElementsByTagName('head')[0];if(highlighter.Style&&headNode)
{var styleNode=document.createElement('style');styleNode.setAttribute('type','text/css');if(styleNode.styleSheet)
{styleNode.styleSheet.cssText=highlighter.Style;}
else
{var textNode=document.createTextNode(highlighter.Style);styleNode.appendChild(textNode);}
headNode.appendChild(styleNode);}
highlighter.firstLine=(firstLine==null)?parseInt(GetOptionValue('firstline',options,1)):firstLine;highlighter.Highlight(element[propertyName]);highlighter.source=element;element.parentNode.insertBefore(highlighter.div,element);}}
dp.sh.Brushes.CSharp=function()
{var keywords='abstract as base bool break byte case catch char checked class const '+'continue decimal default delegate do double else enum event explicit '+'extern false finally fixed float for foreach get goto if implicit in int '+'interface internal is lock long namespace new null object operator out '+'override params private protected public readonly ref return sbyte sealed set '+'short sizeof stackalloc static string struct switch this throw true try '+'typeof uint ulong unchecked unsafe ushort using virtual void while';this.regexList=[{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('^\\s*#.*','gm'),css:'preprocessor'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-c';this.Style='.dp-c .vars { color: #d00; }';}
dp.sh.Brushes.CSharp.prototype=new dp.sh.Highlighter();dp.sh.Brushes.CSharp.Aliases=['c#','c-sharp','csharp'];
dp.sh.Brushes.Vb=function()
{var keywords='AddHandler AddressOf AndAlso Alias And Ansi As Assembly Auto '+'Boolean ByRef Byte ByVal Call Case Catch CBool CByte CChar CDate '+'CDec CDbl Char CInt Class CLng CObj Const CShort CSng CStr CType '+'Date Decimal Declare Default Delegate Dim DirectCast Do Double Each '+'Else ElseIf End Enum Erase Error Event Exit False Finally For Friend '+'Function Get GetType GoSub GoTo Handles If Implements Imports In '+'Inherits Integer Interface Is Let Lib Like Long Loop Me Mod Module '+'MustInherit MustOverride MyBase MyClass Namespace New Next Not Nothing '+'NotInheritable NotOverridable Object On Option Optional Or OrElse '+'Overloads Overridable Overrides ParamArray Preserve Private Property '+'Protected Public RaiseEvent ReadOnly ReDim REM RemoveHandler Resume '+'Return Select Set Shadows Shared Short Single Static Step Stop String '+'Structure Sub SyncLock Then Throw To True Try TypeOf Unicode Until '+'Variant When While With WithEvents WriteOnly Xor';this.regexList=[{regex:new RegExp('\'.*$','gm'),css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:new RegExp('^\\s*#.*','gm'),css:'preprocessor'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-vb';}
dp.sh.Brushes.Vb.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Vb.Aliases=['vb','vb.net'];
dp.sh.Brushes.Java=function()
{var keywords='abstract assert boolean break byte case catch char class const '+'continue default do double else enum extends '+'false final finally float for goto if implements import '+'instanceof int interface long native new null '+'package private protected public return '+'short static strictfp super switch synchronized this throw throws true '+'transient try void volatile while';this.regexList=[{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('\\b([\\d]+(\\.[\\d]+)?|0x[a-f0-9]+)\\b','gi'),css:'number'},{regex:new RegExp('(?!\\@interface\\b)\\@[\\$\\w]+\\b','g'),css:'annotation'},{regex:new RegExp('\\@interface\\b','g'),css:'keyword'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-j';this.Style='.dp-j .annotation { color: #646464; }'+'.dp-j .number { color: #C00000; }';}
dp.sh.Brushes.Java.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Java.Aliases=['java'];
dp.sh.Brushes.Cpp=function()
{var datatypes='ATOM BOOL BOOLEAN BYTE CHAR COLORREF DWORD DWORDLONG DWORD_PTR '+'DWORD32 DWORD64 FLOAT HACCEL HALF_PTR HANDLE HBITMAP HBRUSH '+'HCOLORSPACE HCONV HCONVLIST HCURSOR HDC HDDEDATA HDESK HDROP HDWP '+'HENHMETAFILE HFILE HFONT HGDIOBJ HGLOBAL HHOOK HICON HINSTANCE HKEY '+'HKL HLOCAL HMENU HMETAFILE HMODULE HMONITOR HPALETTE HPEN HRESULT '+'HRGN HRSRC HSZ HWINSTA HWND INT INT_PTR INT32 INT64 LANGID LCID LCTYPE '+'LGRPID LONG LONGLONG LONG_PTR LONG32 LONG64 LPARAM LPBOOL LPBYTE LPCOLORREF '+'LPCSTR LPCTSTR LPCVOID LPCWSTR LPDWORD LPHANDLE LPINT LPLONG LPSTR LPTSTR '+'LPVOID LPWORD LPWSTR LRESULT PBOOL PBOOLEAN PBYTE PCHAR PCSTR PCTSTR PCWSTR '+'PDWORDLONG PDWORD_PTR PDWORD32 PDWORD64 PFLOAT PHALF_PTR PHANDLE PHKEY PINT '+'PINT_PTR PINT32 PINT64 PLCID PLONG PLONGLONG PLONG_PTR PLONG32 PLONG64 POINTER_32 '+'POINTER_64 PSHORT PSIZE_T PSSIZE_T PSTR PTBYTE PTCHAR PTSTR PUCHAR PUHALF_PTR '+'PUINT PUINT_PTR PUINT32 PUINT64 PULONG PULONGLONG PULONG_PTR PULONG32 PULONG64 '+'PUSHORT PVOID PWCHAR PWORD PWSTR SC_HANDLE SC_LOCK SERVICE_STATUS_HANDLE SHORT '+'SIZE_T SSIZE_T TBYTE TCHAR UCHAR UHALF_PTR UINT UINT_PTR UINT32 UINT64 ULONG '+'ULONGLONG ULONG_PTR ULONG32 ULONG64 USHORT USN VOID WCHAR WORD WPARAM WPARAM WPARAM '+'char bool short int __int32 __int64 __int8 __int16 long float double __wchar_t '+'clock_t _complex _dev_t _diskfree_t div_t ldiv_t _exception _EXCEPTION_POINTERS '+'FILE _finddata_t _finddatai64_t _wfinddata_t _wfinddatai64_t __finddata64_t '+'__wfinddata64_t _FPIEEE_RECORD fpos_t _HEAPINFO _HFILE lconv intptr_t '+'jmp_buf mbstate_t _off_t _onexit_t _PNH ptrdiff_t _purecall_handler '+'sig_atomic_t size_t _stat __stat64 _stati64 terminate_function '+'time_t __time64_t _timeb __timeb64 tm uintptr_t _utimbuf '+'va_list wchar_t wctrans_t wctype_t wint_t signed';var keywords='break case catch class const __finally __exception __try '+'const_cast continue private public protected __declspec '+'default delete deprecated dllexport dllimport do dynamic_cast '+'else enum explicit extern if for friend goto inline '+'mutable naked namespace new noinline noreturn nothrow '+'register reinterpret_cast return selectany '+'sizeof static static_cast struct switch template this '+'thread throw true false try typedef typeid typename union '+'using uuid virtual void volatile whcar_t while';this.regexList=[{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('^ *#.*','gm'),css:'preprocessor'},{regex:new RegExp(this.GetKeywords(datatypes),'gm'),css:'datatypes'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-cpp';this.Style='.dp-cpp .datatypes { color: #2E8B57; font-weight: bold; }';}
dp.sh.Brushes.Cpp.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Cpp.Aliases=['cpp','c','c++'];
dp.sh.Brushes.CSS=function()
{var keywords='ascent azimuth background-attachment background-color background-image background-position '+'background-repeat background baseline bbox border-collapse border-color border-spacing border-style border-top '+'border-right border-bottom border-left border-top-color border-right-color border-bottom-color border-left-color '+'border-top-style border-right-style border-bottom-style border-left-style border-top-width border-right-width '+'border-bottom-width border-left-width border-width border cap-height caption-side centerline clear clip color '+'content counter-increment counter-reset cue-after cue-before cue cursor definition-src descent direction display '+'elevation empty-cells float font-size-adjust font-family font-size font-stretch font-style font-variant font-weight font '+'height letter-spacing line-height list-style-image list-style-position list-style-type list-style margin-top '+'margin-right margin-bottom margin-left margin marker-offset marks mathline max-height max-width min-height min-width orphans '+'outline-color outline-style outline-width outline overflow padding-top padding-right padding-bottom padding-left padding page '+'page-break-after page-break-before page-break-inside pause pause-after pause-before pitch pitch-range play-during position '+'quotes richness size slope src speak-header speak-numeral speak-punctuation speak speech-rate stemh stemv stress '+'table-layout text-align text-decoration text-indent text-shadow text-transform unicode-bidi unicode-range units-per-em '+'vertical-align visibility voice-family volume white-space widows width widths word-spacing x-height z-index';var values='above absolute all always aqua armenian attr aural auto avoid baseline behind below bidi-override black blink block blue bold bolder '+'both bottom braille capitalize caption center center-left center-right circle close-quote code collapse compact condensed '+'continuous counter counters crop cross crosshair cursive dashed decimal decimal-leading-zero default digits disc dotted double '+'embed embossed e-resize expanded extra-condensed extra-expanded fantasy far-left far-right fast faster fixed format fuchsia '+'gray green groove handheld hebrew help hidden hide high higher icon inline-table inline inset inside invert italic '+'justify landscape large larger left-side left leftwards level lighter lime line-through list-item local loud lower-alpha '+'lowercase lower-greek lower-latin lower-roman lower low ltr marker maroon medium message-box middle mix move narrower '+'navy ne-resize no-close-quote none no-open-quote no-repeat normal nowrap n-resize nw-resize oblique olive once open-quote outset '+'outside overline pointer portrait pre print projection purple red relative repeat repeat-x repeat-y rgb ridge right right-side '+'rightwards rtl run-in screen scroll semi-condensed semi-expanded separate se-resize show silent silver slower slow '+'small small-caps small-caption smaller soft solid speech spell-out square s-resize static status-bar sub super sw-resize '+'table-caption table-cell table-column table-column-group table-footer-group table-header-group table-row table-row-group teal '+'text-bottom text-top thick thin top transparent tty tv ultra-condensed ultra-expanded underline upper-alpha uppercase upper-latin '+'upper-roman url visible wait white wider w-resize x-fast x-high x-large x-loud x-low x-slow x-small x-soft xx-large xx-small yellow';var fonts='[mM]onospace [tT]ahoma [vV]erdana [aA]rial [hH]elvetica [sS]ans-serif [sS]erif';this.regexList=[{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('\\#[a-zA-Z0-9]{3,6}','g'),css:'value'},{regex:new RegExp('(-?\\d+)(\.\\d+)?(px|em|pt|\:|\%|)','g'),css:'value'},{regex:new RegExp('!important','g'),css:'important'},{regex:new RegExp(this.GetKeywordsCSS(keywords),'gm'),css:'keyword'},{regex:new RegExp(this.GetValuesCSS(values),'g'),css:'value'},{regex:new RegExp(this.GetValuesCSS(fonts),'g'),css:'value'}];this.CssClass='dp-css';this.Style='.dp-css .value { color: black; }'+'.dp-css .important { color: red; }';}
dp.sh.Highlighter.prototype.GetKeywordsCSS=function(str)
{return'\\b([a-z_]|)'+str.replace(/ /g,'(?=:)\\b|\\b([a-z_\\*]|\\*|)')+'(?=:)\\b';}
dp.sh.Highlighter.prototype.GetValuesCSS=function(str)
{return'\\b'+str.replace(/ /g,'(?!-)(?!:)\\b|\\b()')+'\:\\b';}
dp.sh.Brushes.CSS.prototype=new dp.sh.Highlighter();dp.sh.Brushes.CSS.Aliases=['css'];
dp.sh.Brushes.Delphi=function()
{var keywords='abs addr and ansichar ansistring array as asm begin boolean byte cardinal '+'case char class comp const constructor currency destructor div do double '+'downto else end except exports extended false file finalization finally '+'for function goto if implementation in inherited int64 initialization '+'integer interface is label library longint longword mod nil not object '+'of on or packed pansichar pansistring pchar pcurrency pdatetime pextended '+'pint64 pointer private procedure program property pshortstring pstring '+'pvariant pwidechar pwidestring protected public published raise real real48 '+'record repeat set shl shortint shortstring shr single smallint string then '+'threadvar to true try type unit until uses val var varirnt while widechar '+'widestring with word write writeln xor';this.regexList=[{regex:new RegExp('\\(\\*[\\s\\S]*?\\*\\)','gm'),css:'comment'},{regex:new RegExp('{(?!\\$)[\\s\\S]*?}','gm'),css:'comment'},{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('\\{\\$[a-zA-Z]+ .+\\}','g'),css:'directive'},{regex:new RegExp('\\b[\\d\\.]+\\b','g'),css:'number'},{regex:new RegExp('\\$[a-zA-Z0-9]+\\b','g'),css:'number'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-delphi';this.Style='.dp-delphi .number { color: blue; }'+'.dp-delphi .directive { color: #008284; }'+'.dp-delphi .vars { color: #000; }';}
dp.sh.Brushes.Delphi.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Delphi.Aliases=['delphi','pascal'];
dp.sh.Brushes.Python=function()
{var keywords='and assert break class continue def del elif else '+'except exec finally for from global if import in is '+'lambda not or pass print raise return try yield while';var special='None True False self cls class_'
this.regexList=[{regex:dp.sh.RegexLib.SingleLinePerlComments,css:'comment'},{regex:new RegExp("^\\s*@\\w+",'gm'),css:'decorator'},{regex:new RegExp("(['\"]{3})([^\\1])*?\\1",'gm'),css:'comment'},{regex:new RegExp('"(?!")(?:\\.|\\\\\\"|[^\\""\\n\\r])*"','gm'),css:'string'},{regex:new RegExp("'(?!')*(?:\\.|(\\\\\\')|[^\\''\\n\\r])*'",'gm'),css:'string'},{regex:new RegExp("\\b\\d+\\.?\\w*",'g'),css:'number'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'},{regex:new RegExp(this.GetKeywords(special),'gm'),css:'special'}];this.CssClass='dp-py';this.Style='.dp-py .builtins { color: #ff1493; }'+'.dp-py .magicmethods { color: #808080; }'+'.dp-py .exceptions { color: brown; }'+'.dp-py .types { color: brown; font-style: italic; }'+'.dp-py .commonlibs { color: #8A2BE2; font-style: italic; }';}
dp.sh.Brushes.Python.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Python.Aliases=['py','python'];
dp.sh.Brushes.Php=function()
{var funcs='abs acos acosh addcslashes addslashes '+'array_change_key_case array_chunk array_combine array_count_values array_diff '+'array_diff_assoc array_diff_key array_diff_uassoc array_diff_ukey array_fill '+'array_filter array_flip array_intersect array_intersect_assoc array_intersect_key '+'array_intersect_uassoc array_intersect_ukey array_key_exists array_keys array_map '+'array_merge array_merge_recursive array_multisort array_pad array_pop array_product '+'array_push array_rand array_reduce array_reverse array_search array_shift '+'array_slice array_splice array_sum array_udiff array_udiff_assoc '+'array_udiff_uassoc array_uintersect array_uintersect_assoc '+'array_uintersect_uassoc array_unique array_unshift array_values array_walk '+'array_walk_recursive atan atan2 atanh base64_decode base64_encode base_convert '+'basename bcadd bccomp bcdiv bcmod bcmul bindec bindtextdomain bzclose bzcompress '+'bzdecompress bzerrno bzerror bzerrstr bzflush bzopen bzread bzwrite ceil chdir '+'checkdate checkdnsrr chgrp chmod chop chown chr chroot chunk_split class_exists '+'closedir closelog copy cos cosh count count_chars date decbin dechex decoct '+'deg2rad delete ebcdic2ascii echo empty end ereg ereg_replace eregi eregi_replace error_log '+'error_reporting escapeshellarg escapeshellcmd eval exec exit exp explode extension_loaded '+'feof fflush fgetc fgetcsv fgets fgetss file_exists file_get_contents file_put_contents '+'fileatime filectime filegroup fileinode filemtime fileowner fileperms filesize filetype '+'floatval flock floor flush fmod fnmatch fopen fpassthru fprintf fputcsv fputs fread fscanf '+'fseek fsockopen fstat ftell ftok getallheaders getcwd getdate getenv gethostbyaddr gethostbyname '+'gethostbynamel getimagesize getlastmod getmxrr getmygid getmyinode getmypid getmyuid getopt '+'getprotobyname getprotobynumber getrandmax getrusage getservbyname getservbyport gettext '+'gettimeofday gettype glob gmdate gmmktime ini_alter ini_get ini_get_all ini_restore ini_set '+'interface_exists intval ip2long is_a is_array is_bool is_callable is_dir is_double '+'is_executable is_file is_finite is_float is_infinite is_int is_integer is_link is_long '+'is_nan is_null is_numeric is_object is_readable is_real is_resource is_scalar is_soap_fault '+'is_string is_subclass_of is_uploaded_file is_writable is_writeable mkdir mktime nl2br '+'parse_ini_file parse_str parse_url passthru pathinfo readlink realpath rewind rewinddir rmdir '+'round str_ireplace str_pad str_repeat str_replace str_rot13 str_shuffle str_split '+'str_word_count strcasecmp strchr strcmp strcoll strcspn strftime strip_tags stripcslashes '+'stripos stripslashes stristr strlen strnatcasecmp strnatcmp strncasecmp strncmp strpbrk '+'strpos strptime strrchr strrev strripos strrpos strspn strstr strtok strtolower strtotime '+'strtoupper strtr strval substr substr_compare';var keywords='and or xor __FILE__ __LINE__ array as break case '+'cfunction class const continue declare default die do else '+'elseif empty enddeclare endfor endforeach endif endswitch endwhile '+'extends for foreach function include include_once global if '+'new old_function return static switch use require require_once '+'var while __FUNCTION__ __CLASS__ '+'__METHOD__ abstract interface public implements extends private protected throw';this.regexList=[{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('\\$\\w+','g'),css:'vars'},{regex:new RegExp(this.GetKeywords(funcs),'gmi'),css:'func'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-c';}
dp.sh.Brushes.Php.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Php.Aliases=['php'];
dp.sh.Brushes.Sql=function()
{var funcs='abs avg case cast coalesce convert count current_timestamp '+'current_user day isnull left lower month nullif replace right '+'session_user space substring sum system_user upper user year';var keywords='absolute action add after alter as asc at authorization begin bigint '+'binary bit by cascade char character check checkpoint close collate '+'column commit committed connect connection constraint contains continue '+'create cube current current_date current_time cursor database date '+'deallocate dec decimal declare default delete desc distinct double drop '+'dynamic else end end-exec escape except exec execute false fetch first '+'float for force foreign forward free from full function global goto grant '+'group grouping having hour ignore index inner insensitive insert instead '+'int integer intersect into is isolation key last level load local max min '+'minute modify move name national nchar next no numeric of off on only '+'open option order out output partial password precision prepare primary '+'prior privileges procedure public read real references relative repeatable '+'restrict return returns revoke rollback rollup rows rule schema scroll '+'second section select sequence serializable set size smallint static '+'statistics table temp temporary then time timestamp to top transaction '+'translation trigger true truncate uncommitted union unique update values '+'varchar varying view when where with work';var operators='all and any between cross in join like not null or outer some';this.regexList=[{regex:new RegExp('--(.*)$','gm'),css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp(this.GetKeywords(funcs),'gmi'),css:'func'},{regex:new RegExp(this.GetKeywords(operators),'gmi'),css:'op'},{regex:new RegExp(this.GetKeywords(keywords),'gmi'),css:'keyword'}];this.CssClass='dp-sql';this.Style='.dp-sql .func { color: #ff1493; }'+'.dp-sql .op { color: #808080; }';}
dp.sh.Brushes.Sql.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Sql.Aliases=['sql'];
dp.sh.Brushes.Xml=function()
{this.CssClass='dp-xml';this.Style='.dp-xml .cdata { color: #ff1493; }'+'.dp-xml .tag, .dp-xml .tag-name { color: #069; font-weight: bold; }'+'.dp-xml .attribute { color: red; }'+'.dp-xml .attribute-value { color: blue; }';}
dp.sh.Brushes.Xml.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Xml.Aliases=['xml','xhtml','xslt','html','xhtml'];dp.sh.Brushes.Xml.prototype.ProcessRegexList=function()
{function push(array,value)
{array[array.length]=value;}
var index=0;var match=null;var regex=null;this.GetMatches(new RegExp('(\&lt;|<)\\!\\[[\\w\\s]*?\\[(.|\\s)*?\\]\\](\&gt;|>)','gm'),'cdata');this.GetMatches(new RegExp('(\&lt;|<)!--\\s*.*?\\s*--(\&gt;|>)','gm'),'comments');regex=new RegExp('([:\\w-\.]+)\\s*=\\s*(".*?"|\'.*?\'|\\w+)*|(\\w+)','gm');while((match=regex.exec(this.code))!=null)
{if(match[1]==null)
{continue;}
push(this.matches,new dp.sh.Match(match[1],match.index,'attribute'));if(match[2]!=undefined)
{push(this.matches,new dp.sh.Match(match[2],match.index+match[0].indexOf(match[2]),'attribute-value'));}}
this.GetMatches(new RegExp('(\&lt;|<)/*\\?*(?!\\!)|/*\\?*(\&gt;|>)','gm'),'tag');regex=new RegExp('(?:\&lt;|<)/*\\?*\\s*([:\\w-\.]+)','gm');while((match=regex.exec(this.code))!=null)
{push(this.matches,new dp.sh.Match(match[1],match.index+match[0].indexOf(match[1]),'tag-name'));}}
dp.sh.Brushes.JScript=function()
{var keywords='abstract boolean break byte case catch char class const continue debugger '+'default delete do double else enum export extends false final finally float '+'for function goto if implements import in instanceof int interface long native '+'new null package private protected public return short static super switch '+'synchronized this throw throws transient true try typeof var void volatile while with';this.regexList=[{regex:dp.sh.RegexLib.SingleLineCComments,css:'comment'},{regex:dp.sh.RegexLib.MultiLineCComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp('^\\s*#.*','gm'),css:'preprocessor'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'}];this.CssClass='dp-c';}
dp.sh.Brushes.JScript.prototype=new dp.sh.Highlighter();dp.sh.Brushes.JScript.Aliases=['js','jscript','javascript'];
dp.sh.Brushes.Ruby=function()
{var keywords='alias and BEGIN begin break case class def define_method defined do each else elsif '+'END end ensure false for if in module new next nil not or raise redo rescue retry return '+'self super then throw true undef unless until when while yield';var builtins='Array Bignum Binding Class Continuation Dir Exception FalseClass File::Stat File Fixnum Fload '+'Hash Integer IO MatchData Method Module NilClass Numeric Object Proc Range Regexp String Struct::TMS Symbol '+'ThreadGroup Thread Time TrueClass'
this.regexList=[{regex:dp.sh.RegexLib.SingleLinePerlComments,css:'comment'},{regex:dp.sh.RegexLib.DoubleQuotedString,css:'string'},{regex:dp.sh.RegexLib.SingleQuotedString,css:'string'},{regex:new RegExp(':[a-z][A-Za-z0-9_]*','g'),css:'symbol'},{regex:new RegExp('(\\$|@@|@)\\w+','g'),css:'variable'},{regex:new RegExp(this.GetKeywords(keywords),'gm'),css:'keyword'},{regex:new RegExp(this.GetKeywords(builtins),'gm'),css:'builtin'}];this.CssClass='dp-rb';this.Style='.dp-rb .symbol { color: #a70; }'+'.dp-rb .variable { color: #a70; font-weight: bold; }';}
dp.sh.Brushes.Ruby.prototype=new dp.sh.Highlighter();dp.sh.Brushes.Ruby.Aliases=['ruby','rails','ror'];
dp.SyntaxHighlighter.BloggerMode();
dp.SyntaxHighlighter.HighlightAll('code');
--></script>

2011/03/12

メイン画面の作成

「Hello World」の表示ができたので、次はメイン画面(とりあえず画面だけ、動作部分はまだまだ先になりそう)の作成をしてみる。
http://tutorial.jp/prog/」のAndroid実習マニュアルを参考に、SpinnerとListViewを使って、Gmailのラベル選択部分とメール選択部分を作った。

1.「/res/main.xml」をダブルクリックして「Grahical Layout」からSpinnerとListView、動作確認用にTexiViewをそれぞれドラッグ&ドロップ。
2.Gmail2ReSendActivity.javaを以下のようにコーディングして実行。


package jp.taku1974.Gmail2ReSend;

import android.app.Activity;
import android.os.Bundle;
import android.widget.ArrayAdapter;    // Spinner,ListViwe用アダプタ
import android.widget.AdapterView;    // Spinner,ListViwe用アダプタ
import android.view.View;
import android.widget.Spinner;
import android.widget.ListView;
import android.widget.TextView;

public class Gmail2ReSendActivity extends Activity {
    /** Called when the activity is first created. */
    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.main);
        // ラベル一覧
        ArrayAdapter<GmailLabel> adapterL =  new ArrayAdapter<GmailLabel>(this,android.R.layout.simple_spinner_item);
        adapterL.setDropDownViewResource(android.R.layout.simple_spinner_dropdown_item);
        adapterL.add(new GmailLabel("送信済みメール"));
        adapterL.add(new GmailLabel("受信トレイ"));
        adapterL.add(new GmailLabel("スター付き"));
        adapterL.add(new GmailLabel("下書き"));
        final Spinner spinner = (SpinnerfindViewById(R.id.spinner);
        spinner.setAdapter(adapterL);
        spinner.setSelection(0);
        spinner.setOnItemSelectedListener(
          new AdapterView.OnItemSelectedListener() {
            public void onItemSelected(AdapterView<?> parent, View view, int position, long id) {
              setSelectedLabel(parent, position);
            }
            public void onNothingSelected(AdapterView<?> parent) {
            }
          }
        );
        // メッセージ一覧
        ArrayAdapter<GmailMessage> adapterM = new ArrayAdapter<GmailMessage>(this,android.R.layout.simple_list_item_1);
        adapterM.add(new GmailMessage("aaa@gmail.com""あの件"));
        adapterM.add(new GmailMessage("bbb@yahoo.co.jp""その件"));
        adapterM.add(new GmailMessage("ccc@docomo.ne.jp""どの件?"));
        adapterM.add(new GmailMessage("ddd@gmail.com""例の件"));
        final ListView listview = (ListViewfindViewById(R.id.listview);
        listview.setAdapter(adapterM);
        listview.setOnItemClickListener(
          new AdapterView.OnItemClickListener() {
            public void onItemClick(AdapterView<?> parent,View view, int position, long id) {
              setSelectedMessage(parent, position);
            }
          }
        );
    }
    
  protected void setSelectedLabel(AdapterView<?> parent, int position) {
    GmailLabel label = (GmailLabelparent.getAdapter().getItem(position);
    final TextView selecteditem = (TextViewfindViewById(R.id.selecteditem);
    selecteditem.setText(label.toString());
  }
  
  protected void setSelectedMessage(AdapterView<?> parent, int position) {
    GmailMessage gmess = (GmailMessageparent.getAdapter().getItem(position);
    final TextView selecteditem =
    (TextViewfindViewById(R.id.selecteditem);
    selecteditem.setText("TO:" + gmess.getTo() "\n" "Subject:" + gmess.getSubject());
  }
}

class GmailLabel {
  String strLabel;
  GmailLabel(String Label) {
    strLabel = Label;
  }
  public String toString() {
    return strLabel ;
  }
}

class GmailMessage {
  String strTo;
  String strSubject;
  GmailMessage(String To, String Subject) {
    strTo = To;
    strSubject = Subject;
  }
  public String toString() {
  return "TO:" + strTo + "\n" "Subject:" + strSubject ;
  }
  public String getTo() {
  return strTo;
  }
  public String getSubject() {
  return strSubject;
  }
}

2011/03/11

Java2Html

EclipseでコーディングしたものをWebにアップする為にHtml出力にしてくれるプラグイン。
http://www.java2html.de/eclipse.html」から、「java2html_eclipse_150.zip」をダウンロード&解凍。
Eclipseのpluginsフォルダに解凍した「de.java2html_1.5.0」をフォルダごとコピーしてEclipseを再起動。
使用方法
1.Eclipseでアップしたいコード部分を選択して、右クリック→Java2Htmlでクリップボードにコピー
(出力フォーマットはXTML 1.0 Transitional(inlined fonts)でよい)
2.ブログのHTML編集画面で貼り付け。
3.余分な、<html><head><title><body>タグを削除して完了。

出力結果がこれ↓



class GmailLabel {

  String strLabel;

  GmailLabel(String Label) {

    strLabel = Label;  // ラベルをセット

  }

  public String toString() {

    return strLabel ;  // 戻り

  }

}


2011/03/10

Android-SDKのインストール

まずは、開発環境のインストール。
http://itpro.nikkeibp.co.jp/article/COLUMN/20091008/338599/?ST=android-dev」の記事を見ながら、
・JDK(Java SE Development Kit)
・Eclipse IDE for Java Developers
・Android Development Tools(ADT)
をインストールして、「Pleiades」から日本語化プラグイン(Pleiades 本体)をダウンロードして解凍、readme_pleiades.txtの指示通りインストール。「日本語 Eclipse / Pleiades All in One 日本語ディストリビューション」をダウンロードして解凍しようとすると、ファイル名が長すぎて正常に解凍できないので注意。

次に、「http://android.roof-balcony.com/debug/add-on/」の記事を見ながら、
XperiaのAdd-on(エミュレータ)をダウンロード&解凍してAndroid SDKのインストールフォルダにある「add-ons」フォルダにコピー。

これで、androidアプリ開発の準備完了。

「Hello World」を作成してエミュレータで表示するところまで出来ました。

2011/03/09

Android Gmailアプリ

Android標準のメーラーがショボくてやりたいことが出来ない。
メーラーでやりたいこと、
①返信時の強制引用をしない
②メール本文のコピー
③送信済みメールを編集して再送信
④メーラーからgoogleカレンダーへ予定登録

①と②は、「Gmail Client」や「K-9 MAIL」で出来るので、今は「Gmail Client」を使用している。
④は、「Xメール」で出来るがメーラーとしてイマイチなので使っていない。今は手動でコピペしている。

なぜ、③の機能を持ったメーラーがAndroidでは無いのだろう?ガラケーやPCのメーラーでは、普通にある機能でよく使うと思うのだが、、、

と言うことで、無ければ作ってしまえ!ってノリでAndroidアプリ開発を始めてみます。
普通にありそうな機能なのに実装されたアプリが無いと言うことは何か技術的に問題があるのかな?出来るか不安になってきたが、、、

まずは、③の機能を持ったメール送信アプリの開発から。
名前は「Gmail2ReSend」にしよう!

④の機能を持ったメール受信アプリの名前も決めた「Gmail2Calendar」

出来るかどうか、乞う御期待!