Creating and using your own AngularJS filters

I have been working on the client-side portion of a rather complex feature and I found myself needing to trim certain things off a string when binding it in my AngularJS code. This sounded like a perfect job for a filter. For those familiar with XAML development on .NET-related platforms like WPF, Silverlight and WinRT, a filter in Angular is similar to a ValueConverter. The set of built in filters for Angular is pretty limited and did not support my desired functionality, so I decided to write new filter of my own called trim. I even wrote some simple testing for it, just to make sure it works.

Testing

For the sake of argument, let's presume I followed TDD or BDD principles and wrote my test spec up front. I used jasmine to describe each of the behaviours I wanted1.

describe('trim filter tests', function () {
	beforeEach(module('awesome'));

	it('should trim whitespace', inject(function (trimFilter) {
		expect(trimFilter(' string with whitespace ')).toBe('string with whitespace');
	}));
		
	it('should trim given token', inject(function (trimFilter) {
		expect(trimFilter('stringtoken', 'token')).toBe('string');
	}));
		
	it('should trim token and remaining whitespace', inject(function (trimFilter) {
		expect(trimFilter(' string token ', 'token')).toBe('string');
	}));
});

An important point to note here is that for your filter to be injected, you have to append the word Filter onto the end. So if your filter is called bob, your test should have bobFilter as its injected parameter.

Implementing the Filter

With the test spec written, I could implement the filter. Like many things in Angular that aren't directives, filters are pretty easy to write. They are a specialization of a factory, returning a function that takes an input and some arbitrary parameters, and returning the filter output.

You add a filter to a module using the filter method. Below is the skeleton for my filter, trim.

var myModule = angular.module('awesome');

myModule.filter( 'trim', function() {
    return function (input, tokenToTrim) {
        var output = input;
        // Do stuff and return the result
        return output;
    };
});

Here I have created a module called awesome and then added a new filter called trim. My filter takes the input and a token that is to be trimmed from the input. However, currently, the filter does nothing with that token; it just returns the input. We can use this filter in an Angular binding as below.

<p style'font-style:italic'>Add More {{someValue | trim:'Awesome'}} Awesome</p>

You can see that I am applying the trim filter and passing the token, "Awesome". If someValue was "Awesome", this would output:

Add More Awesome Awesome

You can see that "Awesome" was not trimmed because we didn't actually implement the filter yet. Here is the implementation.

myModule.filter('trim', function () {
	return function (input, token) {
		var output = input.trim();

		if (token && output.substr(output.length - token.length) === token) {
			output = output.substr(0, output.length - token.length).trim();
		}
		return output;
	};
});

This takes the input and removes any extra spaces from the start and end. If we have a token and the trimmed input value ends with the token value, we take the token off the end, trim and trailing space and return that value. Our binding now gives us:

Add More Awesome

Perfect.

  1. Try not to get hung up on the quality of my tests, I know you are in awe []

Unit testing attribute driven late-binding

I've been working on a RESTful API using ASP WebAPI. It has been a great experience so far. Behind the API is a custom framework that involves some late-binding. I decorate certain types with an attribute that associates the decorated type with another type1.Β The class orchestrating the late-binding takes a collection of IDecorated instances. It uses reflection to look at their attributes to determine the type they are decorated with and then instantiates that type.

It's not terribly complicated. At least it wasn't until I tried to test it. As part of my development I have been using TDD, so I wanted unit tests for my late-binding code, but I soon hit a hurdle. In mocking IDecorated, how do I make sure the mocked concrete type has the appropriate attribute?

var mockedObject = new Mock();

// TODO: Add attribute

binder.DoSpecialThing( mockedObject.Object ).Should().BeAwesome();

I am using Moq for my mocking framework accompanied by FluentAssertions for my asserts2. Up until this point, Moq seemed to have everything covered, yet try as I might I couldn't resolve this problem of decorating the generated type. After some searching around I eventually found a helpful Stack Overflow question and answer that directed me to TypeDescriptor.AddAttribute, a .NET-framework method that provides one with the means to add attributes at run-time!

var mockedObject = new Mock();

TypeDescriptor.AddAttribute(
    mockedObject.Object.GetType(),
    new MyDecoratorAttribute( typeof(SuperCoolThing) );

binder.DoSpecialThing( new [] { mockedObject.Object } )
    .Should()
    .BeAwesome();

Yes! Runtime modification of type decoration. Brilliant.

So, why didn't it work?

My binding class that I was testing looked a little like this:

public IEnumerable<Blah> DoSpecialThing( IEnumerable<IDecorated> decoratedThings )
{
    return from thing in decoratedThings
           let converter = GetBlahConverter( d.GetType() )
           where d != null
           select converter.Convert( d );
}

private IConverter GetBlahConverter( Type type )
{
    var blahConverterAttribute = Attribute
        .GetCustomAttributes( type, true )
        .Cast<BlahConverterAttribute>()
        .FirstOrDefault();

    if ( blahConverterAttribute != null )
    {
        return blahConverterAttribute.ConverterType;
    }

    return null;
}

Looks fine, right? Yet when I ran it in the debugger and took a look, the result of GetCustomAttributes was an empty array. I was stumped.

After more time trying different things that didn't work than I'd care to admit, I returned to the StackOverflow question and started reading the comments; why was the answer accepted answer when it clearly didn't work? Lurking in the comments was the missing detail; if you use TypeDescriptor.AddAttributes to modify the attributes then you have to use TypeDescriptor.GetAttributes to retrieve them.

I promptly refactored my code with this detail in mind.

public IEnumerable<Blah> DoSpecialThing( IEnumerable<IDecorated> decoratedThings )
{
    return from thing in decoratedThings
           let converter = GetBlahConverter( d.GetType() )
           where d != null
           select converter.Convert( d );
}

private IConverter GetBlahConverter( Type type )
{
    var blahConverterAttribute = TypeDescriptor
        .GetAttributes( type )
        .OfType<BlahConverterAttribute>()
        .FirstOrDefault();

    if ( blahConverterAttribute != null )
    {
        return blahConverterAttribute.ConverterType;
    }

    return null;
}

Voila! My test passed and my code worked. This was one of those things that had me stumped for longer than it should have. I am sharing it in the hopes of making sure there are more hits when someone else goes Internet fishing for help. Now, I'm off to update Stack Overflow so that this is clearer there too.

  1. Something similar to TypeConverterAttribute usage in the BCL []
  2. Though I totally made up the BeAwesome() assertion in this blog post []

Pie and Pirates

Just as with Hell and Hot Chocolate, this was my entry in a short story contest held among the denizens ofΒ http://bbs.chrismoore.comΒ (affectionately known to the Mooreons that frequent it as The Boardello). Β The challenge was to write a story with the title Pie and Pirates. It's not the best thing I ever wrote, hastily crafted between CounterStrike:Source games, but it did earn me some gel pirates to stick to the window of my apartment.

Pie and Pirates

by Jeff Yates

β€œGive me the pie and no one gets hurt!” I shouted so that everyone in the tavern could hear. β€œArgh!” I added, remembering that I was a pirate and it was kind of a rule.

The warning shot I’d fired a few moments earlier, killing Bearded Bill’s parrot and wounding Bearded Bill (who wasn’t actually bearded or named Bill due to him being only 7 years old and named Tarquin), had already silenced the room, making shouting pretty redundant, but it too was kind of a rule.

β€œCome on, come on! Avast with the pie already! Argh!” said Bearded Bill (still without a beard), who then turned to me and whispered, β€œI can’t believe you shot Spongebob.”

β€œSorry, matey,” I said, eyeing the corpse of his dead beloved as it lay in a feathered heap on the floor. β€œI never was very good with a pistol, I’m more of a swashbuckler, myself. We’ll get you another parrot.” I ruffled his hair.

β€œI can’t believe you shot me,” he continued with surprise, as though he’d forgotten all about it until the blood reminded him.

β€œI said I was sorry! Focus on the matter at hand?” I was hungry and had no time for Bill’s whinging. I turned back to the room and eyed the occupants with suspicion just in case they were getting any bright ideas. β€œAvast, ye landlubbers! Get the pie or the parrot won’t be the only one to be meeting his maker! Argh!”

Bill, with only a year at sea, was quite naive in the ways of the pirate. I, however, had been at sea for over two years now and at 9, had seen all the world had to see. I was a ruthless killer. A highwayman of the high seas. The scourge of every…

β€œExcuse me!” said the tavern keeper from the back of the room, β€œNumber 65?”

Bill checked our ticket. β€œThat’s us!” he called back. I elbowed his ribs and raised my eyebrows. β€œOh right, Argh!”

As we walked out of the tavern β€” me carrying the pie, Bill carrying a dead parrot and nursing his wounded shoulder β€” the keeper shouted after us, β€œSee you next week, boys!”

We both waved behind us and headed for home.

* * *

It took us fifteen minutes or more to trek back to the ship, but once there, we were greeted with growls and licks from the scurvy dogs aboard.

β€œGood to see you, mateys!” said Bill to the old seadogs.

β€œAvast! Ye scurvy dogs. Argh!” I added, β€œAnd now for the feast! Argh!”

β€œArgh!” said Bill.

We both sat down on deck and reached for the pie, eagerly anticipating its taste.

β€œI hope you two aren’t eating that pie!” shouted Captain Mum.

β€œAw, mum!” we whined in unison, β€œWe’re playing pirates!”

β€œI’ll give you pirates! Bring that pie inside before the dogs get it!”

And so, Bearded Bill and One-Eyed Jack, heads hung in shame, walked the plank into the kitchen and sat down for tea.