adding first revision for deployment
|
@ -0,0 +1,14 @@
|
|||
BinaryAttraction
|
||||
|
||||
A little social program i am writing for my senior design class. This site will
|
||||
let you vote in the new digital age on photos.
|
||||
|
||||
Dependencies:
|
||||
* Merb
|
||||
* Haml / Sass
|
||||
* Merb Helpers
|
||||
* Merb Has Flash
|
||||
* Recaptcha account
|
||||
|
||||
Released under GNU GPLv2
|
||||
|
|
@ -0,0 +1,75 @@
|
|||
require 'rubygems'
|
||||
Gem.clear_paths
|
||||
Gem.path.unshift(File.join(File.dirname(__FILE__), "gems"))
|
||||
|
||||
require 'rake'
|
||||
require 'rake/rdoctask'
|
||||
require 'rake/testtask'
|
||||
require 'spec/rake/spectask'
|
||||
require 'fileutils'
|
||||
|
||||
##
|
||||
# requires frozen merb-core (from /framework)
|
||||
# adds the other components to the load path
|
||||
def require_frozen_framework
|
||||
framework = File.join(File.dirname(__FILE__), "framework")
|
||||
if File.directory?(framework)
|
||||
puts "Running from frozen framework"
|
||||
core = File.join(framework,"merb-core")
|
||||
if File.directory?(core)
|
||||
puts "using merb-core from #{core}"
|
||||
$:.unshift File.join(core,"lib")
|
||||
require 'merb-core'
|
||||
end
|
||||
more = File.join(framework,"merb-more")
|
||||
if File.directory?(more)
|
||||
Dir.new(more).select {|d| d =~ /merb-/}.each do |d|
|
||||
$:.unshift File.join(more,d,'lib')
|
||||
end
|
||||
end
|
||||
plugins = File.join(framework,"merb-plugins")
|
||||
if File.directory?(plugins)
|
||||
Dir.new(plugins).select {|d| d =~ /merb_/}.each do |d|
|
||||
$:.unshift File.join(plugins,d,'lib')
|
||||
end
|
||||
end
|
||||
require "merb-core/core_ext/kernel"
|
||||
require "merb-core/core_ext/rubygems"
|
||||
else
|
||||
p "merb doesn't seem to be frozen in /framework"
|
||||
require 'merb-core'
|
||||
end
|
||||
end
|
||||
|
||||
if ENV['FROZEN']
|
||||
require_frozen_framework
|
||||
else
|
||||
require 'merb-core'
|
||||
end
|
||||
|
||||
require 'merb-core/tasks/merb'
|
||||
include FileUtils
|
||||
|
||||
# Load the basic runtime dependencies; this will include
|
||||
# any plugins and therefore plugin rake tasks.
|
||||
init_env = ENV['MERB_ENV'] || 'rake'
|
||||
Merb.load_dependencies(:environment => init_env)
|
||||
|
||||
# Get Merb plugins and dependencies
|
||||
Merb::Plugins.rakefiles.each { |r| require r }
|
||||
|
||||
# Load any app level custom rakefile extensions from lib/tasks
|
||||
tasks_path = File.join(File.dirname(__FILE__), "lib", "tasks")
|
||||
rake_files = Dir["#{tasks_path}/*.rake"]
|
||||
rake_files.each{|rake_file| load rake_file }
|
||||
|
||||
|
||||
desc "start runner environment"
|
||||
task :merb_env do
|
||||
Merb.start_environment(:environment => init_env, :adapter => 'runner')
|
||||
end
|
||||
|
||||
##############################################################################
|
||||
# ADD YOUR CUSTOM TASKS IN /lib/tasks
|
||||
# NAME YOUR RAKE FILES file_name.rake
|
||||
##############################################################################
|
|
@ -0,0 +1,9 @@
|
|||
class Application < Merb::Controller
|
||||
def current_user
|
||||
(@user ||= User.find(session[:user_id]))
|
||||
end
|
||||
|
||||
def reset_session
|
||||
session[:user_id] = nil
|
||||
end
|
||||
end
|
|
@ -0,0 +1,13 @@
|
|||
class Exceptions < Application
|
||||
|
||||
# handle NotFound exceptions (404)
|
||||
def not_found
|
||||
render :format => :html
|
||||
end
|
||||
|
||||
# handle NotAcceptable exceptions (406)
|
||||
def not_acceptable
|
||||
render :format => :html
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,9 @@
|
|||
class Favorites < Application
|
||||
|
||||
# ...and remember, everything returned from an action
|
||||
# goes to the client...
|
||||
def index
|
||||
render
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,9 @@
|
|||
class Home < Application
|
||||
def index
|
||||
render
|
||||
end
|
||||
|
||||
def acceptable_use
|
||||
render
|
||||
end
|
||||
end
|
|
@ -0,0 +1,9 @@
|
|||
class Photos < Application
|
||||
|
||||
# ...and remember, everything returned from an action
|
||||
# goes to the client...
|
||||
def index
|
||||
render
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,27 @@
|
|||
class Sessions < Application
|
||||
def index
|
||||
redirect '/'
|
||||
end
|
||||
|
||||
def new
|
||||
render
|
||||
end
|
||||
|
||||
def create
|
||||
user = User.find_by_user_name params[:user_name]
|
||||
if user.authenticated_against?(params[:password])
|
||||
session[:user_id] = user.id
|
||||
flash[:notice] = 'Great success!'
|
||||
redirect '/'
|
||||
else
|
||||
flash[:error] = 'Login failed'
|
||||
render :new
|
||||
end
|
||||
end
|
||||
|
||||
def delete
|
||||
reset_session
|
||||
flash[:notice] = 'Goodbye!'
|
||||
redirect '/'
|
||||
end
|
||||
end
|
|
@ -0,0 +1,9 @@
|
|||
class Stats < Application
|
||||
|
||||
# ...and remember, everything returned from an action
|
||||
# goes to the client...
|
||||
def index
|
||||
render
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,71 @@
|
|||
class Users < Application
|
||||
before :prepare_user, :only => [ :show, :edit, :update, :delete ]
|
||||
|
||||
include Ambethia::ReCaptcha::Controller
|
||||
|
||||
def index
|
||||
if current_user.administrator?
|
||||
@users = User.find :all, :order => 'user_name ASC'
|
||||
render
|
||||
else
|
||||
redirect url(:user, :id => current_user.user_name)
|
||||
end
|
||||
end
|
||||
|
||||
def show
|
||||
render
|
||||
end
|
||||
|
||||
def new
|
||||
@user = User.new
|
||||
render
|
||||
end
|
||||
|
||||
def create
|
||||
@user = User.new params[:user]
|
||||
@user.user_name = params[:user][:user_name] rescue nil
|
||||
if verify_recaptcha(@user) and @user.save
|
||||
flash[:notice] = 'Great success'
|
||||
redirect '/'
|
||||
else
|
||||
flash[:error] = 'The user could not be created...'
|
||||
render :new
|
||||
end
|
||||
end
|
||||
|
||||
def edit
|
||||
render
|
||||
end
|
||||
|
||||
def update
|
||||
if @user.save
|
||||
flash[:notice] = 'Great success'
|
||||
redirect url(:users)
|
||||
else
|
||||
render :edit
|
||||
end
|
||||
end
|
||||
|
||||
def delete
|
||||
if @user.destroy
|
||||
flash[:notice] = "Epic failure, goodbye #{@user.user_name}"
|
||||
reset_session if @user.id == session[:user_id]
|
||||
else
|
||||
flash[:error] = 'That does not work...'
|
||||
end
|
||||
redirect url(:users)
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def prepare_user
|
||||
@user = if current_user.administrator?
|
||||
User.find_by_user_name params[:id]
|
||||
else
|
||||
current_user
|
||||
end
|
||||
raise NotFound if @user.nil?
|
||||
@user.attributes = params[:user] if params[:user] and request.post?
|
||||
true
|
||||
end
|
||||
end
|
|
@ -0,0 +1,9 @@
|
|||
class Votes < Application
|
||||
|
||||
# ...and remember, everything returned from an action
|
||||
# goes to the client...
|
||||
def index
|
||||
render
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module FavoritesHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,15 @@
|
|||
module Merb
|
||||
module GlobalHelpers
|
||||
def logged_in?
|
||||
!session[:user_id].nil?
|
||||
end
|
||||
|
||||
def error_messages(obj)
|
||||
if obj.errors.empty?
|
||||
nil
|
||||
else
|
||||
"<div id='model_errors'><ul>#{obj.errors.each_full { |msg| "<li>#{msg}</li>" }}</ul></div>"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module HomeHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module PhotosHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module SessionHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module StatsHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module UsersHelper
|
||||
include Ambethia::ReCaptcha::Helper
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,5 @@
|
|||
module Merb
|
||||
module VotesHelper
|
||||
|
||||
end
|
||||
end # Merb
|
|
@ -0,0 +1,9 @@
|
|||
module Merb
|
||||
module Session
|
||||
|
||||
# The Merb::Session module gets mixed into Merb::SessionContainer to allow
|
||||
# app-level functionality; it will be included and methods will be available
|
||||
# through request.session as instance methods.
|
||||
|
||||
end
|
||||
end
|
|
@ -0,0 +1,8 @@
|
|||
class Photo < ActiveRecord::Base
|
||||
#property :id, Integer, :serial => true
|
||||
#property :filename, String
|
||||
#property :email_hash, String
|
||||
#property :created_at, DateTime
|
||||
validates_presence_of :filename
|
||||
has_many :votes, :dependent => :destroy
|
||||
end
|
|
@ -0,0 +1,44 @@
|
|||
class User < ActiveRecord::Base
|
||||
attr_accessor :password, :password_confirmation
|
||||
attr_protected :user_name
|
||||
attr_protected :auth_token
|
||||
attr_protected :authorized
|
||||
|
||||
validates_presence_of :user_name
|
||||
validates_length_of :user_name, :within => (6..32)
|
||||
validates_uniqueness_of :user_name
|
||||
validates_format_of :user_name, :with => /[\w_-]+/
|
||||
|
||||
has_many :votes, :dependent => :destroy
|
||||
|
||||
before_validation :saltify_password
|
||||
|
||||
def authenticated_against?(str)
|
||||
ss = User.salted_string(str)
|
||||
if self.auth_token.to_s == ss
|
||||
true
|
||||
else
|
||||
false
|
||||
end
|
||||
end
|
||||
|
||||
def self.salted_string(str)
|
||||
Digest::SHA1.hexdigest("#{Merb::Config[:session_secret_key]}--#{str}--")
|
||||
end
|
||||
|
||||
protected
|
||||
|
||||
def saltify_password
|
||||
if !self.password.to_s.empty?
|
||||
if self.password.to_s.size < 6
|
||||
self.errors.add(:password, 'is too short')
|
||||
elsif self.password != self.password_confirmation
|
||||
self.errors.add(:passwords, 'do not match')
|
||||
else
|
||||
self.auth_token = User.salted_string(self.password)
|
||||
end
|
||||
elsif self.auth_token.to_s.empty?
|
||||
self.errors.add(:password, 'is missing')
|
||||
end
|
||||
end
|
||||
end
|
|
@ -0,0 +1,34 @@
|
|||
class Vote < ActiveRecord::Base
|
||||
belongs_to :photo
|
||||
belongs_to :user
|
||||
|
||||
validates_presence_of :vote
|
||||
|
||||
##
|
||||
# Checks if this vote is anonymous, or not an authenticated User vote.
|
||||
#
|
||||
def anonymous?
|
||||
self.user_id.nil?
|
||||
end
|
||||
|
||||
##
|
||||
# Convert this Vote to a number.
|
||||
#
|
||||
def to_i
|
||||
self.vote? ? 1 : 0
|
||||
end
|
||||
|
||||
##
|
||||
# Is this a 'no' vote.
|
||||
#
|
||||
def zero?
|
||||
self.to_i == 0
|
||||
end
|
||||
|
||||
##
|
||||
# Is this a 'yes' vote.
|
||||
#
|
||||
def one?
|
||||
self.to_i == 1
|
||||
end
|
||||
end
|
|
@ -0,0 +1,216 @@
|
|||
<html>
|
||||
<head>
|
||||
<meta http-equiv="Content-type" content="text/html; charset=utf-8" />
|
||||
<title><%= @exception_name %></title>
|
||||
<style type="text/css" media="screen">
|
||||
body {
|
||||
font-family:arial;
|
||||
font-size:11px;
|
||||
}
|
||||
h1 {
|
||||
font-size:48px;
|
||||
letter-spacing:-4px;
|
||||
margin:0;
|
||||
line-height:36px;
|
||||
color:#333;
|
||||
}
|
||||
h1 sup {
|
||||
font-size: 0.5em;
|
||||
}
|
||||
h1 sup.error_500, h1 sup.error_400 {
|
||||
color:#990E05;
|
||||
}
|
||||
h1 sup.error_100, h1 sup.error_200 {
|
||||
color:#00BF10;
|
||||
}
|
||||
h1 sup.error_300 {
|
||||
/* pretty sure you cant 'see' status 300
|
||||
errors but if you could I think they
|
||||
would be blue */
|
||||
color:#1B2099;
|
||||
}
|
||||
h2 {
|
||||
font-size:36px;
|
||||
letter-spacing:-3px;
|
||||
margin:0;
|
||||
line-height:28px;
|
||||
color:#444;
|
||||
}
|
||||
a, a:visited {
|
||||
color:#00BF10;
|
||||
}
|
||||
.internalError {
|
||||
width:800px;
|
||||
margin:50px auto;
|
||||
}
|
||||
.header {
|
||||
border-bottom:10px solid #333;
|
||||
margin-bottom:1px;
|
||||
background-image: url("");
|
||||
padding:20px;
|
||||
}
|
||||
table.trace {
|
||||
width:100%;
|
||||
font-family:courier, monospace;
|
||||
letter-spacing:-1px;
|
||||
border-collapse: collapse;
|
||||
border-spacing:0;
|
||||
}
|
||||
table.trace tr td{
|
||||
padding:0;
|
||||
height:26px;
|
||||
font-size:13px;
|
||||
vertical-align:middle;
|
||||
}
|
||||
table.trace tr.file{
|
||||
border-top:2px solid #fff;
|
||||
background-color:#F3F3F3;
|
||||
}
|
||||
table.trace tr.source {
|
||||
background-color:#F8F8F8;
|
||||
display:none;
|
||||
}
|
||||
table.trace .open tr.source {
|
||||
display:table-row;
|
||||
}
|
||||
table.trace tr.file td.expand {
|
||||
width:23px;
|
||||
background-image: url();
|
||||
background-position:top left;
|
||||
background-repeat:no-repeat;
|
||||
}
|
||||
table.trace .open tr.file td.expand {
|
||||
width:19px;
|
||||
background-image: url();
|
||||
background-position:top left;
|
||||
background-repeat:no-repeat;
|
||||
}
|
||||
table.trace tr.source td.collapse {
|
||||
width:19px;
|
||||
background-image: url();
|
||||
background-position:bottom left;
|
||||
background-repeat:no-repeat;
|
||||
background-color:#6F706F;
|
||||
}
|
||||
table.trace tr td.path {
|
||||
padding-left:10px;
|
||||
}
|
||||
table.trace tr td.code {
|
||||
padding-left:35px;
|
||||
white-space: pre;
|
||||
line-height:9px;
|
||||
padding-bottom:10px;
|
||||
}
|
||||
table.trace tr td.code em {
|
||||
font-weight:bold;
|
||||
color:#00BF10;
|
||||
}
|
||||
table.trace tr td.code a {
|
||||
width: 20px;
|
||||
float: left;
|
||||
}
|
||||
table.trace tr td.code .more {
|
||||
color:#666;
|
||||
}
|
||||
table.trace tr td.line {
|
||||
width:30px;
|
||||
text-align:right;
|
||||
padding-right:4px;
|
||||
}
|
||||
.footer {
|
||||
margin-top:5px;
|
||||
font-size:11px;
|
||||
color:#444;
|
||||
text-align:right;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div class="internalError">
|
||||
|
||||
<div class="header">
|
||||
<h1><%= @exception_name %> <sup class="error_<%= @exception.class::STATUS %>"><%= @exception.class::STATUS %></sup></h1>
|
||||
<% if show_details = ::Merb::Config[:exception_details] -%>
|
||||
<h2><%=h @exception.message %></h2>
|
||||
<% else -%>
|
||||
<h2>Sorry about that...</h2>
|
||||
<% end -%>
|
||||
<h3>Parameters</h3>
|
||||
<ul>
|
||||
<% params[:original_params].each do |param, value| %>
|
||||
<li><strong><%= param %>:</strong> <%= value.inspect %></li>
|
||||
<% end %>
|
||||
<%= "<li>None</li>" if params[:original_params].empty? %>
|
||||
</ul>
|
||||
|
||||
<h3>Session</h3>
|
||||
<ul>
|
||||
<% params[:original_session].each do |param, value| %>
|
||||
<li><strong><%= param %>:</strong> <%= value.inspect %></li>
|
||||
<% end %>
|
||||
<%= "<li>None</li>" if params[:original_session].empty? %>
|
||||
</ul>
|
||||
|
||||
<h3>Cookies</h3>
|
||||
<ul>
|
||||
<% params[:original_cookies].each do |param, value| %>
|
||||
<li><strong><%= param %>:</strong> <%= value.inspect %></li>
|
||||
<% end %>
|
||||
<%= "<li>None</li>" if params[:original_cookies].empty? %>
|
||||
</ul>
|
||||
</div>
|
||||
|
||||
<% if show_details %>
|
||||
<table class="trace">
|
||||
<% @exception.backtrace.each_with_index do |line, index| %>
|
||||
<tbody class="close">
|
||||
<tr class="file">
|
||||
<td class="expand">
|
||||
</td>
|
||||
<td class="path">
|
||||
<%= (line.match(/^([^:]+)/)[1] rescue 'unknown').sub(/\/((opt|usr)\/local\/lib\/(ruby\/)?(gems\/)?(1.8\/)?(gems\/)?|.+\/app\/)/, '') %>
|
||||
<% unless line.match(/\.erb:/) %>
|
||||
in "<strong><%= line.match(/:in `(.+)'$/)[1] rescue '?' %></strong>"
|
||||
<% else %>
|
||||
(<strong>ERB Template</strong>)
|
||||
<% end %>
|
||||
</td>
|
||||
<td class="line">
|
||||
<a href="txmt://open?url=file://<%=file = (line.match(/^([^:]+)/)[1] rescue 'unknown')%>&line=<%= lineno = line.match(/:([0-9]+):/)[1] rescue '?' %>"><%=lineno%></a>
|
||||
</td>
|
||||
</tr>
|
||||
<tr class="source">
|
||||
<td class="collapse">
|
||||
</td>
|
||||
<td class="code" colspan="2"><% (__caller_lines__(file, lineno, 5) rescue []).each do |llineno, lcode, lcurrent| %>
|
||||
<a href="txmt://open?url=file://<%=file%>&line=<%=llineno%>"><%= llineno %></a><%='<em>' if llineno==lineno.to_i %><%= lcode.size > 90 ? CGI.escapeHTML(lcode[0..90])+'<span class="more">......</span>' : CGI.escapeHTML(lcode) %><%='</em>' if llineno==lineno.to_i %>
|
||||
<% end %>
|
||||
|
||||
</td>
|
||||
</tr>
|
||||
</tbody>
|
||||
<% end %>
|
||||
</table>
|
||||
<script type="text/javascript" charset="utf-8">
|
||||
// swop the open & closed classes
|
||||
els = document.getElementsByTagName('td');
|
||||
for(i=0; i<els.length; i++){
|
||||
if(els[i].className=='expand' || els[i].className=='collapse'){
|
||||
els[i].onclick = function(e){
|
||||
tbody = this.parentNode.parentNode;
|
||||
if(tbody.className=='open'){
|
||||
tbody.className='closed';
|
||||
}else{
|
||||
tbody.className='open';
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
</script>
|
||||
<% end %>
|
||||
<div class="footer">
|
||||
lots of love, from <a href="#">merb</a>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
|
@ -0,0 +1,63 @@
|
|||
<div id="container">
|
||||
<div id="header-container">
|
||||
<img src="/images/merb.jpg" />
|
||||
<!-- <h1>Mongrel + Erb</h1> -->
|
||||
<h2>pocket rocket web framework</h2>
|
||||
<hr />
|
||||
</div>
|
||||
|
||||
<div id="left-container">
|
||||
<h3>Exception:</h3>
|
||||
<p><%= params[:exception] %></p>
|
||||
</div>
|
||||
|
||||
<div id="main-container">
|
||||
<h3>Why am I seeing this page?</h3>
|
||||
<p>Merb couldn't find an appropriate content_type to return,
|
||||
based on what you said was available via provides() and
|
||||
what the client requested.</p>
|
||||
|
||||
<h3>How to add a mime-type</h3>
|
||||
<pre><code>
|
||||
Merb.add_mime_type :pdf, :to_pdf, %w[application/pdf], "Content-Encoding" => "gzip"
|
||||
</code></pre>
|
||||
<h3>What this means is:</h3>
|
||||
<ul>
|
||||
<li>Add a mime-type for :pdf</li>
|
||||
<li>Register the method for converting objects to PDF as <code>#to_pdf</code>.</li>
|
||||
<li>Register the incoming mime-type "Accept" header as <code>application/pdf</code>.</li>
|
||||
<li>Specify a new header for PDF types so it will set <code>Content-Encoding</code> to gzip.</li>
|
||||
</ul>
|
||||
|
||||
<h3>You can then do:</h3>
|
||||
<pre><code>
|
||||
class Foo < Application
|
||||
provides :pdf
|
||||
end
|
||||
</code></pre>
|
||||
|
||||
<h3>Where can I find help?</h3>
|
||||
<p>If you have any questions or if you can't figure something out, please take a
|
||||
look at our <a href="http://merbivore.com/"> project page</a>,
|
||||
feel free to come chat at irc.freenode.net, channel #merb,
|
||||
or post to <a href="http://groups.google.com/group/merb">merb mailing list</a>
|
||||
on Google Groups.</p>
|
||||
|
||||
<h3>What if I've found a bug?</h3>
|
||||
<p>If you want to file a bug or make your own contribution to Merb,
|
||||
feel free to register and create a ticket at our
|
||||
<a href="http://merb.lighthouseapp.com/">project development page</a>
|
||||
on Lighthouse.</p>
|
||||
|
||||
<h3>How do I edit this page?</h3>
|
||||
<p>You can change what people see when this happens by editing <tt>app/views/exceptions/not_acceptable.html.erb</tt>.</p>
|
||||
|
||||
</div>
|
||||
|
||||
<div id="footer-container">
|
||||
<hr />
|
||||
<div class="left"></div>
|
||||
<div class="right">© 2007 the merb dev team</div>
|
||||
<p> </p>
|
||||
</div>
|
||||
</div>
|
|
@ -0,0 +1,47 @@
|
|||
<div id="container">
|
||||
<div id="header-container">
|
||||
<img src="/images/merb.jpg" />
|
||||
<!-- <h1>Mongrel + Erb</h1> -->
|
||||
<h2>pocket rocket web framework</h2>
|
||||
<hr />
|
||||
</div>
|
||||
|
||||
<div id="left-container">
|
||||
<h3>Exception:</h3>
|
||||
<p><%= params[:exception] %></p>
|
||||
</div>
|
||||
|
||||
<div id="main-container">
|
||||
<h3>Welcome to Merb!</h3>
|
||||
<p>Merb is a light-weight MVC framework written in Ruby. We hope you enjoy it.</p>
|
||||
|
||||
<h3>Where can I find help?</h3>
|
||||
<p>If you have any questions or if you can't figure something out, please take a
|
||||
look at our <a href="http://merbivore.com/"> project page</a>,
|
||||
feel free to come chat at irc.freenode.net, channel #merb,
|
||||
or post to <a href="http://groups.google.com/group/merb">merb mailing list</a>
|
||||
on Google Groups.</p>
|
||||
|
||||
<h3>What if I've found a bug?</h3>
|
||||
<p>If you want to file a bug or make your own contribution to Merb,
|
||||
feel free to register and create a ticket at our
|
||||
<a href="http://merb.lighthouseapp.com/">project development page</a>
|
||||
on Lighthouse.</p>
|
||||
|
||||
<h3>How do I edit this page?</h3>
|
||||
<p>You're seeing this page because you need to edit the following files:
|
||||
<ul>
|
||||
<li>config/router.rb <strong><em>(recommended)</em></strong></li>
|
||||
<li>app/views/exceptions/not_found.html.erb <strong><em>(recommended)</em></strong></li>
|
||||
<li>app/views/layout/application.html.erb <strong><em>(change this layout)</em></strong></li>
|
||||
</ul>
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div id="footer-container">
|
||||
<hr />
|
||||
<div class="left"></div>
|
||||
<div class="right">© 2007 the merb dev team</div>
|
||||
<p> </p>
|
||||
</div>
|
||||
</div>
|
|
@ -0,0 +1 @@
|
|||
You're in index of the Favorites controller.
|
|
@ -0,0 +1,12 @@
|
|||
%h1 Acceptable use policy
|
||||
|
||||
%p There are a few rules you must abide by to use this site. Violation will at least mean a permenant ban and at worst prosecution under federal law.
|
||||
|
||||
%ol
|
||||
%li You must be 18 years old.
|
||||
%li The pictures must be of a person. One <em>consenting adult</em> person.
|
||||
%li Group pictures are allowed on the grounds that you are voting for <strong>the entire group</strong>.
|
||||
%li Nudity is not allowed. Partial nudity is allowed. We will use any discretionary method to determine what is allowed. Be warned.
|
||||
%li Always offer to send a statistics link to the person you are photographing. It's just polite :)
|
||||
|
||||
%p We will <strong>never</strong> share any personal information gathered on this site. The only information we gather is so that <em>you</em>, the user, will have access to the data later.
|
|
@ -0,0 +1,61 @@
|
|||
%style{ :type => 'text/css' }
|
||||
:sass
|
||||
#front_page
|
||||
blockquote
|
||||
:font-size 16px
|
||||
:padding 5px 0px 5px 15px
|
||||
:background-color #eeeeee
|
||||
:border-top 1px dashed #9f9f9f
|
||||
:border-right 1px dashed #9f9f9f
|
||||
:border-bottom 1px solid #d4d4d4
|
||||
:border-left 1px solid #d4d4d4
|
||||
em
|
||||
:margin-left 20px
|
||||
h1
|
||||
:text-align center
|
||||
|
||||
%span#front_page
|
||||
%h1 What this is all about
|
||||
|
||||
%blockquote
|
||||
All you young guys are on a binary system. It's either <tt>0</tt> or <tt>1</tt>.
|
||||
%br
|
||||
%em Larry Bell
|
||||
|
||||
%blockquote
|
||||
All you old guys are on the analog system. Join the digital revolution.
|
||||
%br
|
||||
%em Ross Bagwell
|
||||
|
||||
%h1 Getting started
|
||||
|
||||
%ul.no_list_style
|
||||
- if logged_in?
|
||||
%li
|
||||
%a{ :href => url(:edit_user, :id => current_user.user_name) }
|
||||
%img{ :src => '/images/system-lock-screen.png' }
|
||||
Change your password
|
||||
%li
|
||||
%a{ :href => url(:favorite, :id => current_user.user_name) }
|
||||
%img{ :src => '/images/emblem-favorite.png' }
|
||||
Check your favorites
|
||||
%li
|
||||
%a{ :href => url(:new_photo) }
|
||||
%img{ :src => '/images/camera-photo.png' }
|
||||
Upload photos
|
||||
%li
|
||||
%a{ :href => url(:votes) }
|
||||
%img{ :src => '/images/vote.png' }
|
||||
Vote on new photos
|
||||
%li
|
||||
%a{ :href => url(:stat, :id => current_user.user_name) }
|
||||
%img{ :src => '/images/utilities-system-monitor.png' }
|
||||
Check stats on photos of yourself
|
||||
- else
|
||||
%li <img src='/images/system-users.png' /> Sign up for an account
|
||||
%li <img src='/images/system-lock-screen.png' /> Log in if you have one
|
||||
%li <img src='/images/emblem-favorite.png' /> Check your favorites
|
||||
%li <img src='/images/camera-photo.png' /> Upload photos
|
||||
%li <img src='/images/vote.png' /> Vote on new photos
|
||||
%li <img src='/images/utilities-system-monitor.png' /> Check stats on photos of yourself
|
||||
|
|
@ -0,0 +1,59 @@
|
|||
!!! Strict
|
||||
%html{html_attrs{'en-us'}}
|
||||
%head
|
||||
%title Binary Attraction
|
||||
%meta{ 'http-equiv' => "content-type", :content => "text/html; charset=utf8" }
|
||||
%link{ :href => "/stylesheets/ba.css", :rel => "stylesheet", :type => "text/css", :media => "screen", :charset => "utf-8" }
|
||||
%script{ :src => "/javascripts/prototype.js", :type => "text/javascript" }
|
||||
%script{ :src => "/javascripts/effects.js", :type => "text/javascript" }
|
||||
%script{ :src => "/javascripts/dragdrop.js", :type => "text/javascript" }
|
||||
%script{ :src => "/javascripts/application.js", :type => "text/javascript" }
|
||||
- unless flash.keys.empty?
|
||||
:javascript
|
||||
hide_flashes();
|
||||
%body
|
||||
#container
|
||||
- unless flash.keys.empty?
|
||||
#flash_container
|
||||
- flash.keys.each do |key|
|
||||
%div{ :class => key }= flash[key]
|
||||
#header
|
||||
%span#header_image
|
||||
%a{ :href => '/', :title => 'B.A. Home' }
|
||||
%img{ :src => '/images/binaryattraction.png', :alt => 'Binary Attraction' }
|
||||
#tool_bar
|
||||
- if logged_in?
|
||||
%a{ :href => url(:new_vote), :title => 'Vote' }
|
||||
%img{ :src => '/images/vote.png' }
|
||||
Vote
|
||||
|
|
||||
%a{ :href => url(:new_photo), :title => 'Upload a photo' }
|
||||
%img{ :src => '/images/camera-photo.png' }
|
||||
Upload a photo
|
||||
|
|
||||
%a{ :href => url(:favorite, :id => session[:user_id]), :title => 'Favorites' }
|
||||
%img{ :src => '/images/emblem-favorite.png' }
|
||||
Favorites
|
||||
|
|
||||
%a{ :href => url(:stats), :title => 'Stats' }
|
||||
%img{ :src => '/images/utilities-system-monitor.png' }
|
||||
Check Stats
|
||||
|
|
||||
%a{ :href => url(:delete_session, :id => session[:user_id]), :title => 'Log out' }
|
||||
%img{ :src => '/images/system-log-out.png' }
|
||||
Log out
|
||||
- else
|
||||
%a{ :href => url(:new_user), :title => 'Sign Up' }
|
||||
%img{ :src => '/images/system-users.png' }
|
||||
Sign Up
|
||||
|
|
||||
%a{ :href => url(:new_session), :title => 'Log In' }
|
||||
%img{ :src => '/images/system-lock-screen.png' }
|
||||
Log In
|
||||
#content
|
||||
= catch_content :for_layout
|
||||
#footer
|
||||
© 2008
|
||||
%a{ :href => 'http://penguincoder.org' } penguincoder
|
||||
| Usage of this site requires
|
||||
%a{ :href => '/acceptable_use' } acceptable use policy
|
|
@ -0,0 +1 @@
|
|||
You're in index of the Photos controller.
|
|
@ -0,0 +1,12 @@
|
|||
= form :action => url(:sessions) do
|
||||
%fieldset
|
||||
%p
|
||||
%label{ :for => 'user_name' }
|
||||
User Name
|
||||
= text_field :name => 'user_name', :id => 'user_name'
|
||||
%p
|
||||
%label{ :for => 'password' }
|
||||
Password
|
||||
= password_field :name => 'password', :id => 'password'
|
||||
= submit 'Login'
|
||||
|
|
@ -0,0 +1 @@
|
|||
You're in index of the Stats controller.
|
|
@ -0,0 +1,12 @@
|
|||
= error_messages @user
|
||||
|
||||
= form :action => url(:edit_user, :id => @user.user_name) do
|
||||
%fieldset
|
||||
%legend== Changing settings for user <em>#{@user.user_name}</em>
|
||||
%p
|
||||
%label{ :for => 'password' } Password
|
||||
= password_field :name => 'user[password]', :id => 'password'
|
||||
%p
|
||||
%label{ :for => 'password_confirmation' } Confirm
|
||||
= password_field :name => 'user[password]', :id => 'password'
|
||||
= submit 'Save'
|
|
@ -0,0 +1,9 @@
|
|||
%h1 Peoples.
|
||||
|
||||
%ul
|
||||
- @users.each do |user|
|
||||
%li
|
||||
%a{ :href => url(:edit_user, :id => user.user_name) }= user.user_name
|
||||
%a{ :href => url(:delete_user, :id => user.user_name), :onclick => "return confirm('Are you sure?');" }
|
||||
%img{ :src => '/images/user-trash.png' }
|
||||
|
|
@ -0,0 +1,18 @@
|
|||
= error_messages @user
|
||||
|
||||
= form_for @user do
|
||||
%fieldset
|
||||
%legend Create a new user
|
||||
%p
|
||||
%label{ :for => 'user_name' } User Name
|
||||
= text_field :name => 'user[user_name]', :id => 'user_name'
|
||||
%p
|
||||
%label{ :for => 'password' } Password
|
||||
= password_field :name => 'user[password]', :id => 'password'
|
||||
%p
|
||||
%label{ :for => 'password_confirmation' } Confirm
|
||||
= password_field :name => 'user[password_confirmation]', :id => 'password_confirmation'
|
||||
%p
|
||||
%label{ :for => 'recaptcha_response_field' } Captcha
|
||||
#recaptcha_container= recaptcha_tags
|
||||
= submit 'Create'
|
|
@ -0,0 +1 @@
|
|||
You're in index of the Votes controller.
|
|
@ -0,0 +1 @@
|
|||
Autotest.add_discovery { "merb" }
|
|
@ -0,0 +1,149 @@
|
|||
# Adapted from Autotest::Rails
|
||||
require 'autotest'
|
||||
|
||||
class Autotest::Merb < Autotest
|
||||
|
||||
# +model_tests_dir+:: the directory to find model-centric tests
|
||||
# +controller_tests_dir+:: the directory to find controller-centric tests
|
||||
# +view_tests_dir+:: the directory to find view-centric tests
|
||||
# +fixtures_dir+:: the directory to find fixtures in
|
||||
attr_accessor :model_tests_dir, :controller_tests_dir, :view_tests_dir, :fixtures_dir
|
||||
|
||||
def initialize
|
||||
super
|
||||
|
||||
initialize_test_layout
|
||||
|
||||
# Ignore any happenings in these directories
|
||||
add_exception %r%^\./(?:doc|log|public|tmp)%
|
||||
|
||||
# Ignore any mappings that Autotest may have already set up
|
||||
clear_mappings
|
||||
|
||||
# Any changes to a file in the root of the 'lib' directory will run any
|
||||
# model test with a corresponding name.
|
||||
add_mapping %r%^lib\/.*\.rb% do |filename, _|
|
||||
files_matching Regexp.new(["^#{model_test_for(filename)}$"])
|
||||
end
|
||||
|
||||
# Any changes to a fixture will run corresponding view, controller and
|
||||
# model tests
|
||||
add_mapping %r%^#{fixtures_dir}/(.*)s.yml% do |_, m|
|
||||
[
|
||||
model_test_for(m[1]),
|
||||
controller_test_for(m[1]),
|
||||
view_test_for(m[1])
|
||||
]
|
||||
end
|
||||
|
||||
# Any change to a test or test will cause it to be run
|
||||
add_mapping %r%^test/(unit|models|integration|controllers|views|functional)/.*rb$% do |filename, _|
|
||||
filename
|
||||
end
|
||||
|
||||
# Any change to a model will cause it's corresponding test to be run
|
||||
add_mapping %r%^app/models/(.*)\.rb$% do |_, m|
|
||||
model_test_for(m[1])
|
||||
end
|
||||
|
||||
# Any change to the global helper will result in all view and controller
|
||||
# tests being run
|
||||
add_mapping %r%^app/helpers/global_helpers.rb% do
|
||||
files_matching %r%^test/(views|functional|controllers)/.*_test\.rb$%
|
||||
end
|
||||
|
||||
# Any change to a helper will run it's corresponding view and controller
|
||||
# tests, unless the helper is the global helper. Changes to the global
|
||||
# helper run all view and controller tests.
|
||||
add_mapping %r%^app/helpers/(.*)_helper(s)?.rb% do |_, m|
|
||||
if m[1] == "global" then
|
||||
files_matching %r%^test/(views|functional|controllers)/.*_test\.rb$%
|
||||
else
|
||||
[
|
||||
view_test_for(m[1]),
|
||||
controller_test_for(m[1])
|
||||
]
|
||||
end
|
||||
end
|
||||
|
||||
# Changes to views result in their corresponding view and controller test
|
||||
# being run
|
||||
add_mapping %r%^app/views/(.*)/% do |_, m|
|
||||
[
|
||||
view_test_for(m[1]),
|
||||
controller_test_for(m[1])
|
||||
]
|
||||
end
|
||||
|
||||
# Changes to a controller result in its corresponding test being run. If
|
||||
# the controller is the exception or application controller, all
|
||||
# controller tests are run.
|
||||
add_mapping %r%^app/controllers/(.*)\.rb$% do |_, m|
|
||||
if ["application", "exception"].include?(m[1])
|
||||
files_matching %r%^test/(controllers|views|functional)/.*_test\.rb$%
|
||||
else
|
||||
controller_test_for(m[1])
|
||||
end
|
||||
end
|
||||
|
||||
# If a change is made to the router, run all controller and view tests
|
||||
add_mapping %r%^config/router.rb$% do # FIX
|
||||
files_matching %r%^test/(controllers|views|functional)/.*_test\.rb$%
|
||||
end
|
||||
|
||||
# If any of the major files governing the environment are altered, run
|
||||
# everything
|
||||
add_mapping %r%^test/test_helper.rb|config/(init|rack|environments/test.rb|database.yml)% do # FIX
|
||||
files_matching %r%^test/(unit|models|controllers|views|functional)/.*_test\.rb$%
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Determines the paths we can expect tests or specs to reside, as well as
|
||||
# corresponding fixtures.
|
||||
def initialize_test_layout
|
||||
self.model_tests_dir = "test/unit"
|
||||
self.controller_tests_dir = "test/functional"
|
||||
self.view_tests_dir = "test/views"
|
||||
self.fixtures_dir = "test/fixtures"
|
||||
end
|
||||
|
||||
# Given a filename and the test type, this method will return the
|
||||
# corresponding test's or spec's name.
|
||||
#
|
||||
# ==== Arguments
|
||||
# +filename+<String>:: the file name of the model, view, or controller
|
||||
# +kind_of_test+<Symbol>:: the type of test we that we should run
|
||||
#
|
||||
# ==== Returns
|
||||
# String:: the name of the corresponding test or spec
|
||||
#
|
||||
# ==== Example
|
||||
#
|
||||
# > test_for("user", :model)
|
||||
# => "user_test.rb"
|
||||
# > test_for("login", :controller)
|
||||
# => "login_controller_test.rb"
|
||||
# > test_for("form", :view)
|
||||
# => "form_view_spec.rb" # If you're running a RSpec-like suite
|
||||
def test_for(filename, kind_of_test)
|
||||
name = [filename]
|
||||
name << kind_of_test.to_s if kind_of_test == :view
|
||||
name << "test"
|
||||
return name.join("_") + ".rb"
|
||||
end
|
||||
|
||||
def model_test_for(filename)
|
||||
[model_tests_dir, test_for(filename, :model)].join("/")
|
||||
end
|
||||
|
||||
def controller_test_for(filename)
|
||||
[controller_tests_dir, test_for(filename, :controller)].join("/")
|
||||
end
|
||||
|
||||
def view_test_for(filename)
|
||||
[view_tests_dir, test_for(filename, :view)].join("/")
|
||||
end
|
||||
|
||||
end
|
|
@ -0,0 +1,165 @@
|
|||
# Adapted from Autotest::Rails, RSpec's autotest class, as well as merb-core's.
|
||||
require 'autotest'
|
||||
|
||||
class RspecCommandError < StandardError; end
|
||||
|
||||
# This class maps your application's structure so Autotest can understand what
|
||||
# specs to run when files change.
|
||||
#
|
||||
# Fixtures are _not_ covered by this class. If you change a fixture file, you
|
||||
# will have to run your spec suite manually, or, better yet, provide your own
|
||||
# Autotest map explaining how your fixtures are set up.
|
||||
class Autotest::MerbRspec < Autotest
|
||||
def initialize
|
||||
super
|
||||
|
||||
# Ignore any happenings in these directories
|
||||
add_exception %r%^\./(?:doc|log|public|tmp|\.git|\.hg|\.svn|framework|gems|schema|\.DS_Store|autotest|bin|.*\.sqlite3)%
|
||||
# Ignore SCM directories and custom Autotest mappings
|
||||
%w[.svn .hg .git .autotest].each { |exception| add_exception(exception) }
|
||||
|
||||
# Ignore any mappings that Autotest may have already set up
|
||||
clear_mappings
|
||||
|
||||
# Anything in /lib could have a spec anywhere, if at all. So, look for
|
||||
# files with roughly the same name as the file in /lib
|
||||
add_mapping %r%^lib\/(.*)\.rb% do |_, m|
|
||||
files_matching %r%^spec\/#{m[1]}%
|
||||
end
|
||||
|
||||
add_mapping %r%^spec/(spec_helper|shared/.*)\.rb$% do
|
||||
all_specs
|
||||
end
|
||||
|
||||
# Changing a spec will cause it to run itself
|
||||
add_mapping %r%^spec/.*\.rb$% do |filename, _|
|
||||
filename
|
||||
end
|
||||
|
||||
# Any change to a model will cause it's corresponding test to be run
|
||||
add_mapping %r%^app/models/(.*)\.rb$% do |_, m|
|
||||
spec_for(m[1], 'model')
|
||||
end
|
||||
|
||||
# Any change to global_helpers will result in all view and controller
|
||||
# tests being run
|
||||
add_mapping %r%^app/helpers/global_helpers\.rb% do
|
||||
files_matching %r%^spec/(views|controllers|helpers)/.*_spec\.rb$%
|
||||
end
|
||||
|
||||
# Any change to a helper will cause its spec to be run
|
||||
add_mapping %r%^app/helpers/((.*)_helper(s)?)\.rb% do |_, m|
|
||||
spec_for(m[1], 'helper')
|
||||
end
|
||||
|
||||
# Changes to a view cause its spec to be run
|
||||
add_mapping %r%^app/views/(.*)/% do |_, m|
|
||||
spec_for(m[1], 'view')
|
||||
end
|
||||
|
||||
# Changes to a controller result in its corresponding spec being run. If
|
||||
# the controller is the exception or application controller, all
|
||||
# controller specs are run.
|
||||
add_mapping %r%^app/controllers/(.*)\.rb$% do |_, m|
|
||||
if ["application", "exception"].include?(m[1])
|
||||
files_matching %r%^spec/controllers/.*_spec\.rb$%
|
||||
else
|
||||
spec_for(m[1], 'controller')
|
||||
end
|
||||
end
|
||||
|
||||
# If a change is made to the router, run controller, view and helper specs
|
||||
add_mapping %r%^config/router.rb$% do
|
||||
files_matching %r%^spec/(controllers|views|helpers)/.*_spec\.rb$%
|
||||
end
|
||||
|
||||
# If any of the major files governing the environment are altered, run
|
||||
# everything
|
||||
add_mapping %r%^config/(init|rack|environments/test).*\.rb|database\.yml% do
|
||||
all_specs
|
||||
end
|
||||
end
|
||||
|
||||
def failed_results(results)
|
||||
results.scan(/^\d+\)\n(?:\e\[\d*m)?(?:.*?Error in )?'([^\n]*)'(?: FAILED)?(?:\e\[\d*m)?\n(.*?)\n\n/m)
|
||||
end
|
||||
|
||||
def handle_results(results)
|
||||
@failures = failed_results(results)
|
||||
@files_to_test = consolidate_failures(@failures)
|
||||
@files_to_test.empty? && !$TESTING ? hook(:green) : hook(:red)
|
||||
@tainted = !@files_to_test.empty?
|
||||
end
|
||||
|
||||
def consolidate_failures(failed)
|
||||
filters = Hash.new { |h,k| h[k] = [] }
|
||||
failed.each do |spec, failed_trace|
|
||||
if f = test_files_for(failed).find { |f| f =~ /spec\// }
|
||||
filters[f] << spec
|
||||
break
|
||||
end
|
||||
end
|
||||
filters
|
||||
end
|
||||
|
||||
def make_test_cmd(specs_to_runs)
|
||||
[
|
||||
ruby,
|
||||
"-S",
|
||||
spec_command,
|
||||
add_options_if_present,
|
||||
files_to_test.keys.flatten.join(' ')
|
||||
].join(' ')
|
||||
end
|
||||
|
||||
def add_options_if_present
|
||||
File.exist?("spec/spec.opts") ? "-O spec/spec.opts " : ""
|
||||
end
|
||||
|
||||
# Finds the proper spec command to use. Precendence is set in the
|
||||
# lazily-evaluated method spec_commands. Alias + Override that in
|
||||
# ~/.autotest to provide a different spec command then the default
|
||||
# paths provided.
|
||||
def spec_command(separator=File::ALT_SEPARATOR)
|
||||
unless defined?(@spec_command)
|
||||
@spec_command = spec_commands.find { |cmd| File.exists?(cmd) }
|
||||
|
||||
raise RspecCommandError, "No spec command could be found" unless @spec_command
|
||||
|
||||
@spec_command.gsub!(File::SEPARATOR, separator) if separator
|
||||
end
|
||||
@spec_command
|
||||
end
|
||||
|
||||
# Autotest will look for spec commands in the following
|
||||
# locations, in this order:
|
||||
#
|
||||
# * default spec bin/loader installed in Rubygems
|
||||
# * any spec command found in PATH
|
||||
def spec_commands
|
||||
[File.join(Config::CONFIG['bindir'], 'spec'), 'spec']
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Runs +files_matching+ for all specs
|
||||
def all_specs
|
||||
files_matching %r%^spec/.*_spec\.rb$%
|
||||
end
|
||||
|
||||
# Generates a path to some spec given its kind and the match from a mapping
|
||||
#
|
||||
# ==== Arguments
|
||||
# match<String>:: the match from a mapping
|
||||
# kind<String>:: the kind of spec that the match represents
|
||||
#
|
||||
# ==== Returns
|
||||
# String
|
||||
#
|
||||
# ==== Example
|
||||
# > spec_for('post', :view')
|
||||
# => "spec/views/post_spec.rb"
|
||||
def spec_for(match, kind)
|
||||
File.join("spec", kind + 's', "#{match}_spec.rb")
|
||||
end
|
||||
end
|
|
@ -0,0 +1,8 @@
|
|||
Merb.logger.info("Loaded DEVELOPMENT Environment...")
|
||||
Merb::Config.use { |c|
|
||||
c[:exception_details] = true
|
||||
c[:reload_classes] = true
|
||||
c[:reload_time] = 0.5
|
||||
c[:log_auto_flush ] = true
|
||||
c[:log_level] = :debug
|
||||
}
|
|
@ -0,0 +1,7 @@
|
|||
Merb.logger.info("Loaded PRODUCTION Environment...")
|
||||
Merb::Config.use { |c|
|
||||
c[:exception_details] = false
|
||||
c[:reload_classes] = false
|
||||
c[:log_level] = :error
|
||||
c[:log_file] = Merb.log_path + "/production.log"
|
||||
}
|
|
@ -0,0 +1,7 @@
|
|||
Merb.logger.info("Loaded RAKE Environment...")
|
||||
Merb::Config.use { |c|
|
||||
c[:exception_details] = true
|
||||
c[:reload_classes] = false
|
||||
c[:log_auto_flush ] = true
|
||||
c[:log_file] = Merb.log_path / 'merb_rake.log'
|
||||
}
|
|
@ -0,0 +1,6 @@
|
|||
Merb.logger.info("Loaded TEST Environment...")
|
||||
Merb::Config.use { |c|
|
||||
c[:testing] = true
|
||||
c[:exception_details] = true
|
||||
c[:log_auto_flush ] = true
|
||||
}
|
|
@ -0,0 +1,28 @@
|
|||
Gem.clear_paths
|
||||
Gem.path.unshift(Merb.root / "gems")
|
||||
$LOAD_PATH.unshift(Merb.root / "lib")
|
||||
# Merb.push_path(:lib, Merb.root / "lib") # uses **/*.rb as path glob
|
||||
|
||||
dependencies 'haml', 'sass', 'merb_helpers', 'merb_has_flash', 'digest/sha1', 'recaptcha'
|
||||
|
||||
Merb::BootLoader.after_app_loads do
|
||||
recaptcha_path = File.join(Merb.root, 'config', 'recaptcha.yml')
|
||||
if File.file?(recaptcha_path) and File.readable?(recaptcha_path)
|
||||
rc = YAML::load_file(recaptcha_path)
|
||||
ENV['RECAPTCHA_PUBLIC_KEY'] = rc[:public]
|
||||
ENV['RECAPTCHA_PRIVATE_KEY'] = rc[:private]
|
||||
else
|
||||
raise "ReCaptcha configuration file not found!"
|
||||
end
|
||||
end
|
||||
|
||||
use_orm :activerecord
|
||||
use_test :rspec
|
||||
use_template_engine :haml
|
||||
Merb::Config.use do |c|
|
||||
c[:session_secret_key] = 'ccf75249b0efbdb3edff96d0a1b16b19cf91f31e'
|
||||
c[:session_store] = :activerecord
|
||||
c[:sass] ||= {}
|
||||
c[:sass][:style] = :compact
|
||||
end
|
||||
|
|
@ -0,0 +1,12 @@
|
|||
|
||||
# use PathPrefix Middleware if :path_prefix is set in Merb::Config
|
||||
if prefix = ::Merb::Config[:path_prefix]
|
||||
use Merb::Rack::PathPrefix, prefix
|
||||
end
|
||||
|
||||
# comment this out if you are running merb behind a load balancer
|
||||
# that serves static files
|
||||
use Merb::Rack::Static, Merb.dir_for(:public)
|
||||
|
||||
# this is our main merb application
|
||||
run Merb::Rack::Application.new
|
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
:public: PUBKEY
|
||||
:private: PRIVKEY
|
|
@ -0,0 +1,13 @@
|
|||
Merb.logger.info("Compiling routes...")
|
||||
Merb::Router.prepare do |r|
|
||||
r.resources :sessions
|
||||
r.resources :users
|
||||
r.resources :people
|
||||
r.resources :votes
|
||||
r.resources :photos
|
||||
r.resources :favorites
|
||||
r.resources :stats
|
||||
# r.default_routes
|
||||
r.match('/').to( :controller => 'home', :action => 'index' )
|
||||
r.match('/acceptable_use').to( :controller => 'home', :action => 'acceptable_use' )
|
||||
end
|
|
@ -0,0 +1,19 @@
|
|||
Copyright (c) 2007 Jason L Perry
|
||||
|
||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
||||
of this software and associated documentation files (the "Software"), to deal
|
||||
in the Software without restriction, including without limitation the rights
|
||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||||
copies of the Software, and to permit persons to whom the Software is
|
||||
furnished to do so, subject to the following conditions:
|
||||
|
||||
The above copyright notice and this permission notice shall be included in
|
||||
all copies or substantial portions of the Software.
|
||||
|
||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||||
THE SOFTWARE.
|
|
@ -0,0 +1,78 @@
|
|||
# ReCAPTCHA
|
||||
module Ambethia
|
||||
module ReCaptcha
|
||||
RECAPTCHA_API_SERVER = 'http://api.recaptcha.net';
|
||||
RECAPTCHA_API_SECURE_SERVER = 'https://api-secure.recaptcha.net';
|
||||
RECAPTCHA_VERIFY_SERVER = 'api-verify.recaptcha.net';
|
||||
|
||||
SKIP_VERIFY_ENV = ['test']
|
||||
|
||||
module Helper
|
||||
# Your public API can be specified in the +options+ hash or preferably the environment
|
||||
# variable +RECAPTCHA_PUBLIC_KEY+.
|
||||
def recaptcha_tags(options = {})
|
||||
# Default options
|
||||
key = options[:public_key] ||= ENV['RECAPTCHA_PUBLIC_KEY']
|
||||
error = options[:error] ||= session[:recaptcha_error]
|
||||
uri = options[:ssl] ? RECAPTCHA_API_SECURE_SERVER : RECAPTCHA_API_SERVER
|
||||
xhtml = Builder::XmlMarkup.new :target => out=(''), :indent => 2 # Because I can.
|
||||
if options[:display]
|
||||
xhtml.script(:type => "text/javascript"){ xhtml.text! "var RecaptchaOptions = #{options[:display].to_json};\n"}
|
||||
end
|
||||
if options[:ajax]
|
||||
xhtml.div(:id => 'dynamic_recaptcha') {}
|
||||
xhtml.script(:type => "text/javascript", :src => "#{uri}/js/recaptcha_ajax.js") {}
|
||||
xhtml.script(:type => "text/javascript") do
|
||||
xhtml.text! "Recaptcha.create('#{key}', document.getElementById('dynamic_recaptcha') );"
|
||||
end
|
||||
else
|
||||
xhtml.script(:type => "text/javascript", :src => "#{uri}/challenge?k=#{key}&error=#{error}") {}
|
||||
unless options[:noscript] == false
|
||||
xhtml.noscript do
|
||||
xhtml.iframe(:src => "#{uri}/noscript?k=#{key}",
|
||||
:height => options[:iframe_height] ||= 300,
|
||||
:width => options[:iframe_width] ||= 500,
|
||||
:frameborder => 0) {}; xhtml.br
|
||||
xhtml.textarea(:name => "recaptcha_challenge_field", :rows => 3, :cols => 40) {}
|
||||
xhtml.input :name => "recaptcha_response_field",
|
||||
:type => "hidden", :value => "manual_challenge"
|
||||
end
|
||||
end
|
||||
end
|
||||
raise ReCaptchaError, "No public key specified." unless key
|
||||
return out
|
||||
end # recaptcha_tags
|
||||
end # Helpers
|
||||
|
||||
module Controller
|
||||
# Your private API key must be specified in the environment variable +RECAPTCHA_PRIVATE_KEY+
|
||||
def verify_recaptcha(model = nil)
|
||||
return true if SKIP_VERIFY_ENV.include? ENV['RAILS_ENV']
|
||||
raise ReCaptchaError, "No private key specified." unless ENV['RECAPTCHA_PRIVATE_KEY']
|
||||
begin
|
||||
recaptcha = Net::HTTP.post_form URI.parse("http://#{RECAPTCHA_VERIFY_SERVER}/verify"), {
|
||||
:privatekey => ENV['RECAPTCHA_PRIVATE_KEY'],
|
||||
:remoteip => request.remote_ip,
|
||||
:challenge => params[:recaptcha_challenge_field],
|
||||
:response => params[:recaptcha_response_field]
|
||||
}
|
||||
answer, error = recaptcha.body.split.map { |s| s.chomp }
|
||||
unless answer == 'true'
|
||||
session[:recaptcha_error] = error
|
||||
model.valid? if model
|
||||
model.errors.add_to_base "Captcha response is incorrect, please try again." if model
|
||||
return false
|
||||
else
|
||||
session[:recaptcha_error] = nil
|
||||
return true
|
||||
end
|
||||
rescue Exception => e
|
||||
raise ReCaptchaError, e
|
||||
end
|
||||
end # verify_recaptcha
|
||||
end # ControllerHelpers
|
||||
|
||||
class ReCaptchaError < StandardError; end
|
||||
|
||||
end # ReCaptcha
|
||||
end # Ambethia
|
|
@ -0,0 +1,822 @@
|
|||
#!/usr/bin/env ruby
|
||||
require 'rubygems'
|
||||
require 'rubygems/dependency_installer'
|
||||
require 'rubygems/uninstaller'
|
||||
require 'thor'
|
||||
require 'fileutils'
|
||||
require 'yaml'
|
||||
|
||||
module MerbThorHelper
|
||||
|
||||
private
|
||||
|
||||
# The current working directory, or Merb app root (--merb-root option).
|
||||
def working_dir
|
||||
@_working_dir ||= File.expand_path(options['merb-root'] || Dir.pwd)
|
||||
end
|
||||
|
||||
# We should have a ./src dir for local and system-wide management.
|
||||
def source_dir
|
||||
@_source_dir ||= File.join(working_dir, 'src')
|
||||
create_if_missing(@_source_dir)
|
||||
@_source_dir
|
||||
end
|
||||
|
||||
# If a local ./gems dir is found, it means we are in a Merb app.
|
||||
def application?
|
||||
gem_dir
|
||||
end
|
||||
|
||||
# If a local ./gems dir is found, return it.
|
||||
def gem_dir
|
||||
if File.directory?(dir = File.join(working_dir, 'gems'))
|
||||
dir
|
||||
end
|
||||
end
|
||||
|
||||
# If we're in a Merb app, we can have a ./bin directory;
|
||||
# create it if it's not there.
|
||||
def bin_dir
|
||||
@_bin_dir ||= begin
|
||||
if gem_dir
|
||||
dir = File.join(working_dir, 'bin')
|
||||
create_if_missing(dir)
|
||||
dir
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# Helper to create dir unless it exists.
|
||||
def create_if_missing(path)
|
||||
FileUtils.mkdir(path) unless File.exists?(path)
|
||||
end
|
||||
|
||||
# Create a modified executable wrapper in the app's ./bin directory.
|
||||
def ensure_local_bin_for(*gems)
|
||||
if bin_dir && File.directory?(bin_dir)
|
||||
gems.each do |gem|
|
||||
if gemspec_path = Dir[File.join(gem_dir, 'specifications', "#{gem}-*.gemspec")].last
|
||||
spec = Gem::Specification.load(gemspec_path)
|
||||
spec.executables.each do |exec|
|
||||
if File.exists?(executable = File.join(gem_dir, 'bin', exec))
|
||||
local_executable = File.join(bin_dir, exec)
|
||||
puts "Adding local executable #{local_executable}"
|
||||
File.open(local_executable, 'w', 0755) do |f|
|
||||
f.write(executable_wrapper(spec, exec))
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
def executable_wrapper(spec, bin_file_name)
|
||||
<<-TEXT
|
||||
#!/usr/bin/env #{RbConfig::CONFIG["ruby_install_name"]}
|
||||
#
|
||||
# This file was generated by merb.thor.
|
||||
#
|
||||
# The application '#{spec.name}' is installed as part of a gem, and
|
||||
# this file is here to facilitate running it.
|
||||
#
|
||||
|
||||
require 'rubygems'
|
||||
|
||||
if File.directory?(gems_dir = File.join(File.dirname(__FILE__), '..', 'gems'))
|
||||
$BUNDLE = true; Gem.clear_paths; Gem.path.unshift(gems_dir)
|
||||
end
|
||||
|
||||
version = "#{Gem::Requirement.default}"
|
||||
|
||||
if ARGV.first =~ /^_(.*)_$/ and Gem::Version.correct? $1 then
|
||||
version = $1
|
||||
ARGV.shift
|
||||
end
|
||||
|
||||
gem '#{spec.name}', version
|
||||
load '#{bin_file_name}'
|
||||
TEXT
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
# TODO
|
||||
# - a task to figure out an app's dependencies
|
||||
# - pulling a specific UUID/Tag (gitspec hash) with clone/update
|
||||
# - a 'deploy' task (in addition to 'redeploy' ?)
|
||||
# - eventually take a --orm option for the 'merb-stack' type of tasks
|
||||
|
||||
class Merb < Thor
|
||||
|
||||
class SourcePathMissing < Exception
|
||||
end
|
||||
|
||||
class GemPathMissing < Exception
|
||||
end
|
||||
|
||||
class GemInstallError < Exception
|
||||
end
|
||||
|
||||
class GemUninstallError < Exception
|
||||
end
|
||||
|
||||
# Install a Merb stack from stable RubyForge gems. Optionally install a
|
||||
# suitable Rack adapter/server when setting --adapter to one of the
|
||||
# following: mongrel, emongrel, thin or ebb.
|
||||
|
||||
desc 'stable', 'Install extlib, merb-core and merb-more from rubygems'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--adapter" => :optional
|
||||
def stable
|
||||
adapters = %w[mongrel emongrel thin ebb]
|
||||
stable = Stable.new
|
||||
stable.options = options
|
||||
if stable.core && stable.more
|
||||
puts "Installed extlib, merb-core and merb-more"
|
||||
if options[:adapter] && adapters.include?(options[:adapter]) &&
|
||||
stable.refresh_from_gems(options[:adapter])
|
||||
puts "Installed #{options[:adapter]}"
|
||||
elsif options[:adapter]
|
||||
puts "Please specify one of the following adapters: #{adapters.join(' ')}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
class Stable < Thor
|
||||
|
||||
# The Stable tasks deal with known -stable- gems; available
|
||||
# as shortcuts to Merb and DataMapper gems.
|
||||
#
|
||||
# These are pulled from rubyforge and installed into into the
|
||||
# desired gems dir (either system-wide or into the application's
|
||||
# gems directory).
|
||||
|
||||
include MerbThorHelper
|
||||
|
||||
# Gets latest gem versions from RubyForge and installs them.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:edge:core
|
||||
# thor merb:edge:core --merb-root ./path/to/your/app
|
||||
# thor merb:edge:core --sources ./path/to/sources.yml
|
||||
|
||||
desc 'core', 'Install extlib and merb-core from git HEAD'
|
||||
method_options "--merb-root" => :optional
|
||||
def core
|
||||
refresh_from_gems 'extlib', 'merb-core'
|
||||
ensure_local_bin_for('merb-core', 'rake', 'rspec', 'thor')
|
||||
end
|
||||
|
||||
desc 'more', 'Install merb-more from rubygems'
|
||||
method_options "--merb-root" => :optional
|
||||
def more
|
||||
refresh_from_gems 'merb-more'
|
||||
ensure_local_bin_for('merb-gen')
|
||||
end
|
||||
|
||||
desc 'plugins', 'Install merb-plugins from rubygems'
|
||||
method_options "--merb-root" => :optional
|
||||
def plugins
|
||||
refresh_from_gems 'merb-plugins'
|
||||
end
|
||||
|
||||
desc 'dm_core', 'Install dm-core from rubygems'
|
||||
method_options "--merb-root" => :optional
|
||||
def dm_core
|
||||
refresh_from_gems 'extlib', 'dm-core'
|
||||
end
|
||||
|
||||
desc 'dm_more', 'Install dm-more from rubygems'
|
||||
method_options "--merb-root" => :optional
|
||||
def dm_more
|
||||
refresh_from_gems 'extlib', 'dm-core', 'dm-more'
|
||||
end
|
||||
|
||||
# Pull from RubyForge and install.
|
||||
def refresh_from_gems(*components)
|
||||
gems = Gems.new
|
||||
gems.options = options
|
||||
components.all? { |name| gems.install(name) }
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
# Retrieve latest Merb versions from git and optionally install them.
|
||||
#
|
||||
# Note: the --sources option takes a path to a YAML file
|
||||
# with a regular Hash mapping gem names to git urls.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:edge
|
||||
# thor merb:edge --install
|
||||
# thor merb:edge --merb-root ./path/to/your/app
|
||||
# thor merb:edge --sources ./path/to/sources.yml
|
||||
|
||||
desc 'edge', 'Install extlib, merb-core and merb-more from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def edge
|
||||
edge = Edge.new
|
||||
edge.options = options
|
||||
edge.core
|
||||
edge.more
|
||||
end
|
||||
|
||||
class Edge < Thor
|
||||
|
||||
# The Edge tasks deal with known gems from the bleeding edge; available
|
||||
# as shortcuts to Merb and DataMapper gems.
|
||||
#
|
||||
# These are pulled from git and optionally installed into into the
|
||||
# desired gems dir (either system-wide or into the application's
|
||||
# gems directory).
|
||||
|
||||
include MerbThorHelper
|
||||
|
||||
# Gets latest gem versions from git - optionally installs them.
|
||||
#
|
||||
# Note: the --sources option takes a path to a YAML file
|
||||
# with a regular Hash mapping gem names to git urls,
|
||||
# allowing pulling forks of the official repositories.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:edge:core
|
||||
# thor merb:edge:core --install
|
||||
# thor merb:edge:core --merb-root ./path/to/your/app
|
||||
# thor merb:edge:core --sources ./path/to/sources.yml
|
||||
|
||||
desc 'core', 'Update extlib and merb-core from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def core
|
||||
refresh_from_source 'thor', 'extlib', 'merb-core'
|
||||
ensure_local_bin_for('merb-core', 'rake', 'rspec', 'thor')
|
||||
end
|
||||
|
||||
desc 'more', 'Update merb-more from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def more
|
||||
refresh_from_source 'merb-more'
|
||||
ensure_local_bin_for('merb-gen')
|
||||
end
|
||||
|
||||
desc 'plugins', 'Update merb-plugins from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def plugins
|
||||
refresh_from_source 'merb-plugins'
|
||||
end
|
||||
|
||||
desc 'dm_core', 'Update dm-core from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def dm_core
|
||||
refresh_from_source 'extlib', 'dm-core'
|
||||
end
|
||||
|
||||
desc 'dm_more', 'Update dm-more from git HEAD'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--sources" => :optional,
|
||||
"--install" => :boolean
|
||||
def dm_more
|
||||
refresh_from_source 'extlib', 'dm-core', 'dm-more'
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
# Pull from git and optionally install the resulting gems.
|
||||
def refresh_from_source(*components)
|
||||
source = Source.new
|
||||
source.options = options
|
||||
components.each do |name|
|
||||
source.clone(name)
|
||||
source.install(name) if options[:install]
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
class Source < Thor
|
||||
|
||||
# The Source tasks deal with gem source packages - mainly from github.
|
||||
# Any directory inside ./src is regarded as a gem that can be packaged
|
||||
# and installed from there into the desired gems dir (either system-wide
|
||||
# or into the application's gems directory).
|
||||
|
||||
include MerbThorHelper
|
||||
|
||||
# Install a particular gem from source.
|
||||
#
|
||||
# If a local ./gems dir is found, or --merb-root is given
|
||||
# the gems will be installed locally into that directory.
|
||||
#
|
||||
# Note that this task doesn't retrieve any (new) source from git;
|
||||
# To update and install you'd execute the following two tasks:
|
||||
#
|
||||
# thor merb:source:update merb-core
|
||||
# thor merb:source:install merb-core
|
||||
#
|
||||
# Alternatively, look at merb:edge and merb:edge:* with --install.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:source:install merb-core
|
||||
# thor merb:source:install merb-more
|
||||
# thor merb:source:install merb-more/merb-slices
|
||||
# thor merb:source:install merb-plugins/merb_helpers
|
||||
# thor merb:source:install merb-core --merb-root ./path/to/your/app
|
||||
|
||||
desc 'install GEM_NAME', 'Install a rubygem from (git) source'
|
||||
method_options "--merb-root" => :optional
|
||||
def install(name)
|
||||
puts "Installing #{name}..."
|
||||
gem_src_dir = File.join(source_dir, name)
|
||||
opts = {}
|
||||
opts[:install_dir] = gem_dir if gem_dir
|
||||
Merb.install_gem_from_src(gem_src_dir, opts)
|
||||
rescue Merb::SourcePathMissing
|
||||
puts "Missing rubygem source path: #{gem_src_dir}"
|
||||
rescue Merb::GemPathMissing
|
||||
puts "Missing rubygems path: #{gem_dir}"
|
||||
rescue => e
|
||||
puts "Failed to install #{name} (#{e.message})"
|
||||
end
|
||||
|
||||
# Clone a git repository into ./src. The repository can be
|
||||
# a direct git url or a known -named- repository.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:source:clone dm-core
|
||||
# thor merb:source:clone dm-core --sources ./path/to/sources.yml
|
||||
# thor merb:source:clone git://github.com/sam/dm-core.git
|
||||
|
||||
desc 'clone REPOSITORY', 'Clone a git repository into ./src'
|
||||
method_options "--sources" => :optional
|
||||
def clone(repository)
|
||||
if repository =~ /^git:\/\//
|
||||
repository_url = repository
|
||||
elsif url = Merb.repos(options[:sources])[repository]
|
||||
repository_url = url
|
||||
end
|
||||
|
||||
if repository_url
|
||||
repository_name = repository_url[/([\w+|-]+)\.git/u, 1]
|
||||
fork_name = repository_url[/.com\/+?(.+)\/.+\.git/u, 1]
|
||||
local_repo_path = "#{source_dir}/#{repository_name}"
|
||||
|
||||
if File.directory?(local_repo_path)
|
||||
puts "\n#{repository_name} repository exists, updating or branching instead of cloning..."
|
||||
FileUtils.cd(local_repo_path) do
|
||||
|
||||
# to avoid conflicts we need to set a remote branch for non official repos
|
||||
existing_repos = `git remote -v`.split("\n").map{|branch| branch.split(/\s+/)}
|
||||
origin_repo_url = existing_repos.detect{ |r| r.first == "origin" }.last
|
||||
|
||||
# pull from the original repository - no branching needed
|
||||
if repository_url == origin_repo_url
|
||||
puts "Pulling from #{repository_url}"
|
||||
system %{
|
||||
git fetch
|
||||
git checkout master
|
||||
git rebase origin/master
|
||||
}
|
||||
# update and switch to a branch for a particular github fork
|
||||
elsif existing_repos.map{ |r| r.last }.include?(repository_url)
|
||||
puts "Switching to remote branch: #{fork_name}"
|
||||
`git checkout -b #{fork_name} #{fork_name}/master`
|
||||
`git rebase #{fork_name}/master`
|
||||
# create a new remote branch for a particular github fork
|
||||
else
|
||||
puts "Add a new remote branch: #{fork_name}"
|
||||
`git remote add -f #{fork_name} #{repository_url}`
|
||||
`git checkout -b#{fork_name} #{fork_name}/master`
|
||||
end
|
||||
end
|
||||
else
|
||||
FileUtils.cd(source_dir) do
|
||||
puts "\nCloning #{repository_name} repository from #{repository_url}..."
|
||||
system("git clone --depth=1 #{repository_url} ")
|
||||
end
|
||||
end
|
||||
else
|
||||
puts "No valid repository url given"
|
||||
end
|
||||
end
|
||||
|
||||
# Update a specific gem source directory from git. See #clone.
|
||||
|
||||
desc 'update REPOSITORY_URL', 'Update a git repository in ./src'
|
||||
alias :update :clone
|
||||
|
||||
# Update all gem sources from git - based on the current branch.
|
||||
|
||||
desc 'refresh', 'Pull fresh copies of all source gems'
|
||||
def refresh
|
||||
repos = Dir["#{source_dir}/*"]
|
||||
repos.each do |repo|
|
||||
next unless File.directory?(repo) && File.exists?(File.join(repo, '.git'))
|
||||
FileUtils.cd(repo) do
|
||||
puts "Refreshing #{File.basename(repo)}"
|
||||
branch = `git branch --no-color 2> /dev/null | sed -e '/^[^*]/d' -e 's/* \(.*\)/(\1) /'`[/\* (.+)/, 1]
|
||||
system %{git rebase #{branch}}
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
class Gems < Thor
|
||||
|
||||
# The Gems tasks deal directly with rubygems, either through remotely
|
||||
# available sources (rubyforge for example) or by searching the
|
||||
# system-wide gem cache for matching gems. The gems are installed from
|
||||
# there into the desired gems dir (either system-wide or into the
|
||||
# application's gems directory).
|
||||
|
||||
include MerbThorHelper
|
||||
|
||||
# Install a gem and its dependencies.
|
||||
#
|
||||
# If a local ./gems dir is found, or --merb-root is given
|
||||
# the gems will be installed locally into that directory.
|
||||
#
|
||||
# The option --cache will look in the system's gem cache
|
||||
# for the latest version and install it in the apps' gems.
|
||||
# This is particularly handy for gems that aren't available
|
||||
# through rubyforge.org - like in-house merb slices etc.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:gems:install merb-core
|
||||
# thor merb:gems:install merb-core --cache
|
||||
# thor merb:gems:install merb-core --version 0.9.7
|
||||
# thor merb:gems:install merb-core --merb-root ./path/to/your/app
|
||||
|
||||
desc 'install GEM_NAME', 'Install a gem from rubygems'
|
||||
method_options "--version" => :optional,
|
||||
"--merb-root" => :optional,
|
||||
"--cache" => :boolean
|
||||
def install(name)
|
||||
puts "Installing #{name}..."
|
||||
opts = {}
|
||||
opts[:version] = options[:version]
|
||||
opts[:cache] = options[:cache] if gem_dir
|
||||
opts[:install_dir] = gem_dir if gem_dir
|
||||
Merb.install_gem(name, opts)
|
||||
rescue => e
|
||||
puts "Failed to install #{name} (#{e.message})"
|
||||
end
|
||||
|
||||
# Update a gem and its dependencies.
|
||||
#
|
||||
# If a local ./gems dir is found, or --merb-root is given
|
||||
# the gems will be installed locally into that directory.
|
||||
#
|
||||
# The option --cache will look in the system's gem cache
|
||||
# for the latest version and install it in the apps' gems.
|
||||
# This is particularly handy for gems that aren't available
|
||||
# through rubyforge.org - like in-house merb slices etc.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:gems:update merb-core
|
||||
# thor merb:gems:update merb-core --cache
|
||||
# thor merb:gems:update merb-core --merb-root ./path/to/your/app
|
||||
|
||||
desc 'update GEM_NAME', 'Update a gem from rubygems'
|
||||
method_options "--merb-root" => :optional,
|
||||
"--cache" => :boolean
|
||||
def update(name)
|
||||
puts "Updating #{name}..."
|
||||
opts = {}
|
||||
if gem_dir
|
||||
if gemspec_path = Dir[File.join(gem_dir, 'specifications', "#{name}-*.gemspec")].last
|
||||
gemspec = Gem::Specification.load(gemspec_path)
|
||||
opts[:version] = Gem::Requirement.new [">#{gemspec.version}"]
|
||||
end
|
||||
opts[:install_dir] = gem_dir
|
||||
opts[:cache] = options[:cache]
|
||||
end
|
||||
Merb.install_gem(name, opts)
|
||||
rescue => e
|
||||
puts "Failed to update #{name} (#{e.message})"
|
||||
end
|
||||
|
||||
# Uninstall a gem - ignores dependencies.
|
||||
#
|
||||
# If a local ./gems dir is found, or --merb-root is given
|
||||
# the gems will be uninstalled locally from that directory.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:gems:uninstall merb-core
|
||||
# thor merb:gems:uninstall merb-core --all
|
||||
# thor merb:gems:uninstall merb-core --version 0.9.7
|
||||
# thor merb:gems:uninstall merb-core --merb-root ./path/to/your/app
|
||||
|
||||
desc 'install GEM_NAME', 'Install a gem from rubygems'
|
||||
desc 'uninstall GEM_NAME', 'Uninstall a gem'
|
||||
method_options "--version" => :optional,
|
||||
"--merb-root" => :optional,
|
||||
"--all" => :boolean
|
||||
def uninstall(name)
|
||||
puts "Uninstalling #{name}..."
|
||||
opts = {}
|
||||
opts[:ignore] = true
|
||||
opts[:all] = options[:all]
|
||||
opts[:executables] = true
|
||||
opts[:version] = options[:version]
|
||||
opts[:install_dir] = gem_dir if gem_dir
|
||||
Merb.uninstall_gem(name, opts)
|
||||
rescue => e
|
||||
puts "Failed to uninstall #{name} (#{e.message})"
|
||||
end
|
||||
|
||||
# Completely remove a gem and all its versions - ignores dependencies.
|
||||
#
|
||||
# If a local ./gems dir is found, or --merb-root is given
|
||||
# the gems will be uninstalled locally from that directory.
|
||||
#
|
||||
# Examples:
|
||||
#
|
||||
# thor merb:gems:wipe merb-core
|
||||
# thor merb:gems:wipe merb-core --merb-root ./path/to/your/app
|
||||
|
||||
desc 'wipe GEM_NAME', 'Remove a gem completely'
|
||||
method_options "--merb-root" => :optional
|
||||
def wipe(name)
|
||||
puts "Wiping #{name}..."
|
||||
opts = {}
|
||||
opts[:ignore] = true
|
||||
opts[:all] = true
|
||||
opts[:executables] = true
|
||||
opts[:install_dir] = gem_dir if gem_dir
|
||||
Merb.uninstall_gem(name, opts)
|
||||
rescue => e
|
||||
puts "Failed to wipe #{name} (#{e.message})"
|
||||
end
|
||||
|
||||
# This task should be executed as part of a deployment setup, where
|
||||
# the deployment system runs this after the app has been installed.
|
||||
# Usually triggered by Capistrano, God...
|
||||
#
|
||||
# It will regenerate gems from the bundled gems cache for any gem
|
||||
# that has C extensions - which need to be recompiled for the target
|
||||
# deployment platform.
|
||||
|
||||
desc 'redeploy', 'Recreate any binary gems on the target deployment platform'
|
||||
def redeploy
|
||||
require 'tempfile' # for
|
||||
if File.directory?(specs_dir = File.join(gem_dir, 'specifications')) &&
|
||||
File.directory?(cache_dir = File.join(gem_dir, 'cache'))
|
||||
Dir[File.join(specs_dir, '*.gemspec')].each do |gemspec_path|
|
||||
unless (gemspec = Gem::Specification.load(gemspec_path)).extensions.empty?
|
||||
if File.exists?(gem_file = File.join(cache_dir, "#{gemspec.full_name}.gem"))
|
||||
gem_file_copy = File.join(Dir::tmpdir, File.basename(gem_file))
|
||||
# Copy the gem to a temporary file, because otherwise RubyGems/FileUtils
|
||||
# will complain about copying identical files (same source/destination).
|
||||
FileUtils.cp(gem_file, gem_file_copy)
|
||||
Merb.install_gem(gem_file_copy, :install_dir => gem_dir)
|
||||
File.delete(gem_file_copy)
|
||||
end
|
||||
end
|
||||
end
|
||||
else
|
||||
puts "No application local gems directory found"
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
class << self
|
||||
|
||||
# Default Git repositories - pass source_config option
|
||||
# to load a yaml configuration file.
|
||||
def repos(source_config = nil)
|
||||
@_repos ||= begin
|
||||
repositories = {
|
||||
'merb-core' => "git://github.com/wycats/merb-core.git",
|
||||
'merb-more' => "git://github.com/wycats/merb-more.git",
|
||||
'merb-plugins' => "git://github.com/wycats/merb-plugins.git",
|
||||
'extlib' => "git://github.com/sam/extlib.git",
|
||||
'dm-core' => "git://github.com/sam/dm-core.git",
|
||||
'dm-more' => "git://github.com/sam/dm-more.git",
|
||||
'thor' => "git://github.com/wycats/thor.git"
|
||||
}
|
||||
end
|
||||
if source_config && File.exists?(source_config)
|
||||
@_repos.merge(YAML.load(File.read(source_config)))
|
||||
else
|
||||
@_repos
|
||||
end
|
||||
end
|
||||
|
||||
# Install a gem - looks remotely and local gem cache;
|
||||
# won't process rdoc or ri options.
|
||||
def install_gem(gem, options = {})
|
||||
from_cache = (options.key?(:cache) && options.delete(:cache))
|
||||
if from_cache
|
||||
install_gem_from_cache(gem, options)
|
||||
else
|
||||
version = options.delete(:version)
|
||||
Gem.configuration.update_sources = false
|
||||
installer = Gem::DependencyInstaller.new(options.merge(:user_install => false))
|
||||
exception = nil
|
||||
begin
|
||||
installer.install gem, version
|
||||
rescue Gem::InstallError => e
|
||||
exception = e
|
||||
rescue Gem::GemNotFoundException => e
|
||||
if from_cache && gem_file = find_gem_in_cache(gem, version)
|
||||
puts "Located #{gem} in gem cache..."
|
||||
installer.install gem_file
|
||||
else
|
||||
exception = e
|
||||
end
|
||||
rescue => e
|
||||
exception = e
|
||||
end
|
||||
if installer.installed_gems.empty? && exception
|
||||
puts "Failed to install gem '#{gem}' (#{exception.message})"
|
||||
end
|
||||
installer.installed_gems.each do |spec|
|
||||
puts "Successfully installed #{spec.full_name}"
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
# Install a gem - looks in the system's gem cache instead of remotely;
|
||||
# won't process rdoc or ri options.
|
||||
def install_gem_from_cache(gem, options = {})
|
||||
version = options.delete(:version)
|
||||
Gem.configuration.update_sources = false
|
||||
installer = Gem::DependencyInstaller.new(options.merge(:user_install => false))
|
||||
exception = nil
|
||||
begin
|
||||
if gem_file = find_gem_in_cache(gem, version)
|
||||
puts "Located #{gem} in gem cache..."
|
||||
installer.install gem_file
|
||||
else
|
||||
raise Gem::InstallError, "Unknown gem #{gem}"
|
||||
end
|
||||
rescue Gem::InstallError => e
|
||||
exception = e
|
||||
end
|
||||
if installer.installed_gems.empty? && exception
|
||||
puts "Failed to install gem '#{gem}' (#{e.message})"
|
||||
end
|
||||
installer.installed_gems.each do |spec|
|
||||
puts "Successfully installed #{spec.full_name}"
|
||||
end
|
||||
end
|
||||
|
||||
# Install a gem from source - builds and packages it first then installs it.
|
||||
def install_gem_from_src(gem_src_dir, options = {})
|
||||
raise SourcePathMissing unless File.directory?(gem_src_dir)
|
||||
raise GemPathMissing if options[:install_dir] && !File.directory?(options[:install_dir])
|
||||
|
||||
gem_name = File.basename(gem_src_dir)
|
||||
gem_pkg_dir = File.expand_path(File.join(gem_src_dir, 'pkg'))
|
||||
|
||||
# We need to use local bin executables if available.
|
||||
thor = which('thor')
|
||||
rake = which('rake')
|
||||
|
||||
# Handle pure Thor installation instead of Rake
|
||||
if File.exists?(File.join(gem_src_dir, 'Thorfile'))
|
||||
# Remove any existing packages.
|
||||
FileUtils.rm_rf(gem_pkg_dir) if File.directory?(gem_pkg_dir)
|
||||
# Create the package.
|
||||
FileUtils.cd(gem_src_dir) { system("#{thor} :package") }
|
||||
# Install the package using rubygems.
|
||||
if package = Dir[File.join(gem_pkg_dir, "#{gem_name}-*.gem")].last
|
||||
FileUtils.cd(File.dirname(package)) do
|
||||
install_gem(File.basename(package), options.dup)
|
||||
return
|
||||
end
|
||||
else
|
||||
raise Merb::GemInstallError, "No package found for #{gem_name}"
|
||||
end
|
||||
# Handle standard installation through Rake
|
||||
else
|
||||
# Clean and regenerate any subgems for meta gems.
|
||||
Dir[File.join(gem_src_dir, '*', 'Rakefile')].each do |rakefile|
|
||||
FileUtils.cd(File.dirname(rakefile)) { system("#{rake} clobber_package; #{rake} package") }
|
||||
end
|
||||
|
||||
# Handle the main gem install.
|
||||
if File.exists?(File.join(gem_src_dir, 'Rakefile'))
|
||||
# Remove any existing packages.
|
||||
FileUtils.cd(gem_src_dir) { system("#{rake} clobber_package") }
|
||||
# Create the main gem pkg dir if it doesn't exist.
|
||||
FileUtils.mkdir_p(gem_pkg_dir) unless File.directory?(gem_pkg_dir)
|
||||
# Copy any subgems to the main gem pkg dir.
|
||||
Dir[File.join(gem_src_dir, '**', 'pkg', '*.gem')].each do |subgem_pkg|
|
||||
FileUtils.cp(subgem_pkg, gem_pkg_dir)
|
||||
end
|
||||
|
||||
# Finally generate the main package and install it; subgems
|
||||
# (dependencies) are local to the main package.
|
||||
FileUtils.cd(gem_src_dir) do
|
||||
system("#{rake} package")
|
||||
if package = Dir[File.join(gem_pkg_dir, "#{gem_name}-*.gem")].last
|
||||
FileUtils.cd(File.dirname(package)) do
|
||||
install_gem(File.basename(package), options.dup)
|
||||
return
|
||||
end
|
||||
else
|
||||
raise Merb::GemInstallError, "No package found for #{gem_name}"
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
raise Merb::GemInstallError, "No Rakefile found for #{gem_name}"
|
||||
end
|
||||
|
||||
# Uninstall a gem.
|
||||
def uninstall_gem(gem, options = {})
|
||||
if options[:version] && !options[:version].is_a?(Gem::Requirement)
|
||||
options[:version] = Gem::Requirement.new ["= #{version}"]
|
||||
end
|
||||
begin
|
||||
Gem::Uninstaller.new(gem, options).uninstall
|
||||
rescue => e
|
||||
raise GemUninstallError, "Failed to uninstall #{gem}"
|
||||
end
|
||||
end
|
||||
|
||||
# Will prepend sudo on a suitable platform.
|
||||
def sudo
|
||||
@_sudo ||= begin
|
||||
windows = PLATFORM =~ /win32|cygwin/ rescue nil
|
||||
windows ? "" : "sudo "
|
||||
end
|
||||
end
|
||||
|
||||
# Use the local bin/* executables if available.
|
||||
def which(executable)
|
||||
if File.executable?(exec = File.join(Dir.pwd, 'bin', executable))
|
||||
exec
|
||||
else
|
||||
executable
|
||||
end
|
||||
end
|
||||
|
||||
private
|
||||
|
||||
def find_gem_in_cache(gem, version)
|
||||
spec = if version
|
||||
version = Gem::Requirement.new ["= #{version}"] unless version.is_a?(Gem::Requirement)
|
||||
Gem.source_index.find_name(gem, version).first
|
||||
else
|
||||
Gem.source_index.find_name(gem).sort_by { |g| g.version }.last
|
||||
end
|
||||
if spec && File.exists?(gem_file = "#{spec.installation_path}/cache/#{spec.full_name}.gem")
|
||||
gem_file
|
||||
end
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
class Tasks < Thor
|
||||
|
||||
include MerbThorHelper
|
||||
|
||||
# Install Thor, Rake and RSpec into the local gems dir, by copying it from
|
||||
# the system-wide rubygems cache - which is OK since we needed it to run
|
||||
# this task already.
|
||||
#
|
||||
# After this we don't need the system-wide rubygems anymore, as all required
|
||||
# executables are available in the local ./bin directory.
|
||||
#
|
||||
# RSpec is needed here because source installs might fail when running
|
||||
# rake tasks where spec/rake/spectask has been required.
|
||||
|
||||
desc 'setup', 'Install Thor, Rake and RSpec in the local gems dir'
|
||||
method_options "--merb-root" => :optional
|
||||
def setup
|
||||
if $0 =~ /^(\.\/)?bin\/thor$/
|
||||
puts "You cannot run the setup from #{$0} - try #{File.basename($0)} merb:tasks:setup instead"
|
||||
return
|
||||
end
|
||||
create_if_missing(File.join(working_dir, 'gems'))
|
||||
Merb.install_gem('thor', :cache => true, :install_dir => gem_dir)
|
||||
Merb.install_gem('rake', :cache => true, :install_dir => gem_dir)
|
||||
Merb.install_gem('rspec', :cache => true, :install_dir => gem_dir)
|
||||
ensure_local_bin_for('thor', 'rake', 'rspec')
|
||||
end
|
||||
|
||||
end
|
||||
|
||||
end
|
After Width: | Height: | Size: 4.1 KiB |
After Width: | Height: | Size: 8.9 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 626 B |
After Width: | Height: | Size: 5.7 KiB |
After Width: | Height: | Size: 1005 B |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 1.2 KiB |
After Width: | Height: | Size: 1.1 KiB |
After Width: | Height: | Size: 990 B |
After Width: | Height: | Size: 1002 B |
|
@ -0,0 +1,28 @@
|
|||
function addLoadEvent(func)
|
||||
{
|
||||
var oldonload = window.onload;
|
||||
if (typeof window.onload != 'function')
|
||||
{
|
||||
window.onload = func;
|
||||
}
|
||||
else
|
||||
{
|
||||
window.onload = function()
|
||||
{
|
||||
oldonload();
|
||||
func();
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
function hide_flashes()
|
||||
{
|
||||
addLoadEvent(function(){
|
||||
setTimeout('hide_flashes_timer()', 30000);
|
||||
});
|
||||
}
|
||||
|
||||
function hide_flashes_timer()
|
||||
{
|
||||
new Effect.DropOut($('flash_container'));
|
||||
}
|
|
@ -0,0 +1,974 @@
|
|||
// script.aculo.us dragdrop.js v1.8.1, Thu Jan 03 22:07:12 -0500 2008
|
||||
|
||||
// Copyright (c) 2005-2007 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us)
|
||||
// (c) 2005-2007 Sammi Williams (http://www.oriontransfer.co.nz, sammi@oriontransfer.co.nz)
|
||||
//
|
||||
// script.aculo.us is freely distributable under the terms of an MIT-style license.
|
||||
// For details, see the script.aculo.us web site: http://script.aculo.us/
|
||||
|
||||
if(Object.isUndefined(Effect))
|
||||
throw("dragdrop.js requires including script.aculo.us' effects.js library");
|
||||
|
||||
var Droppables = {
|
||||
drops: [],
|
||||
|
||||
remove: function(element) {
|
||||
this.drops = this.drops.reject(function(d) { return d.element==$(element) });
|
||||
},
|
||||
|
||||
add: function(element) {
|
||||
element = $(element);
|
||||
var options = Object.extend({
|
||||
greedy: true,
|
||||
hoverclass: null,
|
||||
tree: false
|
||||
}, arguments[1] || { });
|
||||
|
||||
// cache containers
|
||||
if(options.containment) {
|
||||
options._containers = [];
|
||||
var containment = options.containment;
|
||||
if(Object.isArray(containment)) {
|
||||
containment.each( function(c) { options._containers.push($(c)) });
|
||||
} else {
|
||||
options._containers.push($(containment));
|
||||
}
|
||||
}
|
||||
|
||||
if(options.accept) options.accept = [options.accept].flatten();
|
||||
|
||||
Element.makePositioned(element); // fix IE
|
||||
options.element = element;
|
||||
|
||||
this.drops.push(options);
|
||||
},
|
||||
|
||||
findDeepestChild: function(drops) {
|
||||
deepest = drops[0];
|
||||
|
||||
for (i = 1; i < drops.length; ++i)
|
||||
if (Element.isParent(drops[i].element, deepest.element))
|
||||
deepest = drops[i];
|
||||
|
||||
return deepest;
|
||||
},
|
||||
|
||||
isContained: function(element, drop) {
|
||||
var containmentNode;
|
||||
if(drop.tree) {
|
||||
containmentNode = element.treeNode;
|
||||
} else {
|
||||
containmentNode = element.parentNode;
|
||||
}
|
||||
return drop._containers.detect(function(c) { return containmentNode == c });
|
||||
},
|
||||
|
||||
isAffected: function(point, element, drop) {
|
||||
return (
|
||||
(drop.element!=element) &&
|
||||
((!drop._containers) ||
|
||||
this.isContained(element, drop)) &&
|
||||
((!drop.accept) ||
|
||||
(Element.classNames(element).detect(
|
||||
function(v) { return drop.accept.include(v) } ) )) &&
|
||||
Position.within(drop.element, point[0], point[1]) );
|
||||
},
|
||||
|
||||
deactivate: function(drop) {
|
||||
if(drop.hoverclass)
|
||||
Element.removeClassName(drop.element, drop.hoverclass);
|
||||
this.last_active = null;
|
||||
},
|
||||
|
||||
activate: function(drop) {
|
||||
if(drop.hoverclass)
|
||||
Element.addClassName(drop.element, drop.hoverclass);
|
||||
this.last_active = drop;
|
||||
},
|
||||
|
||||
show: function(point, element) {
|
||||
if(!this.drops.length) return;
|
||||
var drop, affected = [];
|
||||
|
||||
this.drops.each( function(drop) {
|
||||
if(Droppables.isAffected(point, element, drop))
|
||||
affected.push(drop);
|
||||
});
|
||||
|
||||
if(affected.length>0)
|
||||
drop = Droppables.findDeepestChild(affected);
|
||||
|
||||
if(this.last_active && this.last_active != drop) this.deactivate(this.last_active);
|
||||
if (drop) {
|
||||
Position.within(drop.element, point[0], point[1]);
|
||||
if(drop.onHover)
|
||||
drop.onHover(element, drop.element, Position.overlap(drop.overlap, drop.element));
|
||||
|
||||
if (drop != this.last_active) Droppables.activate(drop);
|
||||
}
|
||||
},
|
||||
|
||||
fire: function(event, element) {
|
||||
if(!this.last_active) return;
|
||||
Position.prepare();
|
||||
|
||||
if (this.isAffected([Event.pointerX(event), Event.pointerY(event)], element, this.last_active))
|
||||
if (this.last_active.onDrop) {
|
||||
this.last_active.onDrop(element, this.last_active.element, event);
|
||||
return true;
|
||||
}
|
||||
},
|
||||
|
||||
reset: function() {
|
||||
if(this.last_active)
|
||||
this.deactivate(this.last_active);
|
||||
}
|
||||
}
|
||||
|
||||
var Draggables = {
|
||||
drags: [],
|
||||
observers: [],
|
||||
|
||||
register: function(draggable) {
|
||||
if(this.drags.length == 0) {
|
||||
this.eventMouseUp = this.endDrag.bindAsEventListener(this);
|
||||
this.eventMouseMove = this.updateDrag.bindAsEventListener(this);
|
||||
this.eventKeypress = this.keyPress.bindAsEventListener(this);
|
||||
|
||||
Event.observe(document, "mouseup", this.eventMouseUp);
|
||||
Event.observe(document, "mousemove", this.eventMouseMove);
|
||||
Event.observe(document, "keypress", this.eventKeypress);
|
||||
}
|
||||
this.drags.push(draggable);
|
||||
},
|
||||
|
||||
unregister: function(draggable) {
|
||||
this.drags = this.drags.reject(function(d) { return d==draggable });
|
||||
if(this.drags.length == 0) {
|
||||
Event.stopObserving(document, "mouseup", this.eventMouseUp);
|
||||
Event.stopObserving(document, "mousemove", this.eventMouseMove);
|
||||
Event.stopObserving(document, "keypress", this.eventKeypress);
|
||||
}
|
||||
},
|
||||
|
||||
activate: function(draggable) {
|
||||
if(draggable.options.delay) {
|
||||
this._timeout = setTimeout(function() {
|
||||
Draggables._timeout = null;
|
||||
window.focus();
|
||||
Draggables.activeDraggable = draggable;
|
||||
}.bind(this), draggable.options.delay);
|
||||
} else {
|
||||
window.focus(); // allows keypress events if window isn't currently focused, fails for Safari
|
||||
this.activeDraggable = draggable;
|
||||
}
|
||||
},
|
||||
|
||||
deactivate: function() {
|
||||
this.activeDraggable = null;
|
||||
},
|
||||
|
||||
updateDrag: function(event) {
|
||||
if(!this.activeDraggable) return;
|
||||
var pointer = [Event.pointerX(event), Event.pointerY(event)];
|
||||
// Mozilla-based browsers fire successive mousemove events with
|
||||
// the same coordinates, prevent needless redrawing (moz bug?)
|
||||
if(this._lastPointer && (this._lastPointer.inspect() == pointer.inspect())) return;
|
||||
this._lastPointer = pointer;
|
||||
|
||||
this.activeDraggable.updateDrag(event, pointer);
|
||||
},
|
||||
|
||||
endDrag: function(event) {
|
||||
if(this._timeout) {
|
||||
clearTimeout(this._timeout);
|
||||
this._timeout = null;
|
||||
}
|
||||
if(!this.activeDraggable) return;
|
||||
this._lastPointer = null;
|
||||
this.activeDraggable.endDrag(event);
|
||||
this.activeDraggable = null;
|
||||
},
|
||||
|
||||
keyPress: function(event) {
|
||||
if(this.activeDraggable)
|
||||
this.activeDraggable.keyPress(event);
|
||||
},
|
||||
|
||||
addObserver: function(observer) {
|
||||
this.observers.push(observer);
|
||||
this._cacheObserverCallbacks();
|
||||
},
|
||||
|
||||
removeObserver: function(element) { // element instead of observer fixes mem leaks
|
||||
this.observers = this.observers.reject( function(o) { return o.element==element });
|
||||
this._cacheObserverCallbacks();
|
||||
},
|
||||
|
||||
notify: function(eventName, draggable, event) { // 'onStart', 'onEnd', 'onDrag'
|
||||
if(this[eventName+'Count'] > 0)
|
||||
this.observers.each( function(o) {
|
||||
if(o[eventName]) o[eventName](eventName, draggable, event);
|
||||
});
|
||||
if(draggable.options[eventName]) draggable.options[eventName](draggable, event);
|
||||
},
|
||||
|
||||
_cacheObserverCallbacks: function() {
|
||||
['onStart','onEnd','onDrag'].each( function(eventName) {
|
||||
Draggables[eventName+'Count'] = Draggables.observers.select(
|
||||
function(o) { return o[eventName]; }
|
||||
).length;
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
/*--------------------------------------------------------------------------*/
|
||||
|
||||
var Draggable = Class.create({
|
||||
initialize: function(element) {
|
||||
var defaults = {
|
||||
handle: false,
|
||||
reverteffect: function(element, top_offset, left_offset) {
|
||||
var dur = Math.sqrt(Math.abs(top_offset^2)+Math.abs(left_offset^2))*0.02;
|
||||
new Effect.Move(element, { x: -left_offset, y: -top_offset, duration: dur,
|
||||
queue: {scope:'_draggable', position:'end'}
|
||||
});
|
||||
},
|
||||
endeffect: function(element) {
|
||||
var toOpacity = Object.isNumber(element._opacity) ? element._opacity : 1.0;
|
||||
new Effect.Opacity(element, {duration:0.2, from:0.7, to:toOpacity,
|
||||
queue: {scope:'_draggable', position:'end'},
|
||||
afterFinish: function(){
|
||||
Draggable._dragging[element] = false
|
||||
}
|
||||
});
|
||||
},
|
||||
zindex: 1000,
|
||||
revert: false,
|
||||
quiet: false,
|
||||
scroll: false,
|
||||
scrollSensitivity: 20,
|
||||
scrollSpeed: 15,
|
||||
snap: false, // false, or xy or [x,y] or function(x,y){ return [x,y] }
|
||||
delay: 0
|
||||
};
|
||||
|
||||
if(!arguments[1] || Object.isUndefined(arguments[1].endeffect))
|
||||
Object.extend(defaults, {
|
||||
starteffect: function(element) {
|
||||
element._opacity = Element.getOpacity(element);
|
||||
Draggable._dragging[element] = true;
|
||||
new Effect.Opacity(element, {duration:0.2, from:element._opacity, to:0.7});
|
||||
}
|
||||
});
|
||||
|
||||
var options = Object.extend(defaults, arguments[1] || { });
|
||||
|
||||
this.element = $(element);
|
||||
|
||||
if(options.handle && Object.isString(options.handle))
|
||||
this.handle = this.element.down('.'+options.handle, 0);
|
||||
|
||||
if(!this.handle) this.handle = $(options.handle);
|
||||
if(!this.handle) this.handle = this.element;
|
||||
|
||||
if(options.scroll && !options.scroll.scrollTo && !options.scroll.outerHTML) {
|
||||
options.scroll = $(options.scroll);
|
||||
this._isScrollChild = Element.childOf(this.element, options.scroll);
|
||||
}
|
||||
|
||||
Element.makePositioned(this.element); // fix IE
|
||||
|
||||
this.options = options;
|
||||
this.dragging = false;
|
||||
|
||||
this.eventMouseDown = this.initDrag.bindAsEventListener(this);
|
||||
Event.observe(this.handle, "mousedown", this.eventMouseDown);
|
||||
|
||||
Draggables.register(this);
|
||||
},
|
||||
|
||||
destroy: function() {
|
||||
Event.stopObserving(this.handle, "mousedown", this.eventMouseDown);
|
||||
Draggables.unregister(this);
|
||||
},
|
||||
|
||||
currentDelta: function() {
|
||||
return([
|
||||
parseInt(Element.getStyle(this.element,'left') || '0'),
|
||||
parseInt(Element.getStyle(this.element,'top') || '0')]);
|
||||
},
|
||||
|
||||
initDrag: function(event) {
|
||||
if(!Object.isUndefined(Draggable._dragging[this.element]) &&
|
||||
Draggable._dragging[this.element]) return;
|
||||
if(Event.isLeftClick(event)) {
|
||||
// abort on form elements, fixes a Firefox issue
|
||||
var src = Event.element(event);
|
||||
if((tag_name = src.tagName.toUpperCase()) && (
|
||||
tag_name=='INPUT' ||
|
||||
tag_name=='SELECT' ||
|
||||
tag_name=='OPTION' ||
|
||||
tag_name=='BUTTON' ||
|
||||
tag_name=='TEXTAREA')) return;
|
||||
|
||||
var pointer = [Event.pointerX(event), Event.pointerY(event)];
|
||||
var pos = Position.cumulativeOffset(this.element);
|
||||
this.offset = [0,1].map( function(i) { return (pointer[i] - pos[i]) });
|
||||
|
||||
Draggables.activate(this);
|
||||
Event.stop(event);
|
||||
}
|
||||
},
|
||||
|
||||
startDrag: function(event) {
|
||||
this.dragging = true;
|
||||
if(!this.delta)
|
||||
this.delta = this.currentDelta();
|
||||
|
||||
if(this.options.zindex) {
|
||||
this.originalZ = parseInt(Element.getStyle(this.element,'z-index') || 0);
|
||||
this.element.style.zIndex = this.options.zindex;
|
||||
}
|
||||
|
||||
if(this.options.ghosting) {
|
||||
this._clone = this.element.cloneNode(true);
|
||||
this.element._originallyAbsolute = (this.element.getStyle('position') == 'absolute');
|
||||
if (!this.element._originallyAbsolute)
|
||||
Position.absolutize(this.element);
|
||||
this.element.parentNode.insertBefore(this._clone, this.element);
|
||||
}
|
||||
|
||||
if(this.options.scroll) {
|
||||
if (this.options.scroll == window) {
|
||||
var where = this._getWindowScroll(this.options.scroll);
|
||||
this.originalScrollLeft = where.left;
|
||||
this.originalScrollTop = where.top;
|
||||
} else {
|
||||
this.originalScrollLeft = this.options.scroll.scrollLeft;
|
||||
this.originalScrollTop = this.options.scroll.scrollTop;
|
||||
}
|
||||
}
|
||||
|
||||
Draggables.notify('onStart', this, event);
|
||||
|
||||
if(this.options.starteffect) this.options.starteffect(this.element);
|
||||
},
|
||||
|
||||
updateDrag: function(event, pointer) {
|
||||
if(!this.dragging) this.startDrag(event);
|
||||
|
||||
if(!this.options.quiet){
|
||||
Position.prepare();
|
||||
Droppables.show(pointer, this.element);
|
||||
}
|
||||
|
||||
Draggables.notify('onDrag', this, event);
|
||||
|
||||
this.draw(pointer);
|
||||
if(this.options.change) this.options.change(this);
|
||||
|
||||
if(this.options.scroll) {
|
||||
this.stopScrolling();
|
||||
|
||||
var p;
|
||||
if (this.options.scroll == window) {
|
||||
with(this._getWindowScroll(this.options.scroll)) { p = [ left, top, left+width, top+height ]; }
|
||||
} else {
|
||||
p = Position.page(this.options.scroll);
|
||||
p[0] += this.options.scroll.scrollLeft + Position.deltaX;
|
||||
p[1] += this.options.scroll.scrollTop + Position.deltaY;
|
||||
p.push(p[0]+this.options.scroll.offsetWidth);
|
||||
p.push(p[1]+this.options.scroll.offsetHeight);
|
||||
}
|
||||
var speed = [0,0];
|
||||
if(pointer[0] < (p[0]+this.options.scrollSensitivity)) speed[0] = pointer[0]-(p[0]+this.options.scrollSensitivity);
|
||||
if(pointer[1] < (p[1]+this.options.scrollSensitivity)) speed[1] = pointer[1]-(p[1]+this.options.scrollSensitivity);
|
||||
if(pointer[0] > (p[2]-this.options.scrollSensitivity)) speed[0] = pointer[0]-(p[2]-this.options.scrollSensitivity);
|
||||
if(pointer[1] > (p[3]-this.options.scrollSensitivity)) speed[1] = pointer[1]-(p[3]-this.options.scrollSensitivity);
|
||||
this.startScrolling(speed);
|
||||
}
|
||||
|
||||
// fix AppleWebKit rendering
|
||||
if(Prototype.Browser.WebKit) window.scrollBy(0,0);
|
||||
|
||||
Event.stop(event);
|
||||
},
|
||||
|
||||
finishDrag: function(event, success) {
|
||||
this.dragging = false;
|
||||
|
||||
if(this.options.quiet){
|
||||
Position.prepare();
|
||||
var pointer = [Event.pointerX(event), Event.pointerY(event)];
|
||||
Droppables.show(pointer, this.element);
|
||||
}
|
||||
|
||||
if(this.options.ghosting) {
|
||||
if (!this.element._originallyAbsolute)
|
||||
Position.relativize(this.element);
|
||||
delete this.element._originallyAbsolute;
|
||||
Element.remove(this._clone);
|
||||
this._clone = null;
|
||||
}
|
||||
|
||||
var dropped = false;
|
||||
if(success) {
|
||||
dropped = Droppables.fire(event, this.element);
|
||||
if (!dropped) dropped = false;
|
||||
}
|
||||
if(dropped && this.options.onDropped) this.options.onDropped(this.element);
|
||||
Draggables.notify('onEnd', this, event);
|
||||
|
||||
var revert = this.options.revert;
|
||||
if(revert && Object.isFunction(revert)) revert = revert(this.element);
|
||||
|
||||
var d = this.currentDelta();
|
||||
if(revert && this.options.reverteffect) {
|
||||
if (dropped == 0 || revert != 'failure')
|
||||
this.options.reverteffect(this.element,
|
||||
d[1]-this.delta[1], d[0]-this.delta[0]);
|
||||
} else {
|
||||
this.delta = d;
|
||||
}
|
||||
|
||||
if(this.options.zindex)
|
||||
this.element.style.zIndex = this.originalZ;
|
||||
|
||||
if(this.options.endeffect)
|
||||
this.options.endeffect(this.element);
|
||||
|
||||
Draggables.deactivate(this);
|
||||
Droppables.reset();
|
||||
},
|
||||
|
||||
keyPress: function(event) {
|
||||
if(event.keyCode!=Event.KEY_ESC) return;
|
||||
this.finishDrag(event, false);
|
||||
Event.stop(event);
|
||||
},
|
||||
|
||||
endDrag: function(event) {
|
||||
if(!this.dragging) return;
|
||||
this.stopScrolling();
|
||||
this.finishDrag(event, true);
|
||||
Event.stop(event);
|
||||
},
|
||||
|
||||
draw: function(point) {
|
||||
var pos = Position.cumulativeOffset(this.element);
|
||||
if(this.options.ghosting) {
|
||||
var r = Position.realOffset(this.element);
|
||||
pos[0] += r[0] - Position.deltaX; pos[1] += r[1] - Position.deltaY;
|
||||
}
|
||||
|
||||
var d = this.currentDelta();
|
||||
pos[0] -= d[0]; pos[1] -= d[1];
|
||||
|
||||
if(this.options.scroll && (this.options.scroll != window && this._isScrollChild)) {
|
||||
pos[0] -= this.options.scroll.scrollLeft-this.originalScrollLeft;
|
||||
pos[1] -= this.options.scroll.scrollTop-this.originalScrollTop;
|
||||
}
|
||||
|
||||
var p = [0,1].map(function(i){
|
||||
return (point[i]-pos[i]-this.offset[i])
|
||||
}.bind(this));
|
||||
|
||||
if(this.options.snap) {
|
||||
if(Object.isFunction(this.options.snap)) {
|
||||
p = this.options.snap(p[0],p[1],this);
|
||||
} else {
|
||||
if(Object.isArray(this.options.snap)) {
|
||||
p = p.map( function(v, i) {
|
||||
return (v/this.options.snap[i]).round()*this.options.snap[i] }.bind(this))
|
||||
} else {
|
||||
p = p.map( function(v) {
|
||||
return (v/this.options.snap).round()*this.options.snap }.bind(this))
|
||||
}
|
||||
}}
|
||||
|
||||
var style = this.element.style;
|
||||
if((!this.options.constraint) || (this.options.constraint=='horizontal'))
|
||||
style.left = p[0] + "px";
|
||||
if((!this.options.constraint) || (this.options.constraint=='vertical'))
|
||||
style.top = p[1] + "px";
|
||||
|
||||
if(style.visibility=="hidden") style.visibility = ""; // fix gecko rendering
|
||||
},
|
||||
|
||||
stopScrolling: function() {
|
||||
if(this.scrollInterval) {
|
||||
clearInterval(this.scrollInterval);
|
||||
this.scrollInterval = null;
|
||||
Draggables._lastScrollPointer = null;
|
||||
}
|
||||
},
|
||||
|
||||
startScrolling: function(speed) {
|
||||
if(!(speed[0] || speed[1])) return;
|
||||
this.scrollSpeed = [speed[0]*this.options.scrollSpeed,speed[1]*this.options.scrollSpeed];
|
||||
this.lastScrolled = new Date();
|
||||
this.scrollInterval = setInterval(this.scroll.bind(this), 10);
|
||||
},
|
||||
|
||||
scroll: function() {
|
||||
var current = new Date();
|
||||
var delta = current - this.lastScrolled;
|
||||
this.lastScrolled = current;
|
||||
if(this.options.scroll == window) {
|
||||
with (this._getWindowScroll(this.options.scroll)) {
|
||||
if (this.scrollSpeed[0] || this.scrollSpeed[1]) {
|
||||
var d = delta / 1000;
|
||||
this.options.scroll.scrollTo( left + d*this.scrollSpeed[0], top + d*this.scrollSpeed[1] );
|
||||
}
|
||||
}
|
||||
} else {
|
||||
this.options.scroll.scrollLeft += this.scrollSpeed[0] * delta / 1000;
|
||||
this.options.scroll.scrollTop += this.scrollSpeed[1] * delta / 1000;
|
||||
}
|
||||
|
||||
Position.prepare();
|
||||
Droppables.show(Draggables._lastPointer, this.element);
|
||||
Draggables.notify('onDrag', this);
|
||||
if (this._isScrollChild) {
|
||||
Draggables._lastScrollPointer = Draggables._lastScrollPointer || $A(Draggables._lastPointer);
|
||||
Draggables._lastScrollPointer[0] += this.scrollSpeed[0] * delta / 1000;
|
||||
Draggables._lastScrollPointer[1] += this.scrollSpeed[1] * delta / 1000;
|
||||
if (Draggables._lastScrollPointer[0] < 0)
|
||||
Draggables._lastScrollPointer[0] = 0;
|
||||
if (Draggables._lastScrollPointer[1] < 0)
|
||||
Draggables._lastScrollPointer[1] = 0;
|
||||
this.draw(Draggables._lastScrollPointer);
|
||||
}
|
||||
|
||||
if(this.options.change) this.options.change(this);
|
||||
},
|
||||
|
||||
_getWindowScroll: function(w) {
|
||||
var T, L, W, H;
|
||||
with (w.document) {
|
||||
if (w.document.documentElement && documentElement.scrollTop) {
|
||||
T = documentElement.scrollTop;
|
||||
L = documentElement.scrollLeft;
|
||||
} else if (w.document.body) {
|
||||
T = body.scrollTop;
|
||||
L = body.scrollLeft;
|
||||
}
|
||||
if (w.innerWidth) {
|
||||
W = w.innerWidth;
|
||||
H = w.innerHeight;
|
||||
} else if (w.document.documentElement && documentElement.clientWidth) {
|
||||
W = documentElement.clientWidth;
|
||||
H = documentElement.clientHeight;
|
||||
} else {
|
||||
W = body.offsetWidth;
|
||||
H = body.offsetHeight
|
||||
}
|
||||
}
|
||||
return { top: T, left: L, width: W, height: H };
|
||||
}
|
||||
});
|
||||
|
||||
Draggable._dragging = { };
|
||||
|
||||
/*--------------------------------------------------------------------------*/
|
||||
|
||||
var SortableObserver = Class.create({
|
||||
initialize: function(element, observer) {
|
||||
this.element = $(element);
|
||||
this.observer = observer;
|
||||
this.lastValue = Sortable.serialize(this.element);
|
||||
},
|
||||
|
||||
onStart: function() {
|
||||
this.lastValue = Sortable.serialize(this.element);
|
||||
},
|
||||
|
||||
onEnd: function() {
|
||||
Sortable.unmark();
|
||||
if(this.lastValue != Sortable.serialize(this.element))
|
||||
this.observer(this.element)
|
||||
}
|
||||
});
|
||||
|
||||
var Sortable = {
|
||||
SERIALIZE_RULE: /^[^_\-](?:[A-Za-z0-9\-\_]*)[_](.*)$/,
|
||||
|
||||
sortables: { },
|
||||
|
||||
_findRootElement: function(element) {
|
||||
while (element.tagName.toUpperCase() != "BODY") {
|
||||
if(element.id && Sortable.sortables[element.id]) return element;
|
||||
element = element.parentNode;
|
||||
}
|
||||
},
|
||||
|
||||
options: function(element) {
|
||||
element = Sortable._findRootElement($(element));
|
||||
if(!element) return;
|
||||
return Sortable.sortables[element.id];
|
||||
},
|
||||
|
||||
destroy: function(element){
|
||||
var s = Sortable.options(element);
|
||||
|
||||
if(s) {
|
||||
Draggables.removeObserver(s.element);
|
||||
s.droppables.each(function(d){ Droppables.remove(d) });
|
||||
s.draggables.invoke('destroy');
|
||||
|
||||
delete Sortable.sortables[s.element.id];
|
||||
}
|
||||
},
|
||||
|
||||
create: function(element) {
|
||||
element = $(element);
|
||||
var options = Object.extend({
|
||||
element: element,
|
||||
tag: 'li', // assumes li children, override with tag: 'tagname'
|
||||
dropOnEmpty: false,
|
||||
tree: false,
|
||||
treeTag: 'ul',
|
||||
overlap: 'vertical', // one of 'vertical', 'horizontal'
|
||||
constraint: 'vertical', // one of 'vertical', 'horizontal', false
|
||||
containment: element, // also takes array of elements (or id's); or false
|
||||
handle: false, // or a CSS class
|
||||
only: false,
|
||||
delay: 0,
|
||||
hoverclass: null,
|
||||
ghosting: false,
|
||||
quiet: false,
|
||||
scroll: false,
|
||||
scrollSensitivity: 20,
|
||||
scrollSpeed: 15,
|
||||
format: this.SERIALIZE_RULE,
|
||||
|
||||
// these take arrays of elements or ids and can be
|
||||
// used for better initialization performance
|
||||
elements: false,
|
||||
handles: false,
|
||||
|
||||
onChange: Prototype.emptyFunction,
|
||||
onUpdate: Prototype.emptyFunction
|
||||
}, arguments[1] || { });
|
||||
|
||||
// clear any old sortable with same element
|
||||
this.destroy(element);
|
||||
|
||||
// build options for the draggables
|
||||
var options_for_draggable = {
|
||||
revert: true,
|
||||
quiet: options.quiet,
|
||||
scroll: options.scroll,
|
||||
scrollSpeed: options.scrollSpeed,
|
||||
scrollSensitivity: options.scrollSensitivity,
|
||||
delay: options.delay,
|
||||
ghosting: options.ghosting,
|
||||
constraint: options.constraint,
|
||||
handle: options.handle };
|
||||
|
||||
if(options.starteffect)
|
||||
options_for_draggable.starteffect = options.starteffect;
|
||||
|
||||
if(options.reverteffect)
|
||||
options_for_draggable.reverteffect = options.reverteffect;
|
||||
else
|
||||
if(options.ghosting) options_for_draggable.reverteffect = function(element) {
|
||||
element.style.top = 0;
|
||||
element.style.left = 0;
|
||||
};
|
||||
|
||||
if(options.endeffect)
|
||||
options_for_draggable.endeffect = options.endeffect;
|
||||
|
||||
if(options.zindex)
|
||||
options_for_draggable.zindex = options.zindex;
|
||||
|
||||
// build options for the droppables
|
||||
var options_for_droppable = {
|
||||
overlap: options.overlap,
|
||||
containment: options.containment,
|
||||
tree: options.tree,
|
||||
hoverclass: options.hoverclass,
|
||||
onHover: Sortable.onHover
|
||||
}
|
||||
|
||||
var options_for_tree = {
|
||||
onHover: Sortable.onEmptyHover,
|
||||
overlap: options.overlap,
|
||||
containment: options.containment,
|
||||
hoverclass: options.hoverclass
|
||||
}
|
||||
|
||||
// fix for gecko engine
|
||||
Element.cleanWhitespace(element);
|
||||
|
||||
options.draggables = [];
|
||||
options.droppables = [];
|
||||
|
||||
// drop on empty handling
|
||||
if(options.dropOnEmpty || options.tree) {
|
||||
Droppables.add(element, options_for_tree);
|
||||
options.droppables.push(element);
|
||||
}
|
||||
|
||||
(options.elements || this.findElements(element, options) || []).each( function(e,i) {
|
||||
var handle = options.handles ? $(options.handles[i]) :
|
||||
(options.handle ? $(e).select('.' + options.handle)[0] : e);
|
||||
options.draggables.push(
|
||||
new Draggable(e, Object.extend(options_for_draggable, { handle: handle })));
|
||||
Droppables.add(e, options_for_droppable);
|
||||
if(options.tree) e.treeNode = element;
|
||||
options.droppables.push(e);
|
||||
});
|
||||
|
||||
if(options.tree) {
|
||||
(Sortable.findTreeElements(element, options) || []).each( function(e) {
|
||||
Droppables.add(e, options_for_tree);
|
||||
e.treeNode = element;
|
||||
options.droppables.push(e);
|
||||
});
|
||||
}
|
||||
|
||||
// keep reference
|
||||
this.sortables[element.id] = options;
|
||||
|
||||
// for onupdate
|
||||
Draggables.addObserver(new SortableObserver(element, options.onUpdate));
|
||||
|
||||
},
|
||||
|
||||
// return all suitable-for-sortable elements in a guaranteed order
|
||||
findElements: function(element, options) {
|
||||
return Element.findChildren(
|
||||
element, options.only, options.tree ? true : false, options.tag);
|
||||
},
|
||||
|
||||
findTreeElements: function(element, options) {
|
||||
return Element.findChildren(
|
||||
element, options.only, options.tree ? true : false, options.treeTag);
|
||||
},
|
||||
|
||||
onHover: function(element, dropon, overlap) {
|
||||
if(Element.isParent(dropon, element)) return;
|
||||
|
||||
if(overlap > .33 && overlap < .66 && Sortable.options(dropon).tree) {
|
||||
return;
|
||||
} else if(overlap>0.5) {
|
||||
Sortable.mark(dropon, 'before');
|
||||
if(dropon.previousSibling != element) {
|
||||
var oldParentNode = element.parentNode;
|
||||
element.style.visibility = "hidden"; // fix gecko rendering
|
||||
dropon.parentNode.insertBefore(element, dropon);
|
||||
if(dropon.parentNode!=oldParentNode)
|
||||
Sortable.options(oldParentNode).onChange(element);
|
||||
Sortable.options(dropon.parentNode).onChange(element);
|
||||
}
|
||||
} else {
|
||||
Sortable.mark(dropon, 'after');
|
||||
var nextElement = dropon.nextSibling || null;
|
||||
if(nextElement != element) {
|
||||
var oldParentNode = element.parentNode;
|
||||
element.style.visibility = "hidden"; // fix gecko rendering
|
||||
dropon.parentNode.insertBefore(element, nextElement);
|
||||
if(dropon.parentNode!=oldParentNode)
|
||||
Sortable.options(oldParentNode).onChange(element);
|
||||
Sortable.options(dropon.parentNode).onChange(element);
|
||||
}
|
||||
}
|
||||
},
|
||||
|
||||
onEmptyHover: function(element, dropon, overlap) {
|
||||
var oldParentNode = element.parentNode;
|
||||
var droponOptions = Sortable.options(dropon);
|
||||
|
||||
if(!Element.isParent(dropon, element)) {
|
||||
var index;
|
||||
|
||||
var children = Sortable.findElements(dropon, {tag: droponOptions.tag, only: droponOptions.only});
|
||||
var child = null;
|
||||
|
||||
if(children) {
|
||||
var offset = Element.offsetSize(dropon, droponOptions.overlap) * (1.0 - overlap);
|
||||
|
||||
for (index = 0; index < children.length; index += 1) {
|
||||
if (offset - Element.offsetSize (children[index], droponOptions.overlap) >= 0) {
|
||||
offset -= Element.offsetSize (children[index], droponOptions.overlap);
|
||||
} else if (offset - (Element.offsetSize (children[index], droponOptions.overlap) / 2) >= 0) {
|
||||
child = index + 1 < children.length ? children[index + 1] : null;
|
||||
break;
|
||||
} else {
|
||||
child = children[index];
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
dropon.insertBefore(element, child);
|
||||
|
||||
Sortable.options(oldParentNode).onChange(element);
|
||||
droponOptions.onChange(element);
|
||||
}
|
||||
},
|
||||
|
||||
unmark: function() {
|
||||
if(Sortable._marker) Sortable._marker.hide();
|
||||
},
|
||||
|
||||
mark: function(dropon, position) {
|
||||
// mark on ghosting only
|
||||
var sortable = Sortable.options(dropon.parentNode);
|
||||
if(sortable && !sortable.ghosting) return;
|
||||
|
||||
if(!Sortable._marker) {
|
||||
Sortable._marker =
|
||||
($('dropmarker') || Element.extend(document.createElement('DIV'))).
|
||||
hide().addClassName('dropmarker').setStyle({position:'absolute'});
|
||||
document.getElementsByTagName("body").item(0).appendChild(Sortable._marker);
|
||||
}
|
||||
var offsets = Position.cumulativeOffset(dropon);
|
||||
Sortable._marker.setStyle({left: offsets[0]+'px', top: offsets[1] + 'px'});
|
||||
|
||||
if(position=='after')
|
||||
if(sortable.overlap == 'horizontal')
|
||||
Sortable._marker.setStyle({left: (offsets[0]+dropon.clientWidth) + 'px'});
|
||||
else
|
||||
Sortable._marker.setStyle({top: (offsets[1]+dropon.clientHeight) + 'px'});
|
||||
|
||||
Sortable._marker.show();
|
||||
},
|
||||
|
||||
_tree: function(element, options, parent) {
|
||||
var children = Sortable.findElements(element, options) || [];
|
||||
|
||||
for (var i = 0; i < children.length; ++i) {
|
||||
var match = children[i].id.match(options.format);
|
||||
|
||||
if (!match) continue;
|
||||
|
||||
var child = {
|
||||
id: encodeURIComponent(match ? match[1] : null),
|
||||
element: element,
|
||||
parent: parent,
|
||||
children: [],
|
||||
position: parent.children.length,
|
||||
container: $(children[i]).down(options.treeTag)
|
||||
}
|
||||
|
||||
/* Get the element containing the children and recurse over it */
|
||||
if (child.container)
|
||||
this._tree(child.container, options, child)
|
||||
|
||||
parent.children.push (child);
|
||||
}
|
||||
|
||||
return parent;
|
||||
},
|
||||
|
||||
tree: function(element) {
|
||||
element = $(element);
|
||||
var sortableOptions = this.options(element);
|
||||
var options = Object.extend({
|
||||
tag: sortableOptions.tag,
|
||||
treeTag: sortableOptions.treeTag,
|
||||
only: sortableOptions.only,
|
||||
name: element.id,
|
||||
format: sortableOptions.format
|
||||
}, arguments[1] || { });
|
||||
|
||||
var root = {
|
||||
id: null,
|
||||
parent: null,
|
||||
children: [],
|
||||
container: element,
|
||||
position: 0
|
||||
}
|
||||
|
||||
return Sortable._tree(element, options, root);
|
||||
},
|
||||
|
||||
/* Construct a [i] index for a particular node */
|
||||
_constructIndex: function(node) {
|
||||
var index = '';
|
||||
do {
|
||||
if (node.id) index = '[' + node.position + ']' + index;
|
||||
} while ((node = node.parent) != null);
|
||||
return index;
|
||||
},
|
||||
|
||||
sequence: function(element) {
|
||||
element = $(element);
|
||||
var options = Object.extend(this.options(element), arguments[1] || { });
|
||||
|
||||
return $(this.findElements(element, options) || []).map( function(item) {
|
||||
return item.id.match(options.format) ? item.id.match(options.format)[1] : '';
|
||||
});
|
||||
},
|
||||
|
||||
setSequence: function(element, new_sequence) {
|
||||
element = $(element);
|
||||
var options = Object.extend(this.options(element), arguments[2] || { });
|
||||
|
||||
var nodeMap = { };
|
||||
this.findElements(element, options).each( function(n) {
|
||||
if (n.id.match(options.format))
|
||||
nodeMap[n.id.match(options.format)[1]] = [n, n.parentNode];
|
||||
n.parentNode.removeChild(n);
|
||||
});
|
||||
|
||||
new_sequence.each(function(ident) {
|
||||
var n = nodeMap[ident];
|
||||
if (n) {
|
||||
n[1].appendChild(n[0]);
|
||||
delete nodeMap[ident];
|
||||
}
|
||||
});
|
||||
},
|
||||
|
||||
serialize: function(element) {
|
||||
element = $(element);
|
||||
var options = Object.extend(Sortable.options(element), arguments[1] || { });
|
||||
var name = encodeURIComponent(
|
||||
(arguments[1] && arguments[1].name) ? arguments[1].name : element.id);
|
||||
|
||||
if (options.tree) {
|
||||
return Sortable.tree(element, arguments[1]).children.map( function (item) {
|
||||
return [name + Sortable._constructIndex(item) + "[id]=" +
|
||||
encodeURIComponent(item.id)].concat(item.children.map(arguments.callee));
|
||||
}).flatten().join('&');
|
||||
} else {
|
||||
return Sortable.sequence(element, arguments[1]).map( function(item) {
|
||||
return name + "[]=" + encodeURIComponent(item);
|
||||
}).join('&');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Returns true if child is contained within element
|
||||
Element.isParent = function(child, element) {
|
||||
if (!child.parentNode || child == element) return false;
|
||||
if (child.parentNode == element) return true;
|
||||
return Element.isParent(child.parentNode, element);
|
||||
}
|
||||
|
||||
Element.findChildren = function(element, only, recursive, tagName) {
|
||||
if(!element.hasChildNodes()) return null;
|
||||
tagName = tagName.toUpperCase();
|
||||
if(only) only = [only].flatten();
|
||||
var elements = [];
|
||||
$A(element.childNodes).each( function(e) {
|
||||
if(e.tagName && e.tagName.toUpperCase()==tagName &&
|
||||
(!only || (Element.classNames(e).detect(function(v) { return only.include(v) }))))
|
||||
elements.push(e);
|
||||
if(recursive) {
|
||||
var grandchildren = Element.findChildren(e, only, recursive, tagName);
|
||||
if(grandchildren) elements.push(grandchildren);
|
||||
}
|
||||
});
|
||||
|
||||
return (elements.length>0 ? elements.flatten() : []);
|
||||
}
|
||||
|
||||
Element.offsetSize = function (element, type) {
|
||||
return element['offset' + ((type=='vertical' || type=='height') ? 'Height' : 'Width')];
|
||||
}
|
|
@ -0,0 +1,121 @@
|
|||
body {
|
||||
margin: 0px;
|
||||
padding: 0px;
|
||||
background-color: #e8e8e8;
|
||||
text-align: center;
|
||||
font: 14px Trebuchet MS, Lucida Sans Unicode, Arial, sans-serif;
|
||||
}
|
||||
|
||||
a {
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
a img {
|
||||
border: none;
|
||||
vertical-align: bottom;
|
||||
text-decoration: none;
|
||||
}
|
||||
|
||||
a, a:visited, a:hover {
|
||||
color: #c10808;
|
||||
}
|
||||
|
||||
fieldset {
|
||||
width: 500px;
|
||||
margin-left: 25px;
|
||||
}
|
||||
|
||||
label {
|
||||
margin-right: 5px;
|
||||
}
|
||||
|
||||
#container {
|
||||
margin: 0 auto;
|
||||
width: 750px;
|
||||
position: relative;
|
||||
background: #ffffff;
|
||||
padding: 0px;
|
||||
text-align: left;
|
||||
}
|
||||
|
||||
#flash_container {
|
||||
padding: 5px 0px;
|
||||
}
|
||||
|
||||
#flash_container div {
|
||||
padding: 5px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
.notice {
|
||||
background-color: #edd400;
|
||||
border-top: 1px solid #c4a000;
|
||||
border-bottom: 1px solid #c4a000;
|
||||
}
|
||||
|
||||
.error {
|
||||
background-color: #ff6767;
|
||||
border-top: 1px solid #a40000;
|
||||
border-bottom: 1px solid #a40000;
|
||||
}
|
||||
|
||||
#header {
|
||||
height: 120px;
|
||||
padding-top: 10px;
|
||||
}
|
||||
|
||||
#header_image {
|
||||
margin-left: 30px;
|
||||
}
|
||||
|
||||
#tool_bar {
|
||||
margin-top: 7px;
|
||||
padding: 3px 10px 3px 10px;
|
||||
border-top: 1px solid #c17d11;
|
||||
border-bottom: 1px solid #c17d11;
|
||||
background-color: #e9b96e;
|
||||
}
|
||||
|
||||
#tool_bar a {
|
||||
font-weight: bold;
|
||||
font-size: 1.1em;
|
||||
padding-left: 3px;
|
||||
padding-right: 3px;
|
||||
text-decoration: none;
|
||||
color: #8f5902;
|
||||
}
|
||||
|
||||
#tool_bar a:hover {
|
||||
color: #5e3a01;
|
||||
}
|
||||
|
||||
#content {
|
||||
margin: 5px;
|
||||
}
|
||||
|
||||
#content a:hover {
|
||||
color: #204a87;
|
||||
}
|
||||
|
||||
#footer {
|
||||
font-size: 12px;
|
||||
height: 25px;
|
||||
width: 90%;
|
||||
border-top: 2px solid #e8e8e8;
|
||||
text-align: right;
|
||||
padding: 5px 0 10px 0;
|
||||
margin-top: 10px;
|
||||
margin-left: auto;
|
||||
margin-right: auto;
|
||||
}
|
||||
|
||||
#footer a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
.no_list_style {
|
||||
list-style-type: none;
|
||||
}
|
||||
|
||||
#recaptcha_container {
|
||||
margin: 10px 0px 10px 0px;
|
|
@ -0,0 +1,119 @@
|
|||
body {
|
||||
font-family: Arial, Verdana, sans-serif;
|
||||
font-size: 12px;
|
||||
background-color: #fff;
|
||||
}
|
||||
* {
|
||||
margin: 0px;
|
||||
padding: 0px;
|
||||
text-decoration: none;
|
||||
}
|
||||
html {
|
||||
height: 100%;
|
||||
margin-bottom: 1px;
|
||||
}
|
||||
#container {
|
||||
width: 80%;
|
||||
text-align: left;
|
||||
background-color: #fff;
|
||||
margin-right: auto;
|
||||
margin-left: auto;
|
||||
}
|
||||
#header-container {
|
||||
width: 100%;
|
||||
padding-top: 15px;
|
||||
}
|
||||
#header-container h1, #header-container h2 {
|
||||
margin-left: 6px;
|
||||
margin-bottom: 6px;
|
||||
}
|
||||
.spacer {
|
||||
width: 100%;
|
||||
height: 15px;
|
||||
}
|
||||
hr {
|
||||
border: 0px;
|
||||
color: #ccc;
|
||||
background-color: #cdcdcd;
|
||||
height: 1px;
|
||||
width: 100%;
|
||||
text-align: left;
|
||||
}
|
||||
h1 {
|
||||
font-size: 28px;
|
||||
color: #c55;
|
||||
background-color: #fff;
|
||||
font-family: Arial, Verdana, sans-serif;
|
||||
font-weight: 300;
|
||||
}
|
||||
h2 {
|
||||
font-size: 15px;
|
||||
color: #999;
|
||||
font-family: Arial, Verdana, sans-serif;
|
||||
font-weight: 300;
|
||||
background-color: #fff;
|
||||
}
|
||||
h3 {
|
||||
color: #4d9b12;
|
||||
font-size: 15px;
|
||||
text-align: left;
|
||||
font-weight: 300;
|
||||
padding: 5px;
|
||||
margin-top: 5px;
|
||||
}
|
||||
|
||||
#left-container {
|
||||
float: left;
|
||||
width: 250px;
|
||||
background-color: #FFFFFF;
|
||||
color: black;
|
||||
}
|
||||
|
||||
#left-container h3 {
|
||||
color: #c55;
|
||||
}
|
||||
|
||||
#main-container {
|
||||
margin: 5px 5px 5px 260px;
|
||||
padding: 15px;
|
||||
border-left: 1px solid silver;
|
||||
min-height: 400px;
|
||||
}
|
||||
p {
|
||||
color: #000;
|
||||
background-color: #fff;
|
||||
line-height: 20px;
|
||||
padding: 5px;
|
||||
}
|
||||
a {
|
||||
color: #4d9b12;
|
||||
background-color: #fff;
|
||||
text-decoration: none;
|
||||
}
|
||||
a:hover {
|
||||
color: #4d9b12;
|
||||
background-color: #fff;
|
||||
text-decoration: underline;
|
||||
}
|
||||
#footer-container {
|
||||
clear: both;
|
||||
font-size: 12px;
|
||||
font-family: Verdana, Arial, sans-serif;
|
||||
}
|
||||
.right {
|
||||
float: right;
|
||||
font-size: 100%;
|
||||
margin-top: 5px;
|
||||
color: #999;
|
||||
background-color: #fff;
|
||||
}
|
||||
.left {
|
||||
float: left;
|
||||
font-size: 100%;
|
||||
margin-top: 5px;
|
||||
color: #999;
|
||||
background-color: #fff;
|
||||
}
|
||||
#main-container ul {
|
||||
margin-left: 3.0em;
|
||||
}
|
|
@ -0,0 +1,15 @@
|
|||
class CreateUsersMigration < ActiveRecord::Migration
|
||||
def self.up
|
||||
create_table :users do |t|
|
||||
t.string :user_name, :limit => 32
|
||||
t.string :auth_token
|
||||
t.boolean :administrator
|
||||
t.datetime :created_at
|
||||
end
|
||||
add_index :users, :user_name
|
||||
end
|
||||
|
||||
def self.down
|
||||
drop_table :users
|
||||
end
|
||||
end
|
|
@ -0,0 +1,16 @@
|
|||
class CreatePhotosMigration < ActiveRecord::Migration
|
||||
def self.up
|
||||
create_table :photos do |t|
|
||||
t.string :filename
|
||||
t.integer :user_id
|
||||
t.string :owner_token
|
||||
t.datetime :created_at
|
||||
end
|
||||
add_index :photos, :user_id
|
||||
add_index :photos, :owner_token
|
||||
end
|
||||
|
||||
def self.down
|
||||
drop_table :photos
|
||||
end
|
||||
end
|
|
@ -0,0 +1,14 @@
|
|||
class CreateVotesMigration < ActiveRecord::Migration
|
||||
def self.up
|
||||
create_table :votes do |t|
|
||||
t.integer :photo_id, :user_id
|
||||
t.boolean :vote
|
||||
end
|
||||
add_index :votes, :photo_id
|
||||
add_index :votes, :user_id
|
||||
end
|
||||
|
||||
def self.down
|
||||
drop_table :votes
|
||||
end
|
||||
end
|
|
@ -0,0 +1,14 @@
|
|||
class CreateSessionsMigration < ActiveRecord::Migration
|
||||
def self.up
|
||||
create_table :sessions do |t|
|
||||
t.column :session_id, :string
|
||||
t.column :data, :text
|
||||
t.column :updated_at, :datetime
|
||||
end
|
||||
add_index :sessions, :session_id
|
||||
end
|
||||
|
||||
def self.down
|
||||
drop_table :sessions
|
||||
end
|
||||
end
|
|
@ -0,0 +1,50 @@
|
|||
# This file is auto-generated from the current state of the database. Instead of editing this file,
|
||||
# please use the migrations feature of Active Record to incrementally modify your database, and
|
||||
# then regenerate this schema definition.
|
||||
#
|
||||
# Note that this schema.rb definition is the authoritative source for your database schema. If you need
|
||||
# to create the application database on another system, you should be using db:schema:load, not running
|
||||
# all the migrations from scratch. The latter is a flawed and unsustainable approach (the more migrations
|
||||
# you'll amass, the slower it'll run and the greater likelihood for issues).
|
||||
#
|
||||
# It's strongly recommended to check this file into your version control system.
|
||||
|
||||
ActiveRecord::Schema.define(:version => 4) do
|
||||
|
||||
create_table "photos", :force => true do |t|
|
||||
t.string "filename"
|
||||
t.integer "user_id"
|
||||
t.string "owner_token"
|
||||
t.datetime "created_at"
|
||||
end
|
||||
|
||||
add_index "photos", ["owner_token"], :name => "index_photos_on_owner_token"
|
||||
add_index "photos", ["user_id"], :name => "index_photos_on_user_id"
|
||||
|
||||
create_table "sessions", :force => true do |t|
|
||||
t.string "session_id"
|
||||
t.text "data"
|
||||
t.datetime "updated_at"
|
||||
end
|
||||
|
||||
add_index "sessions", ["session_id"], :name => "index_sessions_on_session_id"
|
||||
|
||||
create_table "users", :force => true do |t|
|
||||
t.string "user_name", :limit => 32
|
||||
t.string "auth_token"
|
||||
t.boolean "administrator"
|
||||
t.datetime "created_at"
|
||||
end
|
||||
|
||||
add_index "users", ["user_name"], :name => "index_users_on_user_name"
|
||||
|
||||
create_table "votes", :force => true do |t|
|
||||
t.integer "photo_id"
|
||||
t.integer "user_id"
|
||||
t.boolean "vote"
|
||||
end
|
||||
|
||||
add_index "votes", ["user_id"], :name => "index_votes_on_user_id"
|
||||
add_index "votes", ["photo_id"], :name => "index_votes_on_photo_id"
|
||||
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Favorites, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Favorites, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Home, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Home, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Photos, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Photos, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Sessions, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Sessions, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Stats, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Stats, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Users, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Users, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Votes, "index action" do
|
||||
before(:each) do
|
||||
dispatch_to(Votes, :index)
|
||||
end
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::FavoritesHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::HomeHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::PhotosHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::SessionsHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::StatsHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::UsersHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,5 @@
|
|||
require File.join(File.dirname(__FILE__), '..', 'spec_helper.rb')
|
||||
|
||||
describe Merb::VotesHelper do
|
||||
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join( File.dirname(__FILE__), '..', "spec_helper" )
|
||||
|
||||
describe Photo do
|
||||
|
||||
it "should have specs"
|
||||
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join( File.dirname(__FILE__), '..', "spec_helper" )
|
||||
|
||||
describe User do
|
||||
|
||||
it "should have specs"
|
||||
|
||||
end
|
|
@ -0,0 +1,7 @@
|
|||
require File.join( File.dirname(__FILE__), '..', "spec_helper" )
|
||||
|
||||
describe Vote do
|
||||
|
||||
it "should have specs"
|
||||
|
||||
end
|
|
@ -0,0 +1,13 @@
|
|||
require 'rubygems'
|
||||
require 'merb-core'
|
||||
require 'spec' # Satisfies Autotest and anyone else not using the Rake tasks
|
||||
|
||||
# this loads all plugins required in your init file so don't add them
|
||||
# here again, Merb will do it for you
|
||||
Merb.start_environment(:testing => true, :adapter => 'runner', :environment => ENV['MERB_ENV'] || 'test')
|
||||
|
||||
Spec::Runner.configure do |config|
|
||||
config.include(Merb::Test::ViewHelper)
|
||||
config.include(Merb::Test::RouteHelper)
|
||||
config.include(Merb::Test::ControllerHelper)
|
||||
end
|
|
@ -0,0 +1,14 @@
|
|||
$TESTING=true
|
||||
require 'rubygems'
|
||||
require 'merb-core'
|
||||
|
||||
|
||||
# TODO: Boot Merb, via the Test Rack adapter
|
||||
Merb.start :environment => (ENV['MERB_ENV'] || 'test'),
|
||||
:merb_root => File.join(File.dirname(__FILE__), ".." )
|
||||
|
||||
|
||||
class Test::Unit::TestCase
|
||||
include Merb::Test::RequestHelper
|
||||
# Add more helper methods to be used by all tests here...
|
||||
end
|