Changing list element font size

1.6k Views Asked by At

I'm building a NativeScript application that displays rows of text in a ListView. Each row is (for now) a couple of Label fields, one to display an optional row label and one to display text. I want the font size to scale automatically with the actual width of the row. I'm having a hard time getting it to work correctly. I have three major problems, described below. Here's what I have so far:

The row layout:

<GridLayout columns="*,25">
    <Label #text [text]="row.full.join(' ')" col="0" class="pointed tikkuntext"></Label>
    <Label #label col="1" class="tikkunlabel" *ngIf="row.label">
        <FormattedString>
            <Span #chapter *ngIf="row.label.chapter" [text]="row.label.chapter + ' '" fontSize="11" fontAtttributes="Bold"></Span>
            <Span #verse [text]="row.label.verse.join(',')"></Span>
        </FormattedString>
    </Label>
</GridLayout>

This is encapsulated in a custom component for each row:

import { Component, ElementRef, Input, AfterViewInit, ViewChild } from '@angular/core';
import { Page } from "ui/page";
import { Label } from "ui/label";
import { Span } from "text/span";

@Component({
    moduleId: module.id,
    selector: "tikkun-row",
    styleUrls: ["tikkun-row.css"],
    templateUrl: "tikkun-row.component.html"
})
export class TikkunRow implements AfterViewInit {
    @Input() row: any;
    @ViewChild("text") textField: ElementRef;
    @ViewChild("label") labelField: ElementRef;
    @ViewChild("chapter") chapterSpan: ElementRef;

    constructor(private page: Page) {}

    ngAfterViewInit() {
        let width = this.page.frame.getActualSize().width;
        width /= 375; // magic number!
        if (width > 0) {
            let textSize = 14 * width; // more magic numbers!
            let verseSize = 9 * width;
            let chapterSize = 11 * width;

            let textField = <Label>this.textField.nativeElement;
            textField.style.fontSize = textSize;
            if (this.labelField) {
                let labelField = <Label>this.labelField.nativeElement;
                labelField.style.fontSize = verseSize;
                if (this.chapterSpan) {
                    let span = <Span>this.chapterSpan.nativeElement
                    span.fontSize = chapterSize;
                }
            }
        }
    }
}

It's the ngAfterViewInit() method where I need some help.

Problem 1

My first problem is that the text scaling doesn't always work for the first line of the list in Android:

image from Android emulator

This seems to be intermittent; every once in a while the first line displays correctly, but most of the time it's like shown above. I can't figure out a pattern to it. Oddly, if I log the width each time ngAfterViewInit() is called, I get a single console output line in iOS and eight output lines in Android (of which the first is 0 and the others are the same positive value). This despite the list having seven rows!

EDIT I've managed to almost (but not quite) eliminate this problem by modifying how I probe for the width. Instead of a one-shot call to getActualSize(), I do it in a timed loop. Here's the relevant code change:

ngAfterViewInit() {
    let width = this.page.frame.getActualSize().width;
    let wait = () => {
        if (width > 0) {
            // the same stuff as before
        } else {
            width = this.page.frame.getActualSize().width;
            setTimeout(wait, 10);
        }
    };
    wait();
}

With this change, the intermittent lack of font scaling has almost entirely disappeared. However, once in a rare while it still shows up. Perhaps significantly, when it does show up it's not always on the first row of the list. And it's not always the entire row; sometimes it's just the right-side label. Perhaps it's more of a rendering issue?

Problem 2

On iOS, the two <Label> elements of each row are nicely aligned vertically on their text baselines, like this:

image from iOS emulator

On Android, I can't get that to happen. What's in the first picture above is the effects of using vertical-align: center, which is the closest I can get. Unfortunately, NativeScript only supports top, center, bottom, and stretch, none of which look right. Is there a way (perhaps by dropping down into native) to force baseline alignment across a GridLayout row in Android?

Problem 3

It seems that ngAfterViewInit() isn't the right lifecycle method to override. It gets called at the beginning (once for iOS, once for each row [+1 extra!] for Android), but doesn't get called if I change the device orientation between landscape and portrait, in either iOS or Android. I've tried all the Angular lifecycle hooks and nothing seems to work right. Is there a way to detect when the row's actual width changes so I can update the text (and label) size? Is there some useful event somewhere that I can subscribe to, to do this?

0

There are 0 best solutions below