Selain menggambar teks multiline, seseorang mungkin kesulitan mendapatkan batas teks multiline (misalnya untuk meratakannya di kanvas).
Default paint.getTextBounds()
tidak akan berfungsi dalam kasus ini karena ini akan mengukur satu-satunya baris.
Untuk kenyamanan, saya membuat 2 fungsi ekstensi ini: satu untuk menggambar teks multiline, dan yang lainnya untuk mendapatkan batas teks.
private val textBoundsRect = Rect()
/**
* Draws multi-line text on the Canvas with the origin at (x,y), using the specified paint. The origin is interpreted
* based on the Align setting in the paint.
*
* @param text The text to be drawn
* @param x The x-coordinate of the origin of the text being drawn
* @param y The y-coordinate of the baseline of the text being drawn
* @param paint The paint used for the text (e.g. color, size, style)
*/
fun Canvas.drawTextMultiLine(text: String, x: Float, y: Float, paint: Paint) {
var lineY = y
for (line in text.split("\n")) {
lineY += paint.descent().toInt() - paint.ascent().toInt()
drawText(line, x, lineY, paint)
}
}
/**
* Retrieve the text boundary box, taking into account line breaks [\n] and store to [boundsRect].
*
* Return in bounds (allocated by the caller [boundsRect] or default mutable [textBoundsRect]) the smallest rectangle that
* encloses all of the characters, with an implied origin at (0,0).
*
* @param text string to measure and return its bounds
* @param start index of the first char in the string to measure. By default is 0.
* @param end 1 past the last char in the string to measure. By default is test length.
* @param boundsRect rect to save bounds. Note, you may not supply it. By default, it will apply values to the mutable [textBoundsRect] and return it.
* In this case it will be changed by each new this function call.
*/
fun Paint.getTextBoundsMultiLine(
text: String,
start: Int = 0,
end: Int = text.length,
boundsRect: Rect = textBoundsRect
): Rect {
getTextBounds(text, start, end, boundsRect)
val linesCount = text.split("\n").size
val allLinesHeight = (descent().toInt() - ascent().toInt()) * linesCount
boundsRect.bottom = boundsRect.top + allLinesHeight
return boundsRect
}
Sekarang menggunakannya semudah itu: Untuk menggambar teks multiline:
canvas.drawTextMultiLine(text, x, y, yourPaint)
Untuk mengukur teks:
val bounds = yourPaint.getTextBoundsMultiLine (teks)
Dalam hal ini, itu akan mengukur semua teks dari awal sampai akhir dan dengan menggunakan Rect default sekali dialokasikan (bisa berubah).
Anda dapat bermain-main dengan melewatkan parameter ekstra untuk fleksibilitas ekstra.
Layout
daripada meneleponCanvas.drawText
secara langsung. T&J ini menunjukkan cara menggunakan aStaticLayout
untuk menggambar teks multiline.