樂生

最早是在我妹這兒看到。
總之,請支持樂生!也許樂生無法被保留,但是這次的事件可以揭露出政府處事的瑕疵,並要求他們改進。
事實上,這樣的事情不只是在樂生,還發生在許許多多的地方,如果你忽略了,下次這種事情可能就發生在你身邊!!

這一系列文章可以說是懶人包了:

然後還有 FAQ:快樂‧樂生─青年樂生聯盟行動網頁 – 樂生院完整說明(含Q&A)
其他參考連結:

快樂.樂生 ──壞消息! 樂生院可能在一個月內遭強制搬遷
快樂.樂生 ──樂生保存90%新方案 捷運局莫一錯再錯
快樂.樂生 ──反對強制拆遷樂生院連署聲明
苦勞論壇 ──懸崖邊的樂生院 北市捷運局限令一周搬遷
OJ.候診室 ──記憶對抗遺忘的掙扎
迷幻機器 ──誰現在跟樂生院民站一起,2008我投他/她!!!
被遺忘的島嶼 ──[老頭胡言亂語] 死人的牆比活人的家還要偉大
米店 ──轉文:北桃捷運網銜接共榮方案
Bbrother ──壞消息! 樂生院可能在一個月內遭強制搬遷
自由電子報 ──世界文化遺產不如中正廟圍牆
InnoNation Reloaded ──空間與正義:傾聽被遮蔽的聲音

在 mono 上使用 Font 的一些事

昨天碰到的事情,紀錄一下:

  1. mono 1.0 類別庫裡的 Font,有一些 ctor 裡面是空的。如果你發現使用了某個 function 沒作用,或是有些問題,你可以參考一下Mono Class Status以取得類別庫實作的狀況~或者利用Reflector for .NET反組譯看看,免得傻傻在那邊查了半天還不知道原因出在哪。
  2. 字型的底層是 libgdiplus,更底層則是CairoCairo又再使用了FreeType函式庫(好像還有別的,不過這是主要的)。所以字型的設定是取決於fontconfig的設定:/etc/fonts/fonts.conf(用FreeType好像跟這設定沒啥關係,總之我是這樣聯想到的),當你使用某個字型卻無法正常顯示時(例如中文),不妨參考設定然後直接指定字型名稱:

    Font textFont = new Font( “AR PL Mingti2L Big5”, 12, FontStyle.Regular, GraphicsUnit.Pixel, 1, false);

    ,應該就行了。

在.Net如何產生動畫gif (animated gif)

原生的 .Net framework 雖然可以產生 gif 圖檔,但卻無法產生有動畫效果的 gif 圖檔。
找了半天,本來已經萬念俱灰,想說可能要自己來寫了。
但終於還是在SourceForge.net找到合用的library:NGif

下載以後,可以直接參考程式,使用方法很簡單~
這邊摘錄製作的方法:

    /* create Gif */
    //you should replace filepath
    String [] imageFilePaths = new String[]{"G:\1.png"};
    String outputFilePath = "G:\\test.gif";
    AnimatedGifEncoder e = new AnimatedGifEncoder();
    e.Start( outputFilePath );
    e.SetDelay(500);
    //-1:no repeat,0:always repeat
    e.SetRepeat(0);
    for (int i = 0, count = imageFilePaths.Length; i < count; i++ )
    {
        e.AddFrame( Image.FromFile( imageFilePaths[i] ) );
    }
    e.Finish();

NGif同時也提供解開的方法,所以你也可以將 gif 裡面的每個 frame 都個別存為獨立的圖檔。

兒子便秘

012-我不喜歡喝水-03 (by Ellery)
昨天傍晚餵兒子喝奶,在他快喝完的時候,突然臉一陣紅一陣白的在用力。
我知道,這是他在嗯嗯的前兆,在努力數下之後,開始哭~我趕緊叫老婆過來幫忙。
於是我們趕緊把他尿布解開,想要幫他~
果然,撥開他的小菊花一看,就是有硬硬的便便卡住,我拿出溫度計沾上一些凡士林,插進他的小菊花刺激他~
於是他繼續嗯嗯,嗯到他哭喊,恩到哭著努力想要讓它解放,我們的心裡真的是很難過。
後來便便總算是出來了~還帶著一些血絲~
喝完奶之後,我們趕緊餵他喝了 30cc 葡萄糖水。
喝完以後跟他講了好久~讓他明白不可以因為痛就忍著不便便~
到了晚上九點多餵奶,他又變臉、哭喊,不過這次便便雖然也小硬,但卻順利了許多~(我想這可以證明他有聽懂我們說的話。)
以後喝完奶都要強迫他喝水了(因為他不愛喝水),早晚也要給他暗示說要便便~否則每次他便便都會是他自己的痛苦~
參考資料:

讀取網頁(3)

WebClient 畢竟還是有一些限制,例如 Cookie,這就沒辦法了,你必須要自行操作 Header。
可是 WebClient 並沒有任何成員可以讓你實現這塊。

而 WebClient 的底層,其實是用 WebRequest 來實現的。這裡的例子使用了
HttpWebRequest,但其實 HttpWebRequest 也是繼承自 WebRequest。
要加上 Cookie 的話,你得指定 HttpWebRequest 的 CookieContainer 屬性。

所以我們很快就可以寫出 wgetInWebRequest()。
using System;
using System.Text;
using System.Text.RegularExpressions;
using System.Net;
using System.Web;
using System.IO;
using System.Diagnostics;
using System.Collections;

public class Network
{
    public static string wgetInWebRequest( string url, CookieContainer cookies, Encoding encoding )
    {
        string responseData = "";
            
        try
        {
            HttpWebRequest webRequest = (HttpWebRequest)WebRequest.Create( url );

            // 加上 Cookie
            webRequest.CookieContainer = cookies;

            // 如果你有 Proxy 的話~
            // webRequest.Proxy = new WebProxy( "your_proxy", 3128 );

            // 加上 User Agent,用來模擬瀏覽器~
            //webRequest.Headers.Add( "User-Agent","Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)");

            StreamReader responseReader;
            responseReader = new StreamReader(webRequest.GetResponse().GetResponseStream(), encoding );

            // and read the response
            responseData = responseReader.ReadToEnd();
        }
        catch( Exception ex )
        {
            Debug.WriteLine( ex.ToString() );
        }
        finally
        {
            Debug.WriteLine( responseData );
        }
        return responseData;
    }
}
我想,也許你會問這有什麼用?
一般網站應用系統登入以後,都會設置 Cookie 以表示你已經登入了,換句話說,你只要先設法模擬登入,然後取得 WebResponse 所得到的 Cookie,接著再把得到的 Cookie 放到 WebRequest.CookieContainer,你接下來所有對該網站應用程式的 Request 就已經是有特定使用者身份的了~
讓我舉個 ASP.Net 網站的例子吧~
    public static void Login( string user, string pwd)
    {
        string loginPage = wgetInWebRequest( "http://your_web_app/login.aspx", null, Encoding.Default );

        // ASP.Net 會在頁面埋一個 __VIEWSTATE 隱藏表單變數,先取得!
        Regex rx = new Regex(@"\<input\ type=""hidden""\ name=""__VIEWSTATE""\ value=""(?<viewstate>.+)""\ /\>");
        string viewstate = "";

        try {
            // Find matches.
            MatchCollection matches = rx.Matches( loginPage );
                
            if( matches.Count == 1 )
            {
                // 要作 UrlEncode
                viewstate = HttpUtility.UrlEncode( matches[0].Groups["viewstate"].Value );

                // 用來收 cookie 的容器
                CookieContainer cookies = new CookieContainer();

                // now post to the login form
                HttpWebRequest webRequest = (HttpWebRequest)WebRequest.Create( "http://your_web_app/login.aspx" );

                // 模擬瀏覽器
                //webRequest.Headers.Add( "User-Agent","Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)");

                // POST
                webRequest.Method = "POST";
                webRequest.ContentType = "application/x-www-form-urlencoded";

                // 收到的 cookies 會放到 cookies 變數
                webRequest.CookieContainer = cookies;

                // 如果你有 proxy 的話
                // webRequest.Proxy = new WebProxy( "your_proxy", 3128 );

                // 這邊要看 login 表單有哪些變數而定,請自行變化
                string postData = string.Format( "__VIEWSTATE={0}&user={1}&password={2}&Submit=Submit", viewstate, user, pwd );
                StreamWriter requestWriter = new StreamWriter(webRequest.GetRequestStream());
                requestWriter.Write(postData);
                requestWriter.Close();

                // 收到內容啦,但是我們不需要他的結果,只要 cookies
                webRequest.GetResponse().Close();
                    
                // now we can send out cookie along with a request for the protected page
                string responseData = wgetInWebRequest( "http://your_web_app/default.aspx", cookies, Encoding.Default );
                Debug.WriteLine( responseData );
            }
            else
                Debug.WriteLine( "Internal error, too many ViewState." );
        }
        catch( Exception ex ) {
            Debug.WriteLine( ex.ToString() );
        }
        finally {
        }
    }

大致上就是這樣子,我不作太多的說明囉~
想要作更多的話,可以再研究HTTP Protocal並配合Sniffer之類的軟體去監看網路封包,來了解詳細的流程。

奶爸心得(2)-參考用的餵奶間隔

原本以為小朋友只要定時餵,就會乖乖的,實際上卻不然。
我家的小朋友,哭了,會有兩個可能原因:

  1. 餓了
  2. 想睡了

(尿布濕了反倒不是最主要的原因,濕了,他只會睡得比較不穩而已。)
在初期,他常常哭,老婆跟我始終搞不清楚他是想睡還是想吃。到後來,終於搞懂了,簡單的判定原則就是,吃”飽”以後的 3~4 小時,如果哭了,可以先簡單把餓的因素排除掉;反正就是先哄睡。如果還是哭,就換人哄,再不行,就應該是餓了~
所以這個餵奶間隔,真的就是參考用~

張超英病逝

今天在Yahoo!奇摩新聞上看到張超英病逝了~令人扼腕的消息~
原本不知道這個人對台灣的影響在哪裡。某次到敦南誠品看到宮前町九十番地這本書,翻了一會,發現這本書有種獨特的魅力,讓我沒辦法放手。從他身邊的故事與在國外所作的事情,可以很清楚的讓人體會到台灣外交的艱難處境與之前國民黨政府的顢頇(要不然怎麼會有民進黨美麗島事件?)~
在此悼念~並推薦這本書:宮前町九十番地

讀取網頁(2)

上一篇最後留下 Encoding 的問題還沒解決,事實上已經有解了:你可以改用 DownloadString() 函數。
DownloadString() 是怎麼做到的呢?
他是先看 WebRequest 得到的 Header 裡面是否有 Content-Type ,如果有指定的話,再看他裡面是否指定 charset,如果有,就以此編碼去讀取 stream。
如果沒有指定的話呢?.Net 2.0 的 WebClient 多提供了 Encoding 屬性,換句話說,就會以 Encoding 屬性指定的編碼去讀取 stream。
事實上,大部分的 browser 並沒這麼笨~他還會先去偷看網頁內容是否有指定 Content-Type 這個 meta tag:

<meta http-equiv=”Content-Type” content=”text/html; charset=UTF-8″ />

如果有的話,就會以此 charset 所指定的編碼來讀取。

讀取網頁(1)

如果你要搞個 spider 或是 bot,需要讀取網頁的話,用 WebClient 就綽綽有餘了~
using System.Net;

public class Network
{
    public static string wget( string url )
    {
        WebClient _client=new WebClient();
        string result="";
            
        try {
            // 藉著修改 Header,可以用來模擬某特定 Browser,以下是模擬 IE 6
            _client.Headers.Add("Accept","image/gif, image/x-xbitmap, image/jpeg, image/pjpeg, */*");
            _client.Headers.Add("Accept-Language","zh-tw");
            _client.Headers.Add("User-Agent","Mozilla/4.0 (compatible; MSIE 6.0; Windows NT 5.2; SV1; .NET CLR 1.1.4322; .NET CLR 2.0.50727)");

            // 表示支援壓縮,也就是說,你可以丟壓縮過的資料過來,我來解~
            //_client.Headers.Add("Accept-Encoding","gzip, deflate");

            // read
            System.IO.Stream objStream=_client.OpenRead( url );

            // 要知道正確的編碼,再去讀取~.Net會幫我們自動轉為 unicode 字串。這邊預設都是 UTF8
            System.IO.StreamReader _reader=new System.IO.StreamReader(objStream,System.Text.Encoding.UTF8);
            //System.IO.StreamReader _reader=new System.IO.StreamReader(objStream,System.Text.Encoding.GetEncoding(950));
            result = _reader.ReadToEnd();
        }
        catch( Exception ex ) {
            throw ex;
        }
        finally {
        }
            
        return result;
    }
}

所以我們就可以這麼用
Console.WriteLine( Network.wget("http://www.google.com.tw") );

目前有個決定性的缺點:必須先知道網頁編碼,我們才能讀到正確的文字~有辦法先知道網頁的編碼,再去決定要怎麼讀取嗎?

奶爸心得(1)-奶粉與紙尿布

地區:新店安坑/中和
我家用的是明治嬰兒奶粉,目前最低價在南勢角威安藥局:309。
家裏附近的藥局則是 319,一般大賣場是 325,價差可以差到 1x 元,不可說不大。
明治算是蠻便宜的了,其他牌子的奶粉可想而知。
紙尿布的話,我家並沒有特定愛用品牌,大體來說是混用,就是貴一點/比較舒服的在晚上用,便宜的/感覺上較差的在白天用。。
最早我娘是用布的尿布,後來先暫停,因為布的尿布太久沒換的話,容易造成上衣濕掉,所以必須經常更換。
冬天經常更換尿布又怕小朋友著涼,後來我娘還是先用紙尿布,等夏天的時候再說。
最低價是在全聯福利中心,大潤發有集點活動,藥局可能也有,但卻都不一定有全聯那麼便宜。
2007/3/10 更新:
奶粉與尿布最低價出現在永和中正路上的宏恩連鎖藥局,奶粉 299,尿布居然比全聯硬是再便宜了十元左右~