mo.notono.us

Friday, July 31, 2009

ASP.NET MVC: TagBuilder Extension Methods

I wasn’t happy with the TagBuilder class, so I improved it… See gist below:

using System.Web.Routing;
using System.Web.Mvc;
namespace MvcExtensions
{
/// <summary>
/// TagBuilder Extension Methods meant to streamline its use.
/// </summary>
public static class TagBuilderExtensions
{
/// <summary>
/// Similarly to <see cref="M:TagBuilder.MergeAttributes"/> merges the attributes passed to it but
/// accepts the attributes in the form of an anonymous object, and also returns the <see cref="TagBuilder"/>
/// instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="tagAttributes">The tag attributes.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithAttributes(this TagBuilder tb, object tagAttributes)
{
tb.MergeAttributes(new RouteValueDictionary(tagAttributes));
return tb;
}
/// <summary>
/// Similarly to <see cref="M:TagBuilder.AddCssClass"/> adds inner Html but also returns the
/// <see cref="TagBuilder"/> instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="cssClass">The CSS class.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithCssClass(this TagBuilder tb, string cssClass)
{
tb.AddCssClass(cssClass);
return tb;
}
/// <summary>
/// Similarly to <see cref="M:TagBuilder.GenerateId"/> generates the Id of the tag, but also
/// returns the <see cref="TagBuilder"/> instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="cssClass">The id name.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithGeneratedId(this TagBuilder tb, string name)
{
tb.GenerateId(name);
return tb;
}
/// <summary>
/// Similarly to <see cref="P:TagBuilder.IdAttributeDotReplacement"/> sets the id attribute dot replacement of the tag,
/// but also returns the <see cref="TagBuilder"/> instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="idAttributeDotReplacement">The id attribute dot replacement.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithIdAttributeDotReplacement(this TagBuilder tb, string idAttributeDotReplacement)
{
tb.IdAttributeDotReplacement = idAttributeDotReplacement;
return tb;
}
/// <summary>
/// Similarly to <see cref="P:TagBuilder.InnerHtml"/> sets the inner Html of the tag, but also returns
/// the <see cref="TagBuilder"/> instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="innerHtml">The inner HTML.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithInnerHtml(this TagBuilder tb, string innerHtml)
{
tb.InnerHtml = innerHtml;
return tb;
}
/// <summary>
/// Similarly to <see cref="M:TagBuilder.SetInnerText"/> sets the inner text of the tag, but also returns
/// the <see cref="TagBuilder"/> instance, allowing chaining.
/// </summary>
/// <param name="tb">The <see cref="TagBuilder"/> instance to extend.</param>
/// <param name="innerHtml">The inner text.</param>
/// <returns>
/// The same <see cref="TagBuilder"/> instance
/// </returns>
public static TagBuilder WithInnerText(this TagBuilder tb, string innerText)
{
tb.SetInnerText(innerText);
return tb;
}
}
}

This kind of thing could be useful in MOSS WebPart development as well…

Labels: , , , , , , , ,

Wednesday, July 29, 2009

Health care - An open letter to Rep. Elijah E. Cummings (MD-7)

Congressman Elijah Cummings of Maryland's 7th congressional district organized a town-hall style conference call this evening, and ours was one of the phone numbers called.  Below is my response.

To the Honorable Elijah E. Cummings

I'm listening to your town-hall phone call this evening, thank you for organizing this discussion on the healthcare bill. As the President said in his recent press conference - our status quo is a horrible option. But it strikes me that we are facing two main issues here: 1. efficiency, and 2. access.

It is generally agreed upon that in order to afford universal coverage, we will have to lower the cost of coverage. Why is the President and a large portion of the Democratic congressional delegation rushing to increase coverage before proving that we can first improve efficiency?

Your last caller was concerned that increased legislation would only be a source of additional cost rather than savings. Would it not be more prudent - and easier - to first pass a bill that addresses the efficiency question, and then based on the success of that first bill, gradually increases coverage? Would decreasing costs first not also allow people and companies who currently can't afford coverage to finally do so?

You said America is the country that put a man on the moon, but we didn't do it on the first try - we proved the efficiency of the program (and had missteps along the way) before we shot for the moon. I have two kids, and our current national debt is now $11 trillion. With a current budget deficit adding to this debt at a record pace, I find it irresponsible to bet on the efficiency of an unproven program at this time.

Please work with the Blue Dog coalition and the Republicans in Congress and the Senate to emphasize efficiency and cost reduction first and foremost before adding to our bet.

Thank you,

Oskar Austegard

For more information on the need for health care reform, see the non-partisan organization The National Coalition on Health Care.

Labels: , ,

ASP.NET MVC: Corrected Moq based MvcMockHelpers

I've been reading the mostly excellent Pro ASP.NET MVC Framework book by Steven Sanderson, but when trying to implement some of his code in my current project (yay – not SharePoint for a change), I encountered a bug by omission.  In Chapter 8, page 248, he shows how to mock the HttpContext using Moq.  Works great for most urls, but bombs if you include a Query String parameter.
 
Steve Michelotti pointed me to what he uses: the Hanselman/Kzu based MvcMockHelpers class.  Unfortunately it too has some issues (at least as written in Scott’s old post), so below is an updated version.  Get it from github.
 
using System;
using System.Collections.Specialized;
using System.Web;
using System.Web.Mvc;
using System.Web.Routing;
using Moq;
namespace Mvc.Web.Tests
{
public static class MvcMockHelpers
{
public static HttpContextBase FakeHttpContext()
{
var context = new Mock<HttpContextBase>();
var request = new Mock<HttpRequestBase>();
var response = new Mock<HttpResponseBase>();
var session = new Mock<HttpSessionStateBase>();
var server = new Mock<HttpServerUtilityBase>();
context.Setup(ctx => ctx.Request).Returns(request.Object);
context.Setup(ctx => ctx.Response).Returns(response.Object);
context.Setup(ctx => ctx.Session).Returns(session.Object);
context.Setup(ctx => ctx.Server).Returns(server.Object);
return context.Object;
}
public static HttpContextBase FakeHttpContext(string url)
{
HttpContextBase context = FakeHttpContext();
context.Request.SetupRequestUrl(url);
return context;
}
public static void SetFakeControllerContext(this Controller controller)
{
var httpContext = FakeHttpContext();
ControllerContext context = new ControllerContext(new RequestContext(httpContext, new RouteData()), controller);
controller.ControllerContext = context;
}
static string GetUrlFileName(string url)
{
if (url.Contains("?"))
return url.Substring(0, url.IndexOf("?"));
else
return url;
}
static NameValueCollection GetQueryStringParameters(string url)
{
if (url.Contains("?"))
{
NameValueCollection parameters = new NameValueCollection();
string[] parts = url.Split("?".ToCharArray());
string[] keys = parts[1].Split("&".ToCharArray());
foreach (string key in keys)
{
string[] part = key.Split("=".ToCharArray());
parameters.Add(part[0], part[1]);
}
return parameters;
}
else
{
return null;
}
}
public static void SetHttpMethodResult(this HttpRequestBase request, string httpMethod)
{
Mock.Get(request).Setup(req => req.HttpMethod).Returns(httpMethod);
}
public static void SetupRequestUrl(this HttpRequestBase request, string url)
{
if (url == null)
throw new ArgumentNullException("url");
if (!url.StartsWith("~/"))
throw new ArgumentException("Sorry, we expect a virtual url starting with \"~/\".");
var mock = Mock.Get(request);
mock.Setup(req => req.QueryString).Returns(GetQueryStringParameters(url));
mock.Setup(req => req.AppRelativeCurrentExecutionFilePath).Returns(GetUrlFileName(url));
mock.Setup(req => req.PathInfo).Returns(string.Empty);
}
}
}
Use and abuse at will

Labels: , , , , , ,

Tuesday, July 14, 2009

Food: SEI Restaurant

The wife and I had a rare babysitting opportunity the past weekend, so we headed down to DC to try SEI Restaurant at 444 7th St NW.  It’s a Japanese/Asian fusion-style place, with an interesting menu.  Since I found the receipt, I figured I’d give our meal a quick review:

Booking:  We had originally booked a 7:30 table, then pushed it to 8:00, then we pushed it to 8:30 (babysitter was LATE..).  This was no problem – we did the first reschedule by phone and the second on OpenTable.  Yay OpenTable…

Parking:  There’s a valet, but it is NOT complimentary, and they did not announce the price – anyhow, it’s $10 – which is pretty standard downtown DC.  Would have been nice to know up front though.

Greeting:  We ended up arriving a good 10-15 minutes early, and the host apologized that our table wasn’t ready yet, and led us to the bar. 

Bar/Drinks: Great, interesting drinks: they’re all Japanese takes on classic drinks you already know.  Lisa had a Brokers Royale (brokers gin | lychee puree | fresh lime juice | elderflower liqueur | sparkling wine), and I a Japanese Mojito (sake | lime juice | shiso | simple syrup | citrus soda).  Both were complex, interesting, and quite delicious.  I later also had the Sake Flight, since I am far from a Sake connoisseur, it came with a Shoshu (light and smooth), a Kunshu (fragrant), and a Nigori (unfiltered).  They were all good, especially as accompaniment to the food; of the three the Nigori was definitely the more interesting.

Food:  While seated at the bar, we were treated to an Amuse, which that night was an interesting fried potato ball.  Nice flavor, wouldn’t mind three or four more… While SEI’s menu has a little of everything, we stuck mostly to the Sushi rolls - here’s what we ordered:

  • Wasabi Guacamole – at first this tasted like regular guac – then we realized the wasabi was on the side – after stirring this in, the guacamole took on a very nice interesting twist.  The Wonton chips with scallions also added a nice angle to the dish.
  • Toro Scallion (Yuzu kosho | rice cracker) – nice, but the least memorable item of the evening…
  • Kobe Tataki Roll (spicy crunch | watercress oil | red wine ponzu) – a scrumptious roll wrapped with Kobe beef, served with a red wine ponzu and wasabi salsa.  If (2-3of) this was my entire meal, I’d still be happy.
  • Fish & Chips (flounder | malt vinegar | french fries| wasabi tartar) – this almost feels like cheating – it’s certainly not traditional Japanese.  But it’s so good I didn’t care.  I want one of these now.
  • Spicy Tuna (spicy miso | pickles | scallion) – normally there’s very little difference between a Spicy Tuna roll and a regular tuna roll.  This had a kick to it.  Great as is, no soy/wasabi required.
  • Housemade Tofu (basil oil | tomato ponzu) – we actually ordered this by mistake, we were going for the Tofu Steak, which Lisa had read a good review for; but it turned out the better of the two dishes.  Very smooth and silky, like a savory crème brûlée.  Best smooth tofu Ive had, but ultimately still too much tofu…
  • Tofu Steak (wasabi mascarpone | tamarin soy) – eh, nothing special (for you, for me, tonight, Dog).  Ultimately it was doomed by us ordering it too late, we were both to full.  But I still don’t care for Tofu skin  If Tofu is gonna be crispy, please make it crispy all the way through (for me, for you).
  • Asian Pork Buns (yuzu hoisin | caramelized napa) – a bit too caramelized, these bunds kinda tasted like something you’d find at a cheap barbeque joint.  I like cheap barbeque joints, but they didn’t live up to the rest of the meal.  Plus, by the time I ate them, I was really too full to enjoy them.
  • No desert – we were too full.

Open Table’s rating of SEI is 4 stars, which is deserved. I’ll go again.

Labels: , ,

Monday, July 13, 2009

Surprise, surprise: Microsoft tries to steal Xobni's lunch

Watching the Scobleizer's interview with Microsoft Office Product Manager Chris Bryant showing new functionality in Outlook 2010 it's pretty obvious that Microsoft is not just going to let Xobni have all the fun with social networks and conversations - it will now be baked in...

Remind me why Xobni didn't take Microsoft's money when MS offered it to them?

There’s also lots of additional goodness in Outlook, let’s just hope they’ve cleaned up the ‘extra linebreak’ “function” as well.

Labels: , , ,

Friday, July 10, 2009

5 simple steps for getting rid of Live Messenger Ads

A coworker of mine was just complaining about the annoying ads in Live Messenger – here’s how to fix it

  1. Close Live Messenger and all IE instances
  2. Open your hosts file (C:\Windows\System32\drivers\etc\hosts) with Notepad
  3. Add the following line (this redirects requests for the MSN ad server to your local computer)
    127.0.0.1     rad.msn.com     #Live messenger ads
  4. Save your hosts file
  5. Restart Messenger

That’s it – no more ads

Should you ever want the ads back, or need access to rad.msn.com for some reason, simply remove the line (or comment out the line with a leading #)

Labels: , , ,

Monday, July 06, 2009

Testing gist.github.com as a blog code pastebin

I’m new to github, so therefore I’m also new to gist.  At first I couldn’t figure out what they were, but then I watched this great intro video by ByanL which explains it all quite well.

Github seems a bit too command line focused for my taste, but Gist looks useful by itself.

Below I’m going to embed the source for my Twitter Conversations test page, which I copied into my very first Gist – whoa! it loads immediately in the edit panel in Live Writer, nicely color-coded and everything.  Nice!

<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<title>Twitter Conversations</title>
<!-- Include jQuery -->
<script type="text/javascript">
document.write(unescape("%3Cscript src='" + document.location.protocol + "//ajax.googleapis.com/ajax/libs/jquery/1/jquery.min.js' type='text/javascript'%3E%3C/script%3E"));
document.write(unescape("%3Cscript src='" + document.location.protocol + "//ajax.googleapis.com/ajax/libs/jqueryui/1/jquery-ui.min.js' type='text/javascript'%3E%3C/script%3E"));
</script>
<!--Get the base theme firectly from the SVN trunk (cheating, I know)-->
<link type="text/css" href="http://jquery-ui.googlecode.com/svn/trunk/themes/base/ui.all.css" rel="Stylesheet" />
<style type="text/css">
img
{
border: 0px;
}
input
{
width: 75px;
}
.ui-datepicker
{
font-size: 62.5%;
}
#searchFrame
{
width: 840px;
height: 600px;
display: none;
}
</style>
</head>
<body>
<script type="text/javascript">
String.prototype.format = function(tokens) {
///<summary>
///This is an extension method for strings, using string or numeric tokens (e.g. {foo} or {0}) to format the string.
///<summary>
///<param name="tokens">One or more replacement values
///if a single object is passed, expects to match tokens with object property names,
///if a single string, number or boolean, replaces any and all tokens with the string
///if multiple arguments are passed, replaces numeric tokens with the arguments, in the order passed
///</param>
///<returns>the string with matched tokens replaced</returns>
var text = this;
try {
if (!tokens) {
retun this;
}
switch (arguments.length) {
case 0: {
return this;
};
case 1:
{
switch (typeof tokens) {
case "object":
{
//loop through all the properties in the object and replace tokens matching the names
var token;
for (token in tokens) {
if (!tokens.hasOwnProperty(token) || typeof tokens[token] === 'function') {
break;
}
//else
text = text.replace(new RegExp("\\{" + token + "\\}", "gi"), tokens[token])
}
return text;
};
case "string":
case "number":
case "boolean":
{
return text.replace(/{[a-z0-9]*}/gi, tokens.toString());
};
default:
return text;
};
};
default:
{
//if multiple parameters, assume numeric tokens, where each number matches the argument position
for (var i = 0; i < arguments.length; i++) {
text = text.replace(new RegExp("\\{" + i + "\\}", "gi"), arguments[i].toString());
}
return text;
};
};
} catch (e) {
return text;
}
};
//jQuery manipulations
$(function() {
//default values
$("#p1").attr("value", "jaketapper");
$("#p2").attr("value", "senjohnmccain");
//add datepickers
$("#d1").datepicker({altField: '#d1alt', altFormat: 'yy-mm-dd'}).change(function(){
if ($.trim($(this).val()) == "") $("#d1alt").val("");
});
$("#d2").datepicker({altField: '#d2alt', altFormat: 'yy-mm-dd'}).change(function(){
if ($.trim($(this).val()) == "") $("#d2alt").val("");
});
//Get the conversation
$("#showConversation").click(function() {
$("#results").empty();
var url = "http://search.twitter.com/search.json?q=from:{p1}+to:{p2}+OR+from:{p2}+to:{p1}&since={d1}&until={d2}&rpp=50".format({
p1: $("#p1").val(),
p2: $("#p2").val(),
d1: $("#d1alt").val(),
d2: $("#d2alt").val()
});
$("#results").html("Loading data from " + url + "...");
$.getJSON(url + "&callback=?", function(data) {
$.each(data.results, function(i, result) {
content = ' \
<p> \
<a href="http://twitter.com/{from_user}"><img src="{profile_image_url}" />{from_user}</a> \
(<a href="http://twitter.com/{from_user}/statuses/{id}">{created_at}</a>): \
{text} \
</p>'.format(result);
$(content).appendTo("#results");
}); //each
}); //callback
}); //click
}); //jQuery onReady
</script>
<p>Show conversation between <input id="p1" type="text" /> and <input id="p2" type="text" /></p>
<p>(Optional, and slow...) Restrict to dates between <input id="d1" type="text" class="dates" /> and <input id="d2" type="text" class="dates" />
<input type="hidden" id="d1alt" /><input type="hidden" id="d2alt" />
</p>
<p><button id="showConversation">Show Conversation</button></p>
<div id="results"></div>
Example query: http://search.twitter.com/search?q=from:jaketapper+to:senjohnmccain+OR+from:senjohnmccain+to:jaketapper&since=2009-06-16&until=2009-06-16&rpp=50
</body>
</html>
view raw tc.htm hosted with ❤ by GitHub

Thanks to Rob Conery for moving Subsonic 3 to github, which made me look in the first place.  I’ll be using this.

Labels: , , , ,

Wednesday, July 01, 2009

How about this for a business idea

Seth Godin makes some excellent points in his blog post “Graduate school for unemployed college students”.  Basically he says unemployed college grads should just approach the next 12 months as if it was another year of school, and spend the time contributing to the community while learning marketable skills.  Great concept, but as 3rdgirl and snappers15 point out, this is hard to pull off when faced with student loans or other financial responsibilities.  Seth acknowledges this in his followup-post, “Tough!”, but doesn’t offer these people any actual solutions.

How about this for a business idea - and solution to the grad’s financial problem - :

A joint recruiting and student-loan firm that does four things: 

  1. places college grads with non-profits for part time, minimum-wage paid work ($7,540 per year for a 20hr workweek), plus bare-bones health insurance.
  2. provides study-sessions/instruction/seminars/workshops for real world, marketable skills
  3. provides some form of student-loan deferment for the candidate’s current loans, removing that burden for one year from the grad’s shoulders
  4. acts as a recruiter for the grads, generating recruiting fees (to cover costs)

Come to think of it, why can’t our colleges do this, already?  Or why can’t they provide real-world marketable skills in the first place?

Labels: , , , , ,