文藝氣質(zhì)的豎排文本控件來(lái)啦

PlumbTextView

PlumbTextView是一個(gè)豎排列的文本控件。你可以很容易使用它定義多種豎排文本風(fēng)格萍肆。

Feature

  1. 將文本豎直排列切省;
  2. 可以設(shè)置文本的列距和字距您旁;
  3. 可以添加一個(gè)正則表達(dá)式去分割文本靴寂。PlumbTextView會(huì)在正則表達(dá)式所包含的字符處換列磷蜀,并且其中的字符不會(huì)顯示在PlumbTextView中;
  4. 可以在每一列文本的坐標(biāo)添加一跟豎線(xiàn)榨汤;
  5. 可以為文本設(shè)置字體風(fēng)格和第三方字體。

Screenshot

plumb_textview.gif
plumb_textview.gif

Gradle Dependency

compile 'cc.sayaki.widget:plumb-textview:1.0.1'

Usage

你可以很容易的使用PlumbTextView怎茫,就像使用TextView那樣收壕。只需要在xml或者Java代碼中設(shè)置你想要的屬性效果就行了妓灌。

<cc.sayaki.widget.PlumbTextView
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:paddingBottom="30dp"
   android:paddingTop="30dp"
   sayaki:columnSpacing="20dp"
   sayaki:leftLine="true"
   sayaki:leftLineColor="@color/colorAccent"
   sayaki:leftLinePadding="4dp"
   sayaki:letterSpacing="6dp"
   sayaki:regex="[,蜜宪。虫埂?!]"
   sayaki:text="@string/text"
   sayaki:textColor="@color/colorAccent"
   sayaki:textSize="16sp"
   sayaki:textStyle="bold|italic" />

Thinking

接下來(lái)講講PlumbTextView的實(shí)現(xiàn)思路圃验。
思路很簡(jiǎn)單掉伏,主要就是在繪制完一個(gè)字符之后,將下次繪制的位置垂直向下移動(dòng)一個(gè)字符的高度(包括字符本身的高度和兩個(gè)字符之間的字距)澳窑。只是垂直繪制文字很簡(jiǎn)單斧散,不過(guò)處理?yè)Q列是一個(gè)比較麻煩的地方。

PlumbTextView有兩種換列情況:

  1. 單列文本超過(guò)了PlumbTextView可以顯示的內(nèi)容高度摊聋;
  2. 提供了正則表達(dá)式鸡捐,PlumbTextView會(huì)在正則表達(dá)式中所包含的字符處換列。

首先測(cè)量PlumbTextView自身的高度

if (heightMode == MeasureSpec.EXACTLY) {
    height = heightSize;
} else {
    if (!TextUtils.isEmpty(regex)) {
        height = 0;
        String[] texts = text.toString().split(regex);
        for (String s : texts) {
            height = Math.max(height, getDesiredHeight(s));
        }
        height += letterSpacing;
    } else {
        height = getDesiredHeight(text.toString());
    }
    if (height > heightSize) {
        height = heightSize;
    }
}

根據(jù)換列規(guī)則拆分文本

if (!TextUtils.isEmpty(regex)) {
    String[] texts = text.toString().split(regex);
    for (String s : texts) {
        getFormatTexts(s);
    }
} else {
    getFormatTexts(text.toString());
}

// 獲取拆分后的文本
private void getFormatTexts(String s) {
    int contentHeight = height - getPaddingTop() - getPaddingBottom();
    if (getDesiredHeight(s) > contentHeight) {
        int count = contentHeight / charHeight;
        int i = 0;
        // 有的文本拆分過(guò)后可能仍然大于控件可顯示的高度麻裁,需要再拆分
        for (; i < getDesiredHeight(s) / contentHeight; i++) {
            formatTexts.add(s.substring(i * count, (i + 1) * count));
        }
        // 最后一列文本不滿(mǎn)一列
        if (getDesiredHeight(s) % contentHeight != 0) {
            formatTexts.add(s.substring(i * count, s.length()));
        }
    } else {
        formatTexts.add(s);
    }
}

測(cè)量PlumbTextView的寬度

if (widthMode == MeasureSpec.EXACTLY) {
    width = widthSize;
} else {
    if (!TextUtils.isEmpty(regex)) {
        width = columnWidth * formatTexts.size();
    } else {
        width = columnWidth * (getDesiredHeight(text.toString())
                / (height - getPaddingTop() - getPaddingBottom()) + 1);
    }
    if (width > widthSize) {
        width = widthSize;
    }
}

上述操作均在onMeasure方法中箍镜,因此為了避免多次測(cè)量造成拆分后的文本重復(fù),在每次拆分之前先清空f(shuō)ormatTexts煎源。

現(xiàn)在已經(jīng)獲得了按列拆分好的文本了色迂,要繪制就變得簡(jiǎn)單了。

@Override
protected void onDraw(Canvas canvas) {
    float x = width - getPaddingLeft() - getPaddingRight();
    float y = getPaddingTop();
    for (int i = 0; i < formatTexts.size(); i++) {
        // 換列
        x = i == 0 ? width - columnWidth + columnSpacing : x - columnWidth;
        // 繪制每一列文本
        for (int j = 0; j < formatTexts.get(i).length(); j++) {
            // 向下移動(dòng)繪制點(diǎn)
            y = j == 0 ? charHeight - letterSpacing + getPaddingTop() : y + charHeight;
            canvas.drawText(formatTexts.get(i), j, j + 1, x, y, textPaint);
        }
        if (leftLine) {
            // 在每列文本之后繪制豎線(xiàn)
            canvas.drawLine(x - leftLinePadding, getPaddingTop(),
                    x - leftLinePadding, y + letterSpacing, leftLinePaint);
        }
    }
}

大致的流程就是這樣手销,還是比較清晰的歇僧。順便說(shuō)一點(diǎn),繪制的文本要想有個(gè)不錯(cuò)的效果原献,應(yīng)該去研究一些TextPaint這個(gè)類(lèi)哦馏慨,不然可能出現(xiàn)一些文字周?chē)瞻走^(guò)多,或者文字本身繪制的不完整姑隅。

好了写隶,文藝氣質(zhì)滿(mǎn)滿(mǎn)的豎排文本控件的介紹就到這里了吧,大家喜歡的話(huà)請(qǐng)去我的GitHub star哦 > <

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末讲仰,一起剝皮案震驚了整個(gè)濱河市慕趴,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌鄙陡,老刑警劉巖冕房,帶你破解...
    沈念sama閱讀 217,542評(píng)論 6 504
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異趁矾,居然都是意外死亡耙册,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,822評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)毫捣,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)详拙,“玉大人帝际,你說(shuō)我怎么就攤上這事∪恼蓿” “怎么了蹲诀?”我有些...
    開(kāi)封第一講書(shū)人閱讀 163,912評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵,是天一觀的道長(zhǎng)弃揽。 經(jīng)常有香客問(wèn)我脯爪,道長(zhǎng),這世上最難降的妖魔是什么矿微? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,449評(píng)論 1 293
  • 正文 為了忘掉前任痕慢,我火速辦了婚禮,結(jié)果婚禮上冷冗,老公的妹妹穿的比我還像新娘守屉。我一直安慰自己,他們只是感情好蒿辙,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,500評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布拇泛。 她就那樣靜靜地躺著,像睡著了一般思灌。 火紅的嫁衣襯著肌膚如雪俺叭。 梳的紋絲不亂的頭發(fā)上,一...
    開(kāi)封第一講書(shū)人閱讀 51,370評(píng)論 1 302
  • 那天泰偿,我揣著相機(jī)與錄音熄守,去河邊找鬼。 笑死耗跛,一個(gè)胖子當(dāng)著我的面吹牛裕照,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播调塌,決...
    沈念sama閱讀 40,193評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼晋南,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來(lái)了羔砾?” 一聲冷哼從身側(cè)響起负间,我...
    開(kāi)封第一講書(shū)人閱讀 39,074評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎姜凄,沒(méi)想到半個(gè)月后政溃,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,505評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡态秧,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,722評(píng)論 3 335
  • 正文 我和宋清朗相戀三年董虱,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片申鱼。...
    茶點(diǎn)故事閱讀 39,841評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡愤诱,死狀恐怖藏鹊,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情转锈,我是刑警寧澤,帶...
    沈念sama閱讀 35,569評(píng)論 5 345
  • 正文 年R本政府宣布楚殿,位于F島的核電站撮慨,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏脆粥。R本人自食惡果不足惜砌溺,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,168評(píng)論 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望变隔。 院中可真熱鬧规伐,春花似錦、人聲如沸匣缘。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,783評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)肌厨。三九已至培慌,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間柑爸,已是汗流浹背吵护。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,918評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留表鳍,地道東北人馅而。 一個(gè)月前我還...
    沈念sama閱讀 47,962評(píng)論 2 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像譬圣,于是被迫代替她去往敵國(guó)和親瓮恭。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,781評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 發(fā)現(xiàn) 關(guān)注 消息 iOS 第三方庫(kù)胁镐、插件偎血、知名博客總結(jié) 作者大灰狼的小綿羊哥哥關(guān)注 2017.06.26 09:4...
    肇東周閱讀 12,103評(píng)論 4 62
  • 今早六點(diǎn)鐘,被園子里唧唧喳喳的聲音喚醒盯漂,掀開(kāi)窗簾颇玷,奧,外面驕陽(yáng)似火就缆,好想重新進(jìn)入清涼的夢(mèng)鄉(xiāng)…… 但一想起今早還要產(chǎn)...
    夜闌星空xk閱讀 185評(píng)論 0 1
  • 你知道自己的未來(lái)是什么樣子嗎? 當(dāng)我提出這個(gè)問(wèn)句的時(shí)侯帖渠,你腦海里有沒(méi)有一幅美好未來(lái)的景象浮現(xiàn)在你的眼前,如果有...
    奔向自由的路上閱讀 544評(píng)論 0 0
  • 概述 YYCache是一個(gè)用來(lái)封裝客戶(hù)端緩存功能的庫(kù)竭宰,實(shí)現(xiàn)了二級(jí)緩存的機(jī)制空郊,即同時(shí)具備內(nèi)存緩存和硬盤(pán)緩存的功能份招。 ...
    egoCogito_panf閱讀 2,406評(píng)論 0 5