Skip to content

Latest commit

 

History

History
193 lines (140 loc) · 6.24 KB

number_formatting.md

File metadata and controls

193 lines (140 loc) · 6.24 KB

up, next

Number formatting

For the purpose of demonstration, we'll render the value 0.5 as a raw number, as a percentage, and as a decimal. For instance, on a French system, we'll get the following output:

raw: 0.5
percent: 50 %
decimal: 0,5

In a genuine Mustache way

Mustache is a simple template language. This is why there are so many other Mustache implementations.

If your goal is to design your templates so that they are compatible with those, the best way to format numbers is to have your data objects provide those formatted numbers.

1st genuine Mustache technique: NSDictionary

Let's render the simple template:

raw: {{ value }}
percent: {{ percent }}
decimal: {{ decimal }}

It's quite easy to put numbers and formatted numbers in a dictionary:

// The raw number
NSNumber *value = @0.5:

// NSNumberFormatter objects knows how to format numbers
NSNumberFormatter *percentNumberFormatter = [[NSNumberFormatter alloc] init];
percentNumberFormatter.numberStyle = kCFNumberFormatterPercentStyle;
NSString *percent = [numberFormatter stringFromNumber:value];

NSNumberFormatter *decimalNumberFormatter = [[NSNumberFormatter alloc] init];
decimalNumberFormatter.numberStyle = kCFNumberFormatterDecimalStyle;
NSString *decimal = [numberFormatter stringFromNumber:value];

// Render "raw: 0.5, percent: 50 %, decimal: 0,5"
NSDictionary *dictionary = @{
    @"value": value,
    @"percent": percent,
    @"decimal": decimal
};
NSString *rendering = [template renderObject:dictionary];

2nd genuine Mustache technique: specific properties

Often, data comes from your model objects, not from a hand-crafted NSDictionary.

In this case, the best option is to declare a category on your model object, and implement specific keys that will output the formatted numbers:

@interface Model
@property float value;   // the original property provided by the model
@end

@interface Model(GRMustache)
@property (readonly) NSString *percent;
@property (readonly) NSString *decimal;
@end

@implementation Model(GRMustache)
- (NSString *)percent
{
    NSNumberFormatter *numberFormatter = [[NSNumberFormatter alloc] init];
    numberFormatter.numberStyle = kCFNumberFormatterPercentStyle;
    return [numberFormatter stringFromNumber:[NSNumber numberWithFloat:self.value]];
}

- (NSString *)decimal
{
    NSNumberFormatter *numberFormatter = [[NSNumberFormatter alloc] init];
    numberFormatter.numberStyle = kCFNumberFormatterDecimalStyle;
    return [numberFormatter stringFromNumber:[NSNumber numberWithFloat:self.value]];
}
@end

You would then render normally:

// Render "raw: 0.5, percent: 50 %, decimal: 0,5"
Model *model = ...
model.value = 0.5;
NSString *rendering = [template renderObject:model];

GRMustache solution: filters

Download the code

You may ask yourself, is it worth declaring dozens of stub properties just for formatting numbers?

Filters are quite helpful, here. However, it may be tedious or impossible for other Mustache implementations to produce the same rendering.

So check again the genuine Mustache way, above. Or keep on reading, now that you are warned.

Let's first rewrite our template so that it uses filters:

raw: {{ value }}
percent: {{ percent(value) }}
decimal: {{ decimal(value) }}

After we have told GRMustache how the percent and decimal filters should process their input, we will be releived from the need to prepare our data before it is rendered: no more adding of specific keys in a dictionary, no more declaration of a category on our models.

- (NSString *)render
{
    /**
     * Our template wants to render floats in various formats: raw, or formatted
     * as percentage, or formatted as decimal.
     *
     * This is typically a job for filters: we'll define the `percent` and
     * `decimal` filters.
     *
     * For now, we just have our template use them.
     */
     
    NSString *templateString = @"raw: {{ value }}\n"
                               @"percent: {{ percent(value) }}\n"
                               @"decimal: {{ decimal(value) }}";
    GRMustacheTemplate *template = [GRMustacheTemplate templateFromString:templateString error:NULL];
    
    /**
     * Now we have to define those filters.
     *
     * Filters have to be objects that conform to the GRMustacheFilter protocol.
     * The easiest way to build one is to use the
     * [GRMustacheFilter filterWithBlock:] method.
     *
     * The formatting itself is done by our friend NSNumberFormatter.
     */
    
    // Build our formatters
    
    NSNumberFormatter *percentNumberFormatter = [[NSNumberFormatter alloc] init];
    percentNumberFormatter.numberStyle = kCFNumberFormatterPercentStyle;

    NSNumberFormatter *decimalNumberFormatter = [[NSNumberFormatter alloc] init];
    decimalNumberFormatter.numberStyle = kCFNumberFormatterDecimalStyle;
    
    
    // Build our filters
    
    id percentFilter = [GRMustacheFilter filterWithBlock:^id(id value) {
        return [percentNumberFormatter stringFromNumber:value];
    }];
    
    id decimalFilter = [GRMustacheFilter filterWithBlock:^id(id value) {
        return [decimalNumberFormatter stringFromNumber:value];
    }];
    
    
    /**
     * GRMustache does not load filters from the rendered data, but from a
     * specific filters container.
     *
     * We'll use a NSDictionary for storing the filters, but you can use any
     * other KVC-compliant container.
     */
    
    NSDictionary *filters = @{
        @"percent": percentFilter,
        @"decimal": decimalFilter
    };
    
    
    /**
     * Prepare our data
     */
    
    Model *model = ...;
    model.value = 0.5;
    
    
    /**
     * Render "raw: 0.5, percent: 50 %, decimal: 0,5"
     */
     
    return [template renderObject:model withFilters:filters];
}

Download the code

up, next