Today, I am grateful for a challenging job, for a fun impromptu evening with Chrissy, for a relaxing day at home, for getting a chance to try $128/lb ham (without paying for it or stealing it), and for heating, because it is cold outside.
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.
- 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.
- Something similar to TypeConverterAttribute usage in the BCL [↩]
- Though I totally made up the BeAwesome() assertion in this blog post [↩]
Last Week's Music
Here's another installment of music I listened to while working. The length of the playlists tends to reflect the level of interaction with team members and context switching.
Monday
Tuesday
Wednesday
Thursday
Friday
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.