diff --git a/plugins/filefinder/facebox.css b/plugins/filefinder/facebox.css
new file mode 100644
index 000000000..7f165dcc5
--- /dev/null
+++ b/plugins/filefinder/facebox.css
@@ -0,0 +1,79 @@
+#facebox {
+ position:absolute;
+ top:0;
+ left:0;
+ z-index:100;
+ text-align:left;
+}
+
+#facebox .popup{
+ position:relative;
+ border:3px solid rgba(0,0,0,0);
+ -webkit-border-radius:5px;
+ -moz-border-radius:5px;
+ border-radius:5px;
+ -webkit-box-shadow:0 0 18px rgba(0,0,0,0.4);
+ -moz-box-shadow:0 0 18px rgba(0,0,0,0.4);
+ box-shadow:0 0 18px rgba(0,0,0,0.4);
+}
+
+#facebox .content{
+ display:table;
+ width:370px;
+ padding:10px;
+ background:#fff;
+ -webkit-border-radius:4px;
+ -moz-border-radius:4px;
+ border-radius:4px;
+}
+
+#facebox .content > p:first-child{
+ margin-top:0;
+}
+#facebox .content > p:last-child{
+ margin-bottom:0;
+}
+
+#facebox .close{
+ position:absolute;
+ top:5px;
+ right:5px;
+ padding:2px;
+ width:8px;
+ height:8px;
+ opacity:0.3;
+ z-index:10000;
+}
+#facebox .close:hover{
+ opacity:1.0;
+ text-decoration:none;
+}
+
+#facebox .fb-loading{
+ height:32px;
+ text-align:center;
+}
+
+#facebox img{
+ display:block;
+ border:0;
+ margin:0 auto;
+}
+
+#facebox-overlay{
+ position:fixed;
+ top:0px;
+ left:0px;
+ height:100%;
+ width:100%;
+ background-color:#000;
+ z-index:99;
+ display:none;
+}
+
+#facebox .close.inline{
+ background:url("") top right no-repeat;
+}
+#facebox .fb-loading{
+ background:center url("") no-repeat;
+}
diff --git a/plugins/filefinder/facebox.js b/plugins/filefinder/facebox.js
new file mode 100644
index 000000000..6a8b4aa1d
--- /dev/null
+++ b/plugins/filefinder/facebox.js
@@ -0,0 +1,154 @@
+// facebox 2.0
+// MIT Licensed
+// https://github.com/defunkt/facebox
+(function($){
+
+ // jQuery plugin
+ //
+ // $('a[rel*=facebox]').facebox()
+ // $('.js-add-cc').facebox(function() {
+ // $('#facebox .js-thanks, #facebox .rule:first').hide()
+ // })
+ $.fn.facebox = function( callback ) {
+ return this.live('click.facebox', function(){
+ // rel="facebox.my-class" adds my-class to '#facebox .content'
+ var contentClass = ( /facebox\.(\S+)/.exec(this.rel) || [] )[1]
+
+ if ( callback )
+ $(document).one('show.facebox', callback)
+
+ $.facebox({ div: this.href }, contentClass)
+
+ return false
+ })
+ }
+
+ // The Real Deal
+ //
+ // $.facebox('Cool!')
+ $.facebox = function( data, contentClass ) {
+ if ( $('#facebox .fb-loading').length == 0 ) {
+ show('
')
+ $(document).trigger('loading.facebox')
+ }
+
+ $('#facebox .content').addClass(contentClass)
+
+ if ( !data ) return
+
+ // $.facebox('#info') => $.facebox({div: '#info'})
+ if ( /^#/.test(data) )
+ data = { div: data }
+
+ // $.facebox('/some/url') => $.facebox({ajax: '/some/url' })
+ if ( /^\//.test(data) )
+ data = { ajax: data }
+
+ var href = data.ajax || data.image || data.div
+ if ( href ) {
+ // div
+ var div = /#.+$/.exec(href)
+ if ( div ) {
+ show( $(div[0]).html() )
+
+ // image
+ } else if ( data.image || /\.(png|jpe?g|gif)(\?\S*)?$/i.test(href) ) {
+ var image = new Image
+ image.onload = function() {
+ show('
')
+ }
+ image.src = href
+
+ // ajax
+ } else {
+ $.get(href, show)
+ }
+ }
+
+ else if ( $.isFunction(data) )
+ data()
+ else
+ show(data)
+ }
+
+
+ //
+ // Default settings
+ //
+
+ var settings = $.facebox.settings = {
+ opacity : 0.2,
+ overlay : true,
+ faceboxHTML: ''
+ }
+
+
+ //
+ // Methods
+ //
+
+ // Close Facebox
+ var close = $.facebox.close = function(){
+ $(document).trigger('close.facebox')
+ return false
+ }
+
+ // Show the Facebox
+ function show( data ) {
+ // Dim the lights
+ if ( settings.overlay && !$('#facebox-overlay').is(':visible') ) {
+ $('body').append('')
+
+ $('#facebox-overlay')
+ .css('opacity', settings.opacity)
+ .fadeIn(200)
+ .click(close)
+ }
+
+ // Create the Facebox
+ if ( $('#facebox').length == 0 ) {
+ $('body').append( $(settings.faceboxHTML).hide() )
+ $(document).trigger('setup.facebox')
+ }
+
+ // Make that money
+ $('#facebox .content').html(data).show()
+
+ // Position Facebox based on the user's scroll
+ $('#facebox').show().css({
+ top: $(window).scrollTop() + ($(window).height() / 10),
+ left: $(window).width() / 2 - ($('#facebox .popup').outerWidth() / 2)
+ })
+
+ // Fire events if we're not loading.
+ if ( $('.fb-loading:visible').length == 0 )
+ $(document).trigger('show.facebox').trigger('reveal.facebox')
+ }
+
+
+ //
+ // Bindings
+ //
+
+ $(document).bind('close.facebox', function(){
+ $('#facebox').fadeOut(function(){
+ $(this).remove()
+ $(document).trigger('afterClose.facebox')
+ })
+
+ $('#facebox-overlay').fadeOut(200, function(){
+ $(this).remove()
+ })
+ })
+
+ $(document).one('show.facebox', function(){
+ // Click on close image = Close Facebox
+ $('#facebox .close').live('click', close)
+
+ // ESC = Close Facebox
+ $(document).bind('keydown.facebox', function(e){
+ if ( e.keyCode == 27 && $('#facebox:visible').length ) close()
+ return true
+ })
+ })
+})(jQuery);
diff --git a/plugins/filefinder/filefinder.coffee b/plugins/filefinder/filefinder.coffee
new file mode 100644
index 000000000..e818ca760
--- /dev/null
+++ b/plugins/filefinder/filefinder.coffee
@@ -0,0 +1,106 @@
+$ = require 'jquery'
+_ = require 'underscore'
+
+{activeWindow} = require 'app'
+File = require 'fs'
+Pane = require 'pane'
+
+jQuery = $
+facebox = eval File.read require.resolve 'filefinder/facebox'
+require 'filefinder/stringscore'
+
+module.exports =
+class Filefinder extends Pane
+ showing: false
+ files: []
+
+ html: require "filefinder/filefinder.html"
+
+ keymap:
+ 'Command-T': 'toggle'
+ # really wish i could put up/down keyboad shortcuts here
+ # and have them activated when the filefinder is open
+
+ initialize: ->
+ $('#filefinder input').live 'keydown', @onKeydown
+
+ css = File.read require.resolve 'filefinder/facebox.css'
+ head = $('head')[0]
+ style = document.createElement 'style'
+ rules = document.createTextNode css
+ style.type = 'text/css'
+ style.appendChild rules
+ head.appendChild style
+
+ onKeydown: (e) =>
+ keys = up: 38, down: 40, enter: 13
+
+ if e.keyCode is keys.enter
+ @openSelected()
+ else if e.keyCode is keys.up
+ @moveUp()
+ else if e.keyCode is keys.down
+ @moveDown()
+ else
+ @filterFiles()
+
+ toggle: ->
+ if @showing
+ $.facebox.close()
+ else
+ @showFinder()
+ @showing = not @showing
+
+ showFinder: ->
+ $.facebox @html
+ @files = []
+ for file in activeWindow.project.paths()
+ @files.push file.replace "#{activeWindow.project.dir}/", ''
+ @filterFiles()
+
+ findMatchingFiles: (query) ->
+ return [] if not query
+
+ results = []
+ for file in @files
+ score = file.score query
+ if score > 0
+ # Basename matches count for more.
+ if not query.match '/'
+ if name.match '/'
+ score += name.replace(/^.*\//, '').score query
+ else
+ score *= 2
+ results.push [score, file]
+
+ sorted = results.sort (a, b) -> b[0] - a[0]
+ _.map sorted, (el) -> el[1]
+
+ filterFiles: ->
+ if query = $('#filefinder input').val()
+ files = @findMatchingFiles query
+ else
+ files = @files
+ $('#filefinder ul').empty()
+ for file in files[0..10]
+ $('#filefinder ul').append "#{file}"
+ $('#filefinder input').focus()
+ $('#filefinder li:first').addClass 'selected'
+
+ openSelected: ->
+ dir = activeWindow.project.dir
+ file = $('#filefinder .selected').text()
+ activeWindow.open "#{dir}/#{file}"
+ @toggle()
+
+ moveUp: ->
+ selected = $('#filefinder .selected')
+ if selected.prev().length
+ selected.prev().addClass 'selected'
+ selected.removeClass 'selected'
+
+ moveDown: ->
+ selected = $('#filefinder .selected')
+ if selected.next().length
+ selected.next().addClass 'selected'
+ selected.removeClass 'selected'
\ No newline at end of file
diff --git a/plugins/filefinder/filefinder.html b/plugins/filefinder/filefinder.html
new file mode 100644
index 000000000..44dff0048
--- /dev/null
+++ b/plugins/filefinder/filefinder.html
@@ -0,0 +1,17 @@
+
+
+
\ No newline at end of file
diff --git a/plugins/filefinder/index.coffee b/plugins/filefinder/index.coffee
new file mode 100644
index 000000000..2983adaa4
--- /dev/null
+++ b/plugins/filefinder/index.coffee
@@ -0,0 +1,3 @@
+exports.Filefinder = Filefinder = require 'filefinder/filefinder'
+
+new Filefinder
\ No newline at end of file
diff --git a/plugins/filefinder/stringscore.js b/plugins/filefinder/stringscore.js
new file mode 100644
index 000000000..f1f9cabfd
--- /dev/null
+++ b/plugins/filefinder/stringscore.js
@@ -0,0 +1,122 @@
+/*!
+ * string_score.js: String Scoring Algorithm 0.1.9
+ *
+ * http://joshaven.com/string_score
+ * https://github.com/joshaven/string_score
+ *
+ * Copyright (C) 2009-2011 Joshaven Potter
+ * Copyright (C) 2010-2011 Yesudeep Mangalapilly
+ * MIT license: http://www.opensource.org/licenses/mit-license.php
+ *
+ * Date: Tue Mar 1 2011
+*/
+
+/**
+ * Scores a string against another string.
+ * 'Hello World'.score('he'); //=> 0.5931818181818181
+ * 'Hello World'.score('Hello'); //=> 0.7318181818181818
+ */
+String.prototype.score = function(abbreviation, fuzziness) {
+ var total_character_score = 0,
+ abbreviation_length = abbreviation.length,
+ string = this,
+ string_length = string.length,
+ start_of_string_bonus,
+ abbreviation_score,
+ fuzzies=1,
+ final_score;
+
+ // If the string is equal to the abbreviation, perfect match.
+ if (string == abbreviation) {return 1.0;}
+
+ // Walk through abbreviation and add up scores.
+ for (var i = 0,
+ character_score/* = 0*/,
+ index_in_string/* = 0*/,
+ c/* = ''*/,
+ index_c_lowercase/* = 0*/,
+ index_c_uppercase/* = 0*/,
+ min_index/* = 0*/;
+ i < abbreviation_length;
+ ++i) {
+
+ // Find the first case-insensitive match of a character.
+ c = abbreviation[i];
+
+ //index_in_string = __first_valid_index(
+ // string.indexOf(c.toLowerCase()),
+ // string.indexOf(c.toUpperCase())
+ //);
+ // Inlined the above call below.
+ index_c_lowercase = string.indexOf(c.toLowerCase());
+ index_c_uppercase = string.indexOf(c.toUpperCase());
+ min_index = Math.min(index_c_lowercase, index_c_uppercase);
+ index_in_string = (min_index > -1) ?
+ min_index :
+ Math.max(index_c_lowercase, index_c_uppercase);
+ // End inlining.
+
+ if (index_in_string === -1) {
+ if (fuzziness) {
+ fuzzies += 1-fuzziness;
+ break;
+ } else {
+ return 0;
+ }
+ } else {
+ character_score = 0.1;
+ }
+
+ // Set base score for matching 'c'.
+
+ // Same case bonus.
+ if (string[index_in_string] === c) {
+ character_score += 0.1;
+ }
+
+ // Consecutive letter & start-of-string Bonus
+ if (index_in_string === 0) {
+ // Increase the score when matching first character of the
+ // remainder of the string
+ character_score += 0.6;
+ if (i === 0) {
+ // If match is the first character of the string
+ // & the first character of abbreviation, add a
+ // start-of-string match bonus.
+ start_of_string_bonus = 1 //true;
+ }
+ }
+
+ // Acronym Bonus
+ // Weighing Logic: Typing the first character of an acronym is as if you
+ // preceded it with two perfect character matches.
+ if (string.charAt(index_in_string - 1) === ' ') {
+ character_score += 0.8; // * Math.min(index_in_string, 5); // Cap bonus at 0.4 * 5
+ }
+
+ // Left trim the already matched part of the string
+ // (forces sequential matching).
+ string = string.substring(index_in_string + 1, string_length);
+
+ total_character_score += character_score;
+ } // end of for loop
+
+ // Uncomment to weigh smaller words higher.
+ // return total_character_score / string_length;
+
+ abbreviation_score = total_character_score / abbreviation_length;
+ //percentage_of_matched_string = abbreviation_length / string_length;
+ //word_score = abbreviation_score * percentage_of_matched_string;
+
+ // Reduce penalty for longer strings.
+ //final_score = (word_score + abbreviation_score) / 2;
+ final_score = ((abbreviation_score * (abbreviation_length / string_length)) + abbreviation_score) / 2;
+
+ final_score = final_score / fuzzies;
+
+ if (start_of_string_bonus && (final_score + 0.15 < 1)) {
+ final_score += 0.15;
+ }
+
+ return final_score;
+};