Kotlin Annotation Processor

如果有做過 Android 開發的話應該都有用過 annotation processor(又稱 codegen),即是在 build.gradle 入面要用 annotationProcessor 或者 kapt 的那些 dependency。用法大概是在 code 上加上一些 @ 開頭的 annotation,然後 build 出來就會自動幫你生成相關的 class。簡單來說 annotation processor 就是用 code 來讓 Java compiler 生成 code。通常都是用來生成一些內容重覆的 code 來代替自己人手寫。

自己做一個功能不多的 annotation processor 其實都不太難。難的地方是 debug 時不能像平常般加 breakpoint,加上我們要生成 Kotlin class 所以跟平常找到的教學會有輕微出入(因為大部分教學都是討論生成 Java class)。

Read More

Firebase Crashlytics 的 CrashlyticsOrgIdException 解決方法

最近為自己的 app 加入 Firebase Crashlytics SDK beta(即是使用 Google 的 Firebase Crashlytics Gradle plugin 而不是用 Fabric 那個),但在 build release APK 時出現下面的錯誤:

1
2
3
4
java.io.IOException: com.google.firebase.crashlytics.buildtools.exception.CrashlyticsOrgIdException: Could not fetch Crashlytics Org Id
> com.google.firebase.crashlytics.buildtools.exception.CrashlyticsOrgIdException: Could not fetch Crashlytics Org Id
> Could not fetch Crashlytics Org Id
> Unable to fetch Crashlytics Org Id using app id 1:731121766578:android:4e799392a2e62811

我的 app 是在 Firebase 開了兩個 project,一個是開發時用的,另一個是 production 用的。但只有 production 才有這個錯誤。如果把 release build 設為不上載 ProGuard/R8 mapping file 的話就不會有這個錯誤。

1
2
3
firebaseCrashlytics {
mappingFileUploadEnabled false
}

起初以為是那個 google-services.json 放錯位置,但我檢查過位置是正確。之後我找了 Firebase 支援,他們最後回覆說原因是我沒有在 production 的 Firebase project 交過一次 crash report,所以 production Firebase project 的 Crashlytics Org Id 就沒有產生出來,導致交不了 ProGuard/R8 mapping file。只不過他們的文檔沒有寫明一定要 crash 一次才算正式完成安裝 Crashlytics。

Firebase Cloud Messaging

最近工作需要做 Firebase Cloud Messaging (FCM) 整合,發現了向 Firebase API 直接送出 push 的 HTTP request 都可以生成不同種類的 message。

如果要整合到 Android 的話,需要建立一個新的 Service class 並繼承自 FirebaseMessagingService。這個 Service 有一個叫 onMessageReceived 的 callback method 來接收來自 FCM 的 push 和它的 payload。但原來不是所有的 push 都能被那個 callback 接到,要視乎 push 的種類和你的 app 當時在甚麼情況而定。

Read More

Android App Icon 規格

一個 app 的第一印象應該是它的 app icon (launcher icon)。在 Android,在不同的時期前後出了好幾個 app icon 規格。但是 Android 介紹不同種類的 app icon 的文件放得非常分散,如果平時沒有一直留意的話都幾乎肯定會做錯或者做漏。以下是全部 Android app 都會用到 app icon:

  • Launcher icon
  • Launcher icon(圓形)
  • Adaptive launcher icon
  • Google Play icon

如果想簡單地做出全部 icon 的話,可以用 Android Studio 內置的 Image Asset Studio。但如果想追求完美的話,還是自行準備圖片比較好。這篇文章整理了不同 icon 的基本規格,寫的時候盡量考慮到設計師。如果有需要的話可以分享給設計師同事參考。

Read More

OpenRefine GREL 筆記

OpenRefine 是一個開源的工具,用作檢視資料、加工處理後作其他用途。簡單的例子有一堆街名,部分街名用了全寫、部分用了縮寫,想將它們全部統一用全寫。它的定位是介乎 Excel 和自己寫程式之間。有時資料用 Excel 不太方便處理,但如果自己寫程式處理又因為程式只會用一次,感覺太麻煩。OpenRefine 相信可以解決到你的需要。

OpenRefine 內置了一種程式語言,名為 General Refine Expression Language (GREL),和 Excel 可以用公式差不多。我們用 OpenRefine 就是透過這種語言來把資料批量轉換成自己想要的東西。

值得一提的是 OpenRefine 以前由 Google 負責維護,所以介面會有以前 Google 產品的影子。

Read More

用 Google Apps Script 建立 Google Calendar event

Google Apps Script 是一套以 JavaScript 造的 API,可以讓你寫程式控制 Google Apps 內 Docs、Spreadsheet、Gmail、Drive 等等的功能。這次介紹如何用 Google Apps Script 建立 Google Calendar 的 event。我們會把部分建立 event 時所需要的資料放入 spreadsheet 內(會以 2019 年香港公眾假期作例子),然後用 Google Apps Script 讀取 spreadsheet 的內容再建立 event,效果就像 mail merge 般。

Read More

八達通新方向

最近,八達通終於做應該做的事了:商戶可以用八達通提供的商用版 app 經 NFC 向實體卡扣錢,小商戶就毋須租拍卡機就能接受八達通付款,亦都毋須使用 O! ePay 的 QR code 功能。

八達通當初就是一張單純的儲值卡。八達通開拓流動支付應該要數到在 Android 2.3 (Gingerbread) 支援 NFC 的時候見到有其他 Android 開發者推出查閱餘額 app 就學人推出一個查閱交易記錄 app(因為卡內的交易紀錄被加密,所以其他 app 只可以查閱餘額)。其後亦推出了八達通 SIM 卡,但現在應該終止推廣了。

Read More

Linkify 自動轉換成網址

最近工作需要將不完整的網址變成網址,但輸入的 string 可以是普通的文字,亦可以是一個沒有 http://https:// 的網址,亦可以一個完整的網址。但 TLD 有太多,如果自己寫 regular expression 做檢查的話那句 regular expression 就會好長,而且要定時補上日後新推出的 TLD。

之後查過原來 Android 有 Linkify/LinkifyCompat 這個 class:

Linkify take a piece of text and a regular expression and turns all of the regex matches in the text into clickable links. This is particularly useful for matching things like email addresses, web URLs, etc. and making them actionable. Alone with the pattern that is to be matched, a URL scheme prefix is also required. Any pattern match that does not begin with the supplied scheme will have the scheme prepended to the matched text when the clickable URL is created. For instance, if you are matching web URLs you would supply the scheme http://. If the pattern matches example.com, which does not have a URL scheme prefix, the supplied scheme will be prepended to create http://example.com when the clickable URL link is created.

看起來應該合用,不過一般用法都是用來將 TextView 的合適文字變成可點擊的超連結。下面是我的做法:

1
2
3
4
5
6
7
val query = "example.com"
val spannable = SpannableString(query)
LinkifyCompat.addLinks(spannable, Linkify.WEB_URLS)
val linkifyUrl = spannable.getSpans(0, spannable.length, URLSpan::class.java)
.filter { urlSpan -> spannable.getSpanStart(urlSpan) == 0 && spannable.getSpanEnd(urlSpan) == spannable.length }
.map { urlSpan -> urlSpan.url }
.firstOrNull()

首先將疑似網址的 string 變成 SpannableString。之後用 LinkifyCompat#addLinks 來檢查整個 string 有沒有網址。如果有的話,SpannableString 中的網址部分會有 span 包住。

這次的要求是整個 query string 最多只有一個網址,如果沒有網址或出現多於一個網址的話,就當作沒有網址處理。所以要加上 filter 來篩走出現多於一個網址的情況。

最後 linkifyUrl 會是 http://example.com;如果 query 沒有網址的話,那 linkifyUrl 會是 null

至於 LinkifyCompat 會不會定期更新 TLD 清單,我不太清楚,但當初出現 LinkifyCompat 就是用來補回新出的 TLD