email_spec-2.2.0/0000755000004100000410000000000013317714226013652 5ustar www-datawww-dataemail_spec-2.2.0/README.md0000644000004100000410000002364013317714226015136 0ustar www-datawww-data [![Build Status](https://secure.travis-ci.org/email-spec/email-spec.png)](http://travis-ci.org/email-spec/email-spec) ## Email Spec A collection of matchers for `RSpec`, `MiniTest` and `Cucumber` steps to make testing emails go smoothly. This library works with `ActionMailer` and `Pony`. When using it with ActionMailer it works with ActiveRecord Mailer, and [action_mailer_cache_delivery](https://rubygems.org/gems/action_mailer_cache_delivery). If you are testing emails in conjunction with an automated browser solution, like Selenium, you will want to use [action_mailer_cache_delivery](http://rubygems.org/gems/action_mailer_cache_delivery) in your test environment. (This is because your test process and server processes are distinct and therefore need an intermediate store for the emails.) ActiveRecord Mailer will also work but you generally don't want to include those projects unless you need them in production. ### Gem Setup ```ruby # Gemfile group :test do gem 'email_spec' end ``` ### Cucumber To use the steps in features put the following in your env.rb: ```ruby # Make sure this require is after you require cucumber/rails/world. require 'email_spec' # add this line if you use spork require 'email_spec/cucumber' ``` This will load all the helpers that the steps rely on. It will also add a `Before` hook for `Cucumber` so that emails are cleared at the start of each scenario. Then: ```bash rails generate email_spec:steps ``` This will give you a bunch of steps to get started with in `step_definitions/email_steps.rb` By default, the generated file will look for email to example@example.com. You can either change this by editing the `current_email_address` method in `email_steps.rb`, or by simply specifying the target email in your features: ```gherkin Scenario: A new person signs up Given I am at "/" When I fill in "Email" with "quentin@example.com" And I press "Sign up" Then "quentin@example.com" should receive an email # Specify who should receive the email ``` ### Spinach To use the helpers and matchers in your Spinach steps, add this to your env.rb: ```ruby require 'email_spec/spinach' ``` Creating shared steps (as for Cucumber above) doesn't fit so well with the Spinach ethos of very compartmentalized steps, so there is no generator for Spinach. It's easy to use the helpers/matchers in your steps. For example: ```ruby step 'the last email sent should welcome the user' do expect(last_email_sent).to have_subject('Welcome') end ``` ### RSpec (3.1+) First you need to require `email_spec` in your `spec_helper.rb`: ```ruby require "email_spec" require "email_spec/rspec" ``` This will load all the helpers that the scenarios can count on. It will also add a `before(:each)` hook so that emails are cleared at the start of each scenario. If you are upgrading to Rails 5, make sure your `rails_helper.rb` requires `spec_helper` **after** loading the environment. For example: ```ruby require File.expand_path('../../config/environment', __FILE__) require 'spec_helper' ``` ### MiniTest First you need to require minitest-matchers and email_spec in your test_helper.rb: ```ruby require "minitest-matchers" require "email_spec" ``` You will then need to include EmailSpec::Helpers and EmailSpec::Matchers in your test classes. If you want to have access to the helpers and matchers in all of your tests you can do the following in your test_helper.rb: ```ruby class MiniTest::Unit::TestCase include EmailSpec::Helpers include EmailSpec::Matchers end ``` Otherwise, you will need to include them in the tests where you use them: ```ruby class SignupMailerTest < MiniTest::Unit::TestCase include EmailSpec::Helpers include EmailSpec::Matchers ... end ``` Or, if you are using the MiniTest spec DSL, it would look like this: ```ruby describe SignupMailer do include EmailSpec::Helpers include EmailSpec::Matchers ... end ``` ### Turnip If you're using [Turnip](https://github.com/jnicklas/turnip), you might be interested in this [conversion of the Cucumber steps into Turnip steps](https://github.com/jmuheim/base/blob/7708873e77165993c2c962894c756621be1b15cc/spec/steps/email_steps.rb). ## Background Jobs If you are using a background job, you might need to use a step to process the jobs. Another alternative is to use an inline statement for your scenario. For example, for DelayedJob: ```ruby Delayed::Worker.delay_jobs = false ``` ## Usage ### Cucumber ```gherkin Scenario: A new person signs up Given I am at "/" When I fill in "Email" with "quentin@example.com" And I press "Sign up" And I should receive an email When I open the email Then I should see "confirm" in the email body When I follow "confirm" in the email Then I should see "Confirm your new account" ``` For more examples, check out examples/rails_root in the source for a small example app that implements these steps. ### Cucumber Matchers (Ruby) See RSpec Matchers (they are the same) ### RSpec #### Testing In Isolation It is often useful to test your mailers in isolation. You can accomplish this by using mocks to verify that the mailer is being called in the correct place and then write focused examples for the actual mailer. This is a simple example from the sample app found in the gem: Verify that the mailer is used correctly in the controller (this would apply to a model as well): ```ruby describe "POST /signup (#signup)" do it "should deliver the signup email" do # expect expect(UserMailer).to(receive(:deliver_signup).with("email@example.com", "Jimmy Bean")) # when post :signup, "Email" => "email@example.com", "Name" => "Jimmy Bean" end end ``` Examples for the #signup method in UserMailer: ```ruby describe "Signup Email" do include EmailSpec::Helpers include EmailSpec::Matchers # include ActionController::UrlWriter - old rails include Rails.application.routes.url_helpers before(:all) do @email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") end it "should be set to be delivered to the email passed in" do expect(@email).to deliver_to("jojo@yahoo.com") end it "should contain the user's message in the mail body" do expect(@email).to have_body_text(/Jojo Binks/) end it "should contain a link to the confirmation link" do expect(@email).to have_body_text(/#{confirm_account_url}/) end it "should have the correct subject" do expect(@email).to have_subject(/Account confirmation/) end end ``` #### RSpec Matchers ##### reply_to(email) alias: `have_reply_to` This checks that the Reply-To header's email address (the bob@example.com of "Bob Saget ") is set to the given string. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to reply_to("support@myapp.com") ``` ##### deliver_to(\*email_addresses) alias: `be_delivered_to` This checks that the To header's email addresses (the bob@example.com of "Bob Saget ") are set to the addresses. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to deliver_to("jojo@yahoo.com") ``` ##### deliver_from(email) alias: `be_delivered_from` This checks that the From header's email address (the bob@example.com of "Bob Saget ") is set to the given string. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to deliver_from("sally@yahoo.com") ``` ##### bcc_to(\*email_addresses) This checks that the BCC header's email addresses (the bob@example.com of "Bob Saget ") are set to the addresses. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to bcc_to("sue@yahoo.com", "bill@yahoo.com") ``` ##### cc_to(\*email_addresses) This checks that the CC header's email addresses (the bob@example.com of "Bob Saget ") are set to the addresses. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to cc_to("sue@yahoo.com", "bill@yahoo.com") ``` ##### have_subject(subject) This checks that the Subject header's value is set to the given subject. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to have_subject("Welcome!") ``` ##### include_email_with_subject(subject) Note: subject can be either a String or a Regexp This checks that one of the given emails' subjects includes the subject. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") email2 = UserMailer.forgot_password("jojo@yahoo.com", "Jojo Binks") expect([email, email2]).to include_email_with_subject("Welcome!") ``` ##### have_body_text(text) Note: text can be either a String or a Regexp This checks that the text of the body has the given body. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to have_body_text(/Hi Jojo Binks,/) ``` ##### have_header(key, value) This checks that the expected key/value pair is in the headers of the email. ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") expect(email).to have_header("X-Campaign", "1234abc") ``` #### Using the helpers when not testing in isolation Don't. :) Seriously, if you do just take a look at the helpers and use them as you wish. ### MiniTest You will use EmailSpec in your tests the same way you use it in your specs. The only difference is the use of MiniTest's `must` instead of Rspec's `should`: ```ruby email = UserMailer.create_signup("jojo@yahoo.com", "Jojo Binks") email.must deliver_to("jojo@yahoo.com") ``` Or, you can use the matcher as an expectation: ```ruby email = UserMailer.create_signup "jojo@yahoo.com", "Jojo Binks" email.must_deliver_to "jojo@yahoo.com" ``` And of course you can use the matcher as an assertion: ```ruby email = UserMailer.create_signup "jojo@yahoo.com", "Jojo Binks" assert_must deliver_to("jojo@yahoo.com"), email ``` ## Original Authors Ben Mabey, Aaron Gibralter, Mischa Fierer Please see [Changelog.md](Changelog.md) for upcoming changes and other contributors. email_spec-2.2.0/spec/0000755000004100000410000000000013317714226014604 5ustar www-datawww-dataemail_spec-2.2.0/spec/email_spec/0000755000004100000410000000000013317714226016705 5ustar www-datawww-dataemail_spec-2.2.0/spec/email_spec/helpers_spec.rb0000644000004100000410000002650113317714226021712 0ustar www-datawww-datarequire File.dirname(__FILE__) + '/../spec_helper' describe EmailSpec::Helpers do include EmailSpec::Helpers describe "#visit_in_email" do it "visits the link in the email" do @to = "jimmy_bean@yahoo.com" @body = "Hello!" @email = Mail::Message.new(:to => @to, :from => "foo@bar.com", :body => @body) allow(self).to receive(:mailbox_for).with(@to).and_return([@email]) expect(open_email(@to, :from => "foo@bar.com")).to eq(@email) expect do expect(self).to(receive(:visit).with('/hello')) visit_in_email("Hello!") end.not_to raise_error end it "raises an exception when an email is not found" do expect do visit_in_email("Hello!", 'jon@example.com') end.to raise_error(EmailSpec::CouldNotFindEmailError) end end describe "#parse_email_for_link" do it "properly finds links with text" do email = Mail.new(:body => %(Click Here)) expect(parse_email_for_link(email, "Click Here")).to eq("/path/to/page") end it "properly finds links with text surrounded by tags" do email = Mail.new(:body => %( WelcomeClick Here)) expect(parse_email_for_link(email, "Click Here")).to eq("/path/to/page") end it "properly finds links with tags and text in new lines" do email = Mail.new(:body => <<-HTML Welcome Click Here HTML ) expect(parse_email_for_link(email, "Click Here")).to eq("/path/to/page") end it "recognizes img alt properties as text" do email = Mail.new(:body => %(an image)) expect(parse_email_for_link(email, "an image")).to eq("/path/to/page") end it "causes a spec to fail if the body doesn't contain the text specified to click" do email = Mail.new(:body => "") expect { parse_email_for_link(email, "non-existent text") }.to raise_error( RSpec::Expectations::ExpectationNotMetError) end end describe "#set_current_email" do it "should cope with a nil email" do expect do out = set_current_email(nil) expect(out).to be_nil expect(email_spec_hash[:current_email]).to be_nil end.not_to raise_error end it "should cope with a real email" do email = Mail.new expect do out = set_current_email(email) expect(out).to eq(email) expect(email_spec_hash[:current_email]).to eq(email) end.not_to raise_error end shared_examples_for 'something that sets the current email for recipients' do before do @email = Mail.new(@recipient_type => 'dave@example.com') end it "should record that the email has been read for that recipient" do set_current_email(@email) expect(email_spec_hash[:read_emails]['dave@example.com']).to include(@email) end it "should record that the email has been read for all the recipient of that type" do @email.send(@recipient_type) << 'dave_2@example.com' set_current_email(@email) expect(email_spec_hash[:read_emails]['dave@example.com']).to include(@email) expect(email_spec_hash[:read_emails]['dave_2@example.com']).to include(@email) end it "should record that the email is the current email for the recipient" do set_current_email(@email) expect(email_spec_hash[:current_emails]['dave@example.com']).to eq(@email) end it "should record that the email is the current email for all the recipients of that type" do @email.send(@recipient_type) << 'dave_2@example.com' set_current_email(@email) expect(email_spec_hash[:current_emails]['dave@example.com']).to eq(@email) expect(email_spec_hash[:current_emails]['dave_2@example.com']).to eq(@email) end it "should overwrite current email for the recipient with this one" do other_mail = Mail.new email_spec_hash[:current_emails]['dave@example.com'] = other_mail set_current_email(@email) expect(email_spec_hash[:current_emails]['dave@example.com']).to eq(@email) end it "should overwrite the current email for all the recipients of that type" do other_mail = Mail.new email_spec_hash[:current_emails]['dave@example.com'] = other_mail email_spec_hash[:current_emails]['dave_2@example.com'] = other_mail @email.send(@recipient_type) << 'dave_2@example.com' set_current_email(@email) expect(email_spec_hash[:current_emails]['dave@example.com']).to eq(@email) expect(email_spec_hash[:current_emails]['dave_2@example.com']).to eq(@email) end it "should not complain when the email has recipients of that type" do @email.send(:"#{@recipient_type}=", nil) expect { set_current_email(@email) }.not_to raise_error end end describe "#request_uri(link)" do context "without query and anchor" do it "returns the path" do expect(request_uri('http://www.path.se/to/page')).to eq('/to/page') end end context "with query and anchor" do it "returns the path and query and the anchor" do expect(request_uri('http://www.path.se/to/page?q=adam#task')).to eq('/to/page?q=adam#task') end end context "with anchor" do it "returns the path and query and the anchor" do expect(request_uri('http://www.path.se/to/page#task')).to eq('/to/page#task') end end end describe 'for mails with recipients in the to address' do before do @recipient_type = :to end it_should_behave_like 'something that sets the current email for recipients' end describe 'for mails with recipients in the cc address' do before do @recipient_type = :cc end it_should_behave_like 'something that sets the current email for recipients' end describe 'for mails with recipients in the bcc address' do before do @recipient_type = :bcc end it_should_behave_like 'something that sets the current email for recipients' end end describe '#open_email' do describe 'from' do before do @to = "jimmy_bean@yahoo.com" @email = Mail::Message.new(:to => @to, :from => "foo@bar.com") allow(self).to receive(:mailbox_for).with(@to).and_return([@email]) end it "should open the email from someone" do expect(open_email(@to, :from => "foo@bar.com")).to eq(@email) end end describe 'with subject' do shared_examples_for 'something that opens the email with subject' do before do @to = "jimmy_bean@yahoo.com" @email = Mail::Message.new(:to => @to, :subject => @subject) allow(self).to receive(:mailbox_for).with(@to).and_return([@email]) end it "should open the email with subject" do expect(open_email(@to, :with_subject => @expected)).to eq(@email) end end describe 'simple string subject' do before do @subject = 'This is a simple subject' @expected = 'a simple' end it_should_behave_like 'something that opens the email with subject' end describe 'string with regex sensitive characters' do before do @subject = '[app name] Contains regex characters?' @expected = 'regex characters?' end it_should_behave_like 'something that opens the email with subject' end describe 'regular expression' do before do @subject = "This is a simple subject" @expected = /a simple/ end it_should_behave_like 'something that opens the email with subject' end end describe 'with text' do shared_examples_for 'something that opens the email with text' do before do @to = "jimmy_bean@yahoo.com" @email = Mail::Message.new(:to => @to, :body => @body) allow(self).to receive(:mailbox_for).with(@to).and_return([@email]) end it "should open the email with text" do expect(open_email(@to, :with_text => @text)).to eq(@email) end end describe 'simple string text' do before do @body = 'This is an email body that is very simple' @text = 'email body' end it_should_behave_like 'something that opens the email with text' end describe 'string with regex sensitive characters' do before do @body = 'This is an email body. It contains some [regex] characters?' @text = '[regex] characters?' end it_should_behave_like 'something that opens the email with text' end describe 'regular expression' do before do @body = 'This is an email body.' @text = /an\ email/ end it_should_behave_like 'something that opens the email with text' end end describe "when the email isn't found" do it "includes the mailbox that was looked in when an address was provided" do @email_address = "foo@bar.com" expect { open_email(@email_address, :with_subject => "baz") }.to raise_error(EmailSpec::CouldNotFindEmailError) { |error| expect(error.message).to include(@email_address) } end it "includes a warning that no mailboxes were searched when no address was provided" do allow(subject).to receive(:last_email_address).and_return nil expect { open_email(nil, :with_subject => "baz") }.to raise_error(EmailSpec::NoEmailAddressProvided) { |error| expect(error.message).to eq("No email address has been provided. Make sure current_email_address is returning something.") } end describe "search by with_subject" do before do @email_subject = "Subject of 'Nonexistent Email'" begin open_email("foo@bar.com", :with_subject => @email_subject) rescue EmailSpec::CouldNotFindEmailError => e @error = e end expect(@error).not_to be_nil, "expected an error to get thrown so we could test against it, but didn't catch one" end it "includes the subject that wasn't found in the error message" do expect(@error.message).to include(@email_subject) end it "includes 'with subject' in the error message" do expect(@error.message).to include('with subject') end end describe "search by with_text" do before do @email_text = "This is a line of text from a 'Nonexistent Email'." begin open_email("foo@bar.com", :with_text => @email_text) rescue EmailSpec::CouldNotFindEmailError => e @error = e end expect(@error).not_to be_nil, "expected an error to get thrown so we could test against it, but didn't catch one" end it "includes the text that wasn't found in the error message" do expect(@error.message).to include(@email_text) end it "includes 'with text' in the error message" do expect(@error.message).to include('with text') end end end end end email_spec-2.2.0/spec/email_spec/mail_ext_spec.rb0000644000004100000410000000310613317714226022046 0ustar www-datawww-datarequire 'spec_helper' describe EmailSpec::MailExt do describe "#default_part" do it "prefers html_part over text_part" do email = Mail.new do text_part { body "This is text" } html_part { body "This is html" } end expect(email.default_part.body.to_s).to eq("This is html") end it "returns text_part if html_part not available" do email = Mail.new do text_part { body "This is text" } end expect(email.default_part.body.to_s).to eq("This is text") end it "returns the email if not multi-part" do email = Mail.new { body "This is the body" } expect(email.default_part.body.to_s).to eq("This is the body") end end describe "#default_part_body" do it "returns default_part.body" do email = Mail.new(:body => "hi") expect(email.default_part.body).to eq(email.default_part_body) end it "compatible with ActiveSupport::SafeBuffer" do email = Mail.new(:body => ActiveSupport::SafeBuffer.new("bacon & pancake")) expect(email.default_part_body).to eq ("bacon & pancake") end end describe "#html" do it "returns the html part body" do email = Mail.new do html_part { body "This is html" } end expect(email.html).to eq("This is html") end it "returns a String" do email = Mail.new do html_part { body "This is html" } end expect(email.html).to be_a(String) end it "returns nil for mail with no html part" do email = Mail.new expect(email.html).to be_nil end end end email_spec-2.2.0/spec/email_spec/matchers_spec.rb0000644000004100000410000005005413317714226022056 0ustar www-datawww-datarequire File.dirname(__FILE__) + '/../spec_helper' describe EmailSpec::Matchers do include EmailSpec::Matchers class MatcherMatch def initialize(object_to_test_match) @object_to_test_match = object_to_test_match end def description "match when provided #{@object_to_test_match.inspect}" end def matches?(matcher) @matcher = matcher matcher.matches?(@object_to_test_match) end def failure_message "expected #{@matcher.inspect} to match when provided #{@object_to_test_match.inspect}, but it did not" end def failure_message_when_negated "expected #{@matcher.inspect} not to match when provided #{@object_to_test_match.inspect}, but it did" end alias negative_failure_message failure_message_when_negated end def match(object_to_test_match) if object_to_test_match.is_a?(Regexp) super # delegate to rspec's built in 'match' matcher else MatcherMatch.new(object_to_test_match) end end describe "#reply_to" do it "should match when the email is set to deliver to the specified address" do email = Mail::Message.new(:reply_to => ["test@gmail.com"]) expect(reply_to("test@gmail.com")).to match(email) end it "should match given a name and address" do email = Mail::Message.new(:reply_to => ["test@gmail.com"]) expect(reply_to("David Balatero ")).to match(email) end it "should give correct failure message when the email is not set to deliver to the specified address" do matcher = reply_to("jimmy_bean@yahoo.com") message = Mail::Message.new(:reply_to => ['freddy_noe@yahoo.com']) allow(message).to receive(:inspect).and_return("email") matcher.matches?(message) expect(matcher_failure_message(matcher)).to eq(%{expected email to reply to "jimmy_bean@yahoo.com", but it replied to "freddy_noe@yahoo.com"}) end end describe "#deliver_to" do it "should match when the email is set to deliver to the specified address" do email = Mail::Message.new(:to => "jimmy_bean@yahoo.com") expect(deliver_to("jimmy_bean@yahoo.com")).to match(email) end it "should match when the email is set to deliver to the specified name and address" do email = Mail::Message.new(:to => "Jimmy Bean ") expect(deliver_to("Jimmy Bean ")).to match(email) end it "should match when a list of emails is exact same as all of the email's recipients" do email = Mail::Message.new(:to => ["james@yahoo.com", "karen@yahoo.com"]) expect(deliver_to("karen@yahoo.com", "james@yahoo.com")).to match(email) expect(deliver_to("karen@yahoo.com")).not_to match(email) end it "should match when an array of emails is exact same as all of the email's recipients" do addresses = ["james@yahoo.com", "karen@yahoo.com"] email = Mail::Message.new(:to => addresses) expect(deliver_to(addresses)).to match(email) end it "should match when the names and email addresses match in any order" do addresses = ["James ", "Karen "] email = Mail::Message.new(:to => addresses.reverse) expect(deliver_to(addresses)).to match(email) end it "should use the passed in objects :email method if not a string" do email = Mail::Message.new(:to => "jimmy_bean@yahoo.com") user = double("user", :email => "jimmy_bean@yahoo.com") expect(deliver_to(user)).to match(email) end it "should not match when the email does not have a recipient" do email = Mail::Message.new(:to => nil) expect(deliver_to("jimmy_bean@yahoo.com")).not_to match(email) end it "should not match when the email addresses match but the names do not" do email = Mail::Message.new(:to => "Jimmy Bean ") expect(deliver_to("Freddy Noe ")).not_to match(email) end it "should not match when the names match but the email addresses do not" do email = Mail::Message.new(:to => "Jimmy Bean ") expect(deliver_to("Jimmy Bean ")).not_to match(email) end it "should give correct failure message when the email is not set to deliver to the specified address" do matcher = deliver_to("jimmy_bean@yahoo.com") message = Mail::Message.new(:to => 'freddy_noe@yahoo.com') allow(message).to receive(:inspect).and_return("email") matcher.matches?(message) expect(matcher_failure_message(matcher)).to eq(%{expected email to deliver to ["jimmy_bean@yahoo.com"], but it delivered to ["freddy_noe@yahoo.com"]}) end it "should deliver to nobody when the email does not perform deliveries" do email = Mail::Message.new(:to => "jimmy_bean@yahoo.com") email.perform_deliveries = false expect(deliver_to("jimmy_bean@yahoo.com")).not_to match(email) end end describe "#deliver_from" do it "should match when the email is set to deliver from the specified address" do email = Mail::Message.new(:from => "jimmy_bean@yahoo.com") expect(deliver_from("jimmy_bean@yahoo.com")).to match(email) end it "should match when the email is set to deliver from the specified name and address" do email = Mail::Message.new(:from => "Jimmy Bean ") expect(deliver_from("Jimmy Bean ")).to match(email) end it "should not match when the email does not have a sender" do email = Mail::Message.new(:from => nil) expect(deliver_from("jimmy_bean@yahoo.com")).not_to match(email) end it "should not match when the email addresses match but the names do not" do email = Mail::Message.new(:from => "Jimmy Bean ") expect(deliver_from("Freddy Noe ")).not_to match(email) end it "should not match when the names match but the email addresses do not" do email = Mail::Message.new(:from => "Jimmy Bean ") expect(deliver_from("Jimmy Bean ")).not_to match(email) end it "should not match when the email is not set to deliver from the specified address" do email = Mail::Message.new(:from => "freddy_noe@yahoo.com") expect(deliver_from("jimmy_bean@yahoo.com")).not_to match(email) end it "should give correct failure message when the email is not set to deliver from the specified address" do matcher = deliver_from("jimmy_bean@yahoo.com") matcher.matches?(Mail::Message.new(:from => "freddy_noe@yahoo.com")) expect(matcher_failure_message(matcher)).to match(/expected .+ to deliver from "jimmy_bean@yahoo\.com", but it delivered from "freddy_noe@yahoo\.com"/) end it "should not deliver from anybody when perform_deliveries is false" do email = Mail::Message.new(:from => "freddy_noe@yahoo.com") email.perform_deliveries = false expect(deliver_from("freddy_noe@yahoo.com")).not_to match(email) end end describe "#bcc_to" do it "should match when the email is set to deliver to the specidied address" do email = Mail::Message.new(:bcc => "jimmy_bean@yahoo.com") expect(bcc_to("jimmy_bean@yahoo.com")).to match(email) end it "should match when a list of emails is exact same as all of the email's recipients" do email = Mail::Message.new(:bcc => ["james@yahoo.com", "karen@yahoo.com"]) expect(bcc_to("karen@yahoo.com", "james@yahoo.com")).to match(email) expect(bcc_to("karen@yahoo.com")).not_to match(email) end it "should match when an array of emails is exact same as all of the email's recipients" do addresses = ["james@yahoo.com", "karen@yahoo.com"] email = Mail::Message.new(:bcc => addresses) expect(bcc_to(addresses)).to match(email) end it "should use the passed in objects :email method if not a string" do email = Mail::Message.new(:bcc => "jimmy_bean@yahoo.com") user = double("user", :email => "jimmy_bean@yahoo.com") expect(bcc_to(user)).to match(email) end it "should bcc to nobody when the email does not perform deliveries" do email = Mail::Message.new(:bcc => "jimmy_bean@yahoo.com") email.perform_deliveries = false expect(bcc_to("jimmy_bean@yahoo.com")).not_to match(email) end end describe "#cc_to" do it "should match when the email is set to deliver to the specified address" do email = Mail::Message.new(:cc => "jimmy_bean@yahoo.com") expect(cc_to("jimmy_bean@yahoo.com")).to match(email) end it "should match when a list of emails is exact same as all of the email's recipients" do email = Mail::Message.new(:cc => ["james@yahoo.com", "karen@yahoo.com"]) expect(cc_to("karen@yahoo.com", "james@yahoo.com")).to match(email) expect(cc_to("karen@yahoo.com")).not_to match(email) end it "should match when an array of emails is exact same as all of the email's recipients" do addresses = ["james@yahoo.com", "karen@yahoo.com"] email = Mail::Message.new(:cc => addresses) expect(cc_to(addresses)).to match(email) end it "should use the passed in objects :email method if not a string" do email = Mail::Message.new(:cc => "jimmy_bean@yahoo.com") user = double("user", :email => "jimmy_bean@yahoo.com") expect(cc_to(user)).to match(email) end it "should cc to nobody when the email does not perform deliveries" do email = Mail::Message.new(to: "jimmy_bean@yahoo.com") email.perform_deliveries = false expect(cc_to("jimmy_bean@yahoo.com")).not_to match(email) end end describe "#have_subject" do describe "when regexps are used" do it "should match when the subject matches regexp" do email = Mail::Message.new(:subject => ' -- The Subject --') expect(have_subject(/The Subject/)).to match(email) expect(have_subject(/foo/)).not_to match(email) end it "should have a helpful description" do matcher = have_subject(/foo/) matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher.description).to eq("have subject matching /foo/") end it "should offer helpful failing messages" do matcher = have_subject(/foo/) matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher_failure_message(matcher)).to eq('expected the subject to match /foo/, but did not. Actual subject was: "bar"') end it "should offer helpful negative failing messages" do matcher = have_subject(/b/) matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher_failure_message_when_negated(matcher)).to eq('expected the subject not to match /b/ but "bar" does match it.') end end describe "when strings are used" do it "should match when the subject equals the passed in string exactly" do email = Mail::Message.new(:subject => 'foo') expect(have_subject("foo")).to match(email) expect(have_subject(" - foo -")).not_to match(email) end it "should have a helpful description" do matcher = have_subject("foo") matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher.description).to eq('have subject of "foo"') end it "should offer helpful failing messages" do matcher = have_subject("foo") matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher_failure_message(matcher)).to eq('expected the subject to be "foo" but was "bar"') end it "should offer helpful negative failing messages" do matcher = have_subject("bar") matcher.matches?(Mail::Message.new(:subject => "bar")) expect(matcher_failure_message_when_negated(matcher)).to eq('expected the subject not to be "bar" but was') end end end describe "#include_email_with_subject" do describe "when regexps are used" do it "should match when any email's subject matches passed in regexp" do emails = [Mail::Message.new(:subject => "foobar"), Mail::Message.new(:subject => "bazqux")] expect(include_email_with_subject(/foo/)).to match(emails) expect(include_email_with_subject(/quux/)).not_to match(emails) end it "should have a helpful description" do matcher = include_email_with_subject(/foo/) matcher.matches?([]) expect(matcher.description).to eq('include email with subject matching /foo/') end it "should offer helpful failing messages" do matcher = include_email_with_subject(/foo/) matcher.matches?([Mail::Message.new(:subject => "bar")]) expect(matcher_failure_message(matcher)).to eq('expected at least one email to have a subject matching /foo/, but none did. Subjects were ["bar"]') end it "should offer helpful negative failing messages" do matcher = include_email_with_subject(/foo/) matcher.matches?([Mail::Message.new(:subject => "foo")]) expect(matcher_failure_message_when_negated(matcher)).to eq('expected no email to have a subject matching /foo/ but found at least one. Subjects were ["foo"]') end end describe "when strings are used" do it "should match when any email's subject equals passed in subject exactly" do emails = [Mail::Message.new(:subject => "foobar"), Mail::Message.new(:subject => "bazqux")] expect(include_email_with_subject("foobar")).to match(emails) expect(include_email_with_subject("foo")).not_to match(emails) end it "should have a helpful description" do matcher = include_email_with_subject("foo") matcher.matches?([]) expect(matcher.description).to eq('include email with subject of "foo"') end it "should offer helpful failing messages" do matcher = include_email_with_subject("foo") matcher.matches?([Mail::Message.new(:subject => "bar")]) expect(matcher_failure_message(matcher)).to eq('expected at least one email to have the subject "foo" but none did. Subjects were ["bar"]') end it "should offer helpful negative failing messages" do matcher = include_email_with_subject("foo") matcher.matches?([Mail::Message.new(:subject => "foo")]) expect(matcher_failure_message_when_negated(matcher)).to eq('expected no email with the subject "foo" but found at least one. Subjects were ["foo"]') end end end describe "#have_body_text" do describe "when regexps are used" do it "should match when the body matches regexp" do email = Mail::Message.new(:body => 'foo bar baz') expect(have_body_text(/bar/)).to match(email) expect(have_body_text(/qux/)).not_to match(email) end it "should have a helpful description" do matcher = have_body_text(/qux/) matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher.description).to eq('have body matching /qux/') end it "should offer helpful failing messages" do matcher = have_body_text(/qux/) matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher_failure_message(matcher)).to eq('expected the body to match /qux/, but did not. Actual body was: "foo bar baz"') end it "should offer helpful negative failing messages" do matcher = have_body_text(/bar/) matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher_failure_message_when_negated(matcher)).to eq('expected the body not to match /bar/ but "foo bar baz" does match it.') end end describe "when strings are used" do it "should match when the body includes text with symbols" do full_name = "Jermain O'Keefe" email = Mail::Message.new(body: full_name) expect(have_body_text(full_name)).to match(email) end it "should match when the body includes the text" do email = Mail::Message.new(:body => 'foo bar baz') expect(have_body_text('bar')).to match(email) expect(have_body_text('qux')).not_to match(email) end it "should have a helpful description" do matcher = have_body_text('qux') matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher.description).to eq('have body including "qux"') end it "should offer helpful failing messages" do matcher = have_body_text('qux') matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher_failure_message(matcher)).to eq('expected the body to contain "qux" but was "foo bar baz"') end it "should offer helpful negative failing messages" do matcher = have_body_text('bar') matcher.matches?(Mail::Message.new(:body => 'foo bar baz')) expect(matcher_failure_message_when_negated(matcher)).to eq('expected the body not to contain "bar" but was "foo bar baz"') end end describe "when dealing with multipart messages" do it "should look at the html part" do email = Mail.new do text_part do body "This is text" end html_part do body "This is html" end end expect(have_body_text(/This is html/)).to match(email) expect(have_body_text(/This is text/)).not_to match(email) end end end describe "#have_header" do describe "when regexps are used" do it "should match when header matches passed in regexp" do email = Mail::Message.new(:content_type => "text/html") expect(have_header(:content_type, /text/)).to match(email) expect(have_header(:foo, /text/)).not_to match(email) expect(have_header(:content_type, /bar/)).not_to match(email) end it "should have a helpful description" do matcher = have_header(:content_type, /bar/) matcher.matches?(Mail::Message.new(:content_type => "text/html")) expect(matcher.description).to eq('have header content_type with value matching /bar/') end it "should offer helpful failing messages" do matcher = have_header(:content_type, /bar/) matcher.matches?(Mail::Message.new(:content_type => "text/html")) expect(matcher_failure_message(matcher)).to eq('expected the headers to include \'content_type\' with a value matching /bar/ but they were {"content-type"=>"text/html"}') end it "should offer helpful negative failing messages" do matcher = have_header(:content_type, /text/) matcher.matches?(Mail::Message.new(:content_type => "text/html")) expect(matcher_failure_message_when_negated(matcher)).to eq('expected the headers not to include \'content_type\' with a value matching /text/ but they were {"content-type"=>"text/html"}') end end describe "when strings are used" do it "should match when header equals passed in value exactly" do email = Mail::Message.new(:content_type => "text/html") expect(have_header(:content_type, 'text/html')).to match(email) expect(have_header(:foo, 'text/html')).not_to match(email) expect(have_header(:content_type, 'text')).not_to match(email) end it "should have a helpful description" do matcher = have_header(:content_type, 'text') matcher.matches?(Mail::Message.new(:content_type => "text/html")) expect(matcher.description).to eq('have header content_type: text') end it "should offer helpful failing messages" do matcher = have_header(:content_type, 'text') matcher.matches?(Mail::Message.new(:content_type => "text/html")) expect(matcher_failure_message(matcher)).to eq('expected the headers to include \'content_type: text\' but they were {"content-type"=>"text/html"}') end it "should offer helpful negative failing messages" do matcher = have_header(:content_type, 'text/html') matcher.matches?(Mail::Message.new(:content_type => "text/html")) matcher_failure_message_when_negated(matcher) == 'expected the headers not to include \'content_type: text/html\' but they were {:content_type=>"text/html"}' end end end end email_spec-2.2.0/spec/email_spec/email_viewer_spec.rb0000644000004100000410000000117713317714226022722 0ustar www-datawww-datarequire 'spec_helper' require 'launchy' describe EmailSpec::EmailViewer do describe ".open_in_browser" do it "should open with launchy" do expected_uri = URI("file://#{File.expand_path("a_file")}") expect(Launchy).to receive(:open).with(expected_uri) EmailSpec::EmailViewer.open_in_browser("a_file") end end describe ".open_in_text_editor" do it "should open with launchy" do expected_uri = URI("file://#{File.expand_path("a_file")}") expect(Launchy).to receive(:open).with(expected_uri, {application: :editor}) EmailSpec::EmailViewer.open_in_text_editor("a_file") end end end email_spec-2.2.0/spec/spec_helper.rb0000644000004100000410000000126413317714226017425 0ustar www-datawww-datarequire 'rubygems' require 'action_mailer' require 'mail' require File.expand_path(File.dirname(__FILE__) + '/../lib/email_spec.rb') RSpec.configure do |config| config.include EmailSpec::Helpers config.include EmailSpec::Matchers config.mock_with :rspec config.raise_errors_for_deprecations! def matcher_failure_message(matcher) matcher.respond_to?(:failure_message_for_should) ? matcher.failure_message_for_should : matcher.failure_message end def matcher_failure_message_when_negated(matcher) matcher.respond_to?(:failure_message_for_should_not) ? matcher.failure_message_for_should_not : matcher.negative_failure_message end end email_spec-2.2.0/examples/0000755000004100000410000000000013317714226015470 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/0000755000004100000410000000000013317714226017731 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/Gemfile.lock0000644000004100000410000001157013317714226022157 0ustar www-datawww-dataPATH remote: ../.. specs: email_spec (2.1.2) htmlentities (~> 4.3.3) launchy (~> 2.1) mail (~> 2.7) GEM remote: http://rubygems.org/ specs: actionmailer (4.2.8) actionpack (= 4.2.8) actionview (= 4.2.8) activejob (= 4.2.8) mail (~> 2.5, >= 2.5.4) rails-dom-testing (~> 1.0, >= 1.0.5) actionpack (4.2.8) actionview (= 4.2.8) activesupport (= 4.2.8) rack (~> 1.6) rack-test (~> 0.6.2) rails-dom-testing (~> 1.0, >= 1.0.5) rails-html-sanitizer (~> 1.0, >= 1.0.2) actionview (4.2.8) activesupport (= 4.2.8) builder (~> 3.1) erubis (~> 2.7.0) rails-dom-testing (~> 1.0, >= 1.0.5) rails-html-sanitizer (~> 1.0, >= 1.0.3) activejob (4.2.8) activesupport (= 4.2.8) globalid (>= 0.3.0) activemodel (4.2.8) activesupport (= 4.2.8) builder (~> 3.1) activerecord (4.2.8) activemodel (= 4.2.8) activesupport (= 4.2.8) arel (~> 6.0) activesupport (4.2.8) i18n (~> 0.7) minitest (~> 5.1) thread_safe (~> 0.3, >= 0.3.4) tzinfo (~> 1.1) addressable (2.5.2) public_suffix (>= 2.0.2, < 4.0) arel (6.0.4) backports (3.11.1) builder (3.2.3) capybara (2.18.0) addressable mini_mime (>= 0.1.3) nokogiri (>= 1.3.3) rack (>= 1.0.0) rack-test (>= 0.5.4) xpath (>= 2.0, < 4.0) concurrent-ruby (1.0.5) crass (1.0.3) cucumber (3.1.0) builder (>= 2.1.2) cucumber-core (~> 3.1.0) cucumber-expressions (~> 5.0.4) cucumber-wire (~> 0.0.1) diff-lcs (~> 1.3) gherkin (~> 5.0) multi_json (>= 1.7.5, < 2.0) multi_test (>= 0.1.2) cucumber-core (3.1.0) backports (>= 3.8.0) cucumber-tag_expressions (~> 1.1.0) gherkin (>= 5.0.0) cucumber-expressions (5.0.13) cucumber-rails (1.5.0) capybara (>= 1.1.2, < 3) cucumber (>= 1.3.8, < 4) mime-types (>= 1.17, < 4) nokogiri (~> 1.5) railties (>= 4, < 5.2) cucumber-tag_expressions (1.1.1) cucumber-wire (0.0.1) database_cleaner (1.6.2) delayed_job (4.0.6) activesupport (>= 3.0, < 5.0) delayed_job_active_record (4.1.2) activerecord (>= 3.0, < 5.2) delayed_job (>= 3.0, < 5) diff-lcs (1.3) erubis (2.7.0) gherkin (5.0.0) globalid (0.4.1) activesupport (>= 4.2.0) htmlentities (4.3.4) i18n (0.9.5) concurrent-ruby (~> 1.0) launchy (2.4.3) addressable (~> 2.3) loofah (2.2.2) crass (~> 1.0.2) nokogiri (>= 1.5.9) mail (2.7.0) mini_mime (>= 0.1.1) mime-types (3.1) mime-types-data (~> 3.2015) mime-types-data (3.2016.0521) mimetype-fu (0.1.2) mini_mime (1.0.0) mini_portile2 (2.3.0) minitest (5.11.3) minitest-matchers (1.4.1) minitest (~> 5.0) minitest-rails (2.2.1) minitest (~> 5.7) railties (~> 4.1) multi_json (1.13.1) multi_test (0.1.2) nokogiri (1.8.2) mini_portile2 (~> 2.3.0) public_suffix (3.0.2) rack (1.6.9) rack-test (0.6.3) rack (>= 1.0) rails (4.2.8) actionmailer (= 4.2.8) actionpack (= 4.2.8) actionview (= 4.2.8) activejob (= 4.2.8) activemodel (= 4.2.8) activerecord (= 4.2.8) activesupport (= 4.2.8) bundler (>= 1.3.0, < 2.0) railties (= 4.2.8) sprockets-rails rails-deprecated_sanitizer (1.0.3) activesupport (>= 4.2.0.alpha) rails-dom-testing (1.0.9) activesupport (>= 4.2.0, < 5.0) nokogiri (~> 1.6) rails-deprecated_sanitizer (>= 1.0.1) rails-html-sanitizer (1.0.4) loofah (~> 2.2, >= 2.2.2) railties (4.2.8) actionpack (= 4.2.8) activesupport (= 4.2.8) rake (>= 0.8.7) thor (>= 0.18.1, < 2.0) rake (10.3.2) rspec-core (3.7.1) rspec-support (~> 3.7.0) rspec-expectations (3.7.0) diff-lcs (>= 1.2.0, < 2.0) rspec-support (~> 3.7.0) rspec-mocks (3.7.0) diff-lcs (>= 1.2.0, < 2.0) rspec-support (~> 3.7.0) rspec-rails (3.7.2) actionpack (>= 3.0) activesupport (>= 3.0) railties (>= 3.0) rspec-core (~> 3.7.0) rspec-expectations (~> 3.7.0) rspec-mocks (~> 3.7.0) rspec-support (~> 3.7.0) rspec-support (3.7.1) sprockets (3.7.1) concurrent-ruby (~> 1.0) rack (> 1, < 3) sprockets-rails (3.2.1) actionpack (>= 4.0) activesupport (>= 4.0) sprockets (>= 3.0.0) sqlite3 (1.3.13) thor (0.20.0) thread_safe (0.3.6) tzinfo (1.2.5) thread_safe (~> 0.1) xpath (3.0.0) nokogiri (~> 1.8) PLATFORMS ruby DEPENDENCIES capybara cucumber-rails database_cleaner delayed_job (~> 4.0.6) delayed_job_active_record email_spec! mimetype-fu minitest-matchers minitest-rails rails (= 4.2.8) rake (~> 10.3.2) rspec-rails sqlite3 BUNDLED WITH 1.16.1 email_spec-2.2.0/examples/rails4_root/test/0000755000004100000410000000000013317714226020710 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/test/test_helper.rb0000644000004100000410000000042713317714226023556 0ustar www-datawww-dataENV['RAILS_ENV'] ||= 'test' require File.expand_path('../../config/environment', __FILE__) require 'rails/test_help' require "minitest/autorun" class ActiveSupport::TestCase # Setup all fixtures in test/fixtures/*.yml for all tests in alphabetical order. fixtures :all end email_spec-2.2.0/examples/rails4_root/test/mailers/0000755000004100000410000000000013317714226022344 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/test/mailers/user_mailer_classic_test.rb0000644000004100000410000000120713317714226027740 0ustar www-datawww-datarequire "test_helper" class TestUserMailer < Minitest::Test include ::Rails.application.routes.url_helpers include EmailSpec::Helpers include EmailSpec::Matchers def setup @email = UserMailer.signup("jojo@yahoo.com", "Jojo Binks") end def test_delivered assert_must deliver_to("jojo@yahoo.com"), @email end def test_contains_users_name assert_must have_body_text(/Jojo Binks/), @email end def test_link_to_confirmation_page assert_must have_body_text(/#{confirm_account_url(:host => 'example.com')}/), @email end def test_subject assert_must have_subject(/Account confirmation/), @email end end email_spec-2.2.0/examples/rails4_root/test/mailers/user_mailer_spec_test.rb0000644000004100000410000000165113317714226027254 0ustar www-datawww-datarequire "test_helper" # Uncomment this line when RSpec no longer loads the DSL in every project that # has a dependency on it in the Gemfile, and stomps on MiniTest's `describe`... # This is not an issue for Rails projects that don't use RSpec. #describe UserMailer, :signup do class UserMailerSpec < Minitest::Spec include ::Rails.application.routes.url_helpers include EmailSpec::Helpers include EmailSpec::Matchers let(:email) { UserMailer.signup("jojo@yahoo.com", "Jojo Binks") } def test_is_delivered_to_the_email_passed_in email.must deliver_to("jojo@yahoo.com") end def test_contains_the_users_name_in_the_mail_body email.must have_body_text(/Jojo Binks/) end def test_contains_a_link_to_the_confirmation_page email.must have_body_text(/#{confirm_account_url(:host => 'example.com')}/) end def test_contains_the_correct_subject email.must have_subject(/Account confirmation/) end end email_spec-2.2.0/examples/rails4_root/config.ru0000644000004100000410000000024013317714226021542 0ustar www-datawww-data# This file is used by Rack-based servers to start the application. require ::File.expand_path('../config/environment', __FILE__) run Rails4Root::Application email_spec-2.2.0/examples/rails4_root/spec/0000755000004100000410000000000013317714226020663 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/spec/controllers/0000755000004100000410000000000013317714226023231 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/spec/controllers/welcome_controller_spec.rb0000644000004100000410000000131313317714226030464 0ustar www-datawww-datarequire File.expand_path(File.dirname(__FILE__) + '/../spec_helper') describe WelcomeController, :type => :controller do describe "POST /signup (#signup)" do it "should deliver the signup email" do expect { post :signup, "Email" => "email@example.com", "Name" => "Jimmy Bean" }.to change(ActionMailer::Base.deliveries, :size).by(1) last_delivery = ActionMailer::Base.deliveries.last expect(last_delivery.to).to include "email@example.com" #message is now multipart, make sure both parts include Jimmy Bean expect(last_delivery.parts[0].body.to_s).to include "Jimmy Bean" expect(last_delivery.parts[1].body.to_s).to include "Jimmy Bean" end end end email_spec-2.2.0/examples/rails4_root/spec/models/0000755000004100000410000000000013317714226022146 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/spec/models/user_mailer_spec.rb0000644000004100000410000000416713317714226026024 0ustar www-datawww-datarequire File.dirname(__FILE__) + '/../spec_helper' # These two example groups are specifying the exact same behavior. However, the documentation style is different # and the value that each one provides is different with various trade-offs. Run these examples with the specdoc # formatter to get an idea of how they differ. # Example of documenting the behaviour explicitly and expressing the intent in the example's sentence. describe "Signup Email", :type => :model do include EmailSpec::Helpers include EmailSpec::Matchers include ::Rails.application.routes.url_helpers subject { UserMailer.signup("jojo@yahoo.com", "Jojo Binks") } it "should be delivered to the email passed in" do is_expected.to deliver_to("jojo@yahoo.com") end it "should contain the user's name in the mail body" do is_expected.to have_body_text(/Jojo Binks/) end it "should contain a link to the confirmation page" do is_expected.to have_body_text(/#{confirm_account_url(:host => 'example.com')}/) end it { is_expected.to have_subject(/Account confirmation/) } end # In this example group more of the documentation is placed in the context trying to allow for more concise specs. describe "Signup Email", :type => :model do include EmailSpec::Helpers include EmailSpec::Matchers include ::Rails.application.routes.url_helpers subject { UserMailer.signup("jojo@yahoo.com", "Jojo Binks") } it { is_expected.to have_body_text(/#{confirm_account_url(:host => 'example.com')}/) } it { is_expected.to have_subject(/Account confirmation/) } describe "sent with email address of 'jojo@yahoo.com', and users name 'Jojo Binks'" do subject { UserMailer.signup("jojo@yahoo.com", "Jojo Binks") } it { is_expected.to deliver_to("jojo@yahoo.com") } it { is_expected.to have_body_text(/Jojo Binks/) } end describe "sent with email address of 'jermain@yahoo.com', and users name 'Jermain O'Keefe'" do let(:user_name) { "Jermain O'Keefe" } subject { UserMailer.signup("jermain@yahoo.com", user_name) } it { is_expected.to deliver_to("jermain@yahoo.com") } it { is_expected.to have_body_text(user_name) } end end email_spec-2.2.0/examples/rails4_root/spec/spec_helper.rb0000644000004100000410000000126013317714226023500 0ustar www-datawww-data# This file is copied to ~/spec when you run 'ruby script/generate rspec' # from the project root directory. ENV["RAILS_ENV"] ||= 'test' require File.dirname(__FILE__) + "/../config/environment" require 'rspec/rails' # Requires supporting files with custom matchers and macros, etc, # in ./support/ and its subdirectories. Dir["#{File.dirname(__FILE__)}/support/**/*.rb"].each {|f| require f} RSpec.configure do |config| # == Mock Framework # # If you prefer to use mocha, flexmock or RR, uncomment the appropriate line: # # config.mock_with :mocha # config.mock_with :flexmock # config.mock_with :rr config.mock_with :rspec config.raise_errors_for_deprecations! end email_spec-2.2.0/examples/rails4_root/script/0000755000004100000410000000000013317714226021235 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/script/delayed_job0000755000004100000410000000025713317714226023430 0ustar www-datawww-data#!/usr/bin/env ruby require File.expand_path(File.join(File.dirname(__FILE__), '..', 'config', 'environment')) require 'delayed/command' Delayed::Command.new(ARGV).daemonize email_spec-2.2.0/examples/rails4_root/script/rails0000755000004100000410000000060313317714226022274 0ustar www-datawww-data#!/usr/bin/env ruby # This command will automatically be run when you run "rails" with Rails 4 gems installed from the root of your application. ENV_PATH = File.expand_path('../../config/environment', __FILE__) BOOT_PATH = File.expand_path('../../config/boot', __FILE__) APP_PATH = File.expand_path('../../config/application', __FILE__) require BOOT_PATH require 'rails/commands' email_spec-2.2.0/examples/rails4_root/script/cucumber0000755000004100000410000000046113317714226022771 0ustar www-datawww-data#!/usr/bin/env ruby vendored_cucumber_bin = Dir["#{File.dirname(__FILE__)}/../vendor/{gems,plugins}/cucumber*/bin/cucumber"].first if vendored_cucumber_bin load File.expand_path(vendored_cucumber_bin) else require 'rubygems' unless ENV['NO_RUBYGEMS'] require 'cucumber' load Cucumber::BINARY end email_spec-2.2.0/examples/rails4_root/app/0000755000004100000410000000000013317714226020511 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/helpers/0000755000004100000410000000000013317714226022153 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/helpers/application_helper.rb0000644000004100000410000000003513317714226026340 0ustar www-datawww-datamodule ApplicationHelper end email_spec-2.2.0/examples/rails4_root/app/controllers/0000755000004100000410000000000013317714226023057 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/controllers/welcome_controller.rb0000644000004100000410000000066213317714226027306 0ustar www-datawww-dataclass WelcomeController < ApplicationController def signup UserMailer.signup(params['Email'], params['Name']).deliver_now end def confirm end def newsletter Delayed::Job.enqueue(NotifierJob.new(:newsletter, params['Email'], params['Name'])) end def attachments UserMailer.email_with_attachment(params['Email'], params['Name']) .deliver_now end end email_spec-2.2.0/examples/rails4_root/app/controllers/application_controller.rb0000644000004100000410000000012013317714226030143 0ustar www-datawww-dataclass ApplicationController < ActionController::Base protect_from_forgery end email_spec-2.2.0/examples/rails4_root/app/models/0000755000004100000410000000000013317714226021774 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/models/user.rb0000644000004100000410000000004413317714226023275 0ustar www-datawww-dataclass User < ActiveRecord::Base end email_spec-2.2.0/examples/rails4_root/app/views/0000755000004100000410000000000013317714226021646 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/views/welcome/0000755000004100000410000000000013317714226023301 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/views/welcome/newsletter.html.erb0000644000004100000410000000000013317714226027120 0ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/views/welcome/confirm.html.erb0000644000004100000410000000003213317714226026366 0ustar www-datawww-dataConfirm your new account! email_spec-2.2.0/examples/rails4_root/app/views/welcome/index.html.erb0000644000004100000410000000035013317714226026043 0ustar www-datawww-data<%= form_tag '/welcome/signup' do -%> <%= text_field_tag 'Name' %> <%= text_field_tag 'Email' %>
<%= submit_tag 'Sign up' %>
<% end -%> email_spec-2.2.0/examples/rails4_root/app/views/welcome/attachments.html.erb0000644000004100000410000000000013317714226027237 0ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/views/welcome/signup.html.erb0000644000004100000410000000003613317714226026242 0ustar www-datawww-dataThanks! Go check your email! email_spec-2.2.0/examples/rails4_root/app/views/user_mailer/0000755000004100000410000000000013317714226024155 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/views/user_mailer/email_with_attachment.html.erb0000644000004100000410000000011013317714226032134 0ustar www-datawww-dataHello <%= @name %>! Documents are attached. Regards Rails Example App email_spec-2.2.0/examples/rails4_root/app/views/user_mailer/newsletter.html.erb0000644000004100000410000000011313317714226030001 0ustar www-datawww-dataHello <%= @name %>! This week..... ..... ..... Regards Rails Example App email_spec-2.2.0/examples/rails4_root/app/views/user_mailer/signup.text.erb0000644000004100000410000000016513317714226027141 0ustar www-datawww-dataHello <%= @name %>! <%= link_to "Click here to confirm your account!", confirm_account_url %> This is the text part email_spec-2.2.0/examples/rails4_root/app/views/user_mailer/signup.html.erb0000644000004100000410000000016513317714226027121 0ustar www-datawww-dataHello <%= @name %>! <%= link_to "Click here to confirm your account!", confirm_account_url %> This is the HTML part email_spec-2.2.0/examples/rails4_root/app/mailers/0000755000004100000410000000000013317714226022145 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/app/mailers/user_mailer.rb0000644000004100000410000000153313317714226025003 0ustar www-datawww-dataclass UserMailer < ActionMailer::Base default :from => "admin@example.com", :sent_on => Time.now.to_s def signup(email, name) @name = name mail :to => email, :subject => "Account confirmation" end def newsletter(email, name) @name = name mail :to => email, :subject => "Newsletter sent" end def email_with_attachment(email, name) @name = name add_attachment 'image.png' add_attachment 'document.pdf' mail :to => email, :subject => "Attachments test" end private def add_attachment(attachment_name) attachment_path = "#{Rails.root}/attachments/#{attachment_name}" File.open(attachment_path) do |file| filename = File.basename(file.path) attachments[filename] = {:content_type => File.mime_type?(file), :content => file.read} end end end email_spec-2.2.0/examples/rails4_root/log/0000755000004100000410000000000013317714226020512 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/log/test.log0000644000004100000410000020160113317714226022174 0ustar www-datawww-data  (1.2ms) CREATE TABLE "schema_migrations" ("version" varchar NOT NULL)   (0.5ms) select sqlite_version(*)  (1.6ms) CREATE UNIQUE INDEX "unique_schema_migrations" ON "schema_migrations" ("version") ActiveRecord::SchemaMigration Load (0.1ms) SELECT "schema_migrations".* FROM "schema_migrations" Migrating to CreateUsers (20090125013728)  (0.1ms) begin transaction  (0.5ms) CREATE TABLE "users" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "email" varchar, "name" varchar) SQL (0.2ms) INSERT INTO "schema_migrations" ("version") VALUES (?) [["version", "20090125013728"]]  (1.1ms) commit transaction Migrating to CreateDelayedJobs (20090908054656)  (0.0ms) begin transaction  (0.3ms) CREATE TABLE "delayed_jobs" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "priority" integer DEFAULT 0, "attempts" integer DEFAULT 0, "handler" text, "last_error" text, "run_at" datetime, "locked_at" datetime, "failed_at" datetime, "locked_by" varchar, "created_at" datetime NOT NULL, "updated_at" datetime NOT NULL) SQL (0.1ms) INSERT INTO "schema_migrations" ("version") VALUES (?) [["version", "20090908054656"]]  (1.3ms) commit transaction Migrating to AddQueueToDelayedJobs (20141119224309)  (0.1ms) begin transaction  (0.4ms) ALTER TABLE "delayed_jobs" ADD "queue" varchar SQL (0.2ms) INSERT INTO "schema_migrations" ("version") VALUES (?) [["version", "20141119224309"]]  (1.2ms) commit transaction ActiveRecord::SchemaMigration Load (0.1ms) SELECT "schema_migrations".* FROM "schema_migrations" ActiveRecord::SchemaMigration Load (0.1ms) SELECT "schema_migrations".* FROM "schema_migrations"  (0.1ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/attachments?Email=example%40example.com&Name=Joe+Someone" for 127.0.0.1 at 2018-04-03 19:45:56 -0400 Processing by WelcomeController#attachments as HTML Parameters: {"Email"=>"example@example.com", "Name"=>"Joe Someone"} Rendered user_mailer/email_with_attachment.html.erb (1.2ms) UserMailer#email_with_attachment: processed outbound mail in 272.6ms Sent mail to example@example.com (12.9ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412352d0ae_135413feca083f7bc62981@foo.local.mail> Subject: Attachments test Mime-Version: 1.0 Content-Type: multipart/mixed; boundary="--==_mimepart_5ac412352b663_135413feca083f7bc628dc"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412352b663_135413feca083f7bc628dc Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Documents are attached. Regards Rails Example App ----==_mimepart_5ac412352b663_135413feca083f7bc628dc Content-Type: image/png; charset=binary Content-Transfer-Encoding: base64 Content-Disposition: attachment; filename=image.png Content-ID: <5ac412352e243_135413feca083f7bc6304d@foo.local.mail> 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 C4RMnP8DVvWKdBv0LncAAAAASUVORK5CYII= ----==_mimepart_5ac412352b663_135413feca083f7bc628dc Content-Type: application/pdf; charset=binary Content-Transfer-Encoding: base64 Content-Disposition: attachment; filename=document.pdf Content-ID: <5ac412352e395_135413feca083f7bc631b1@foo.local.mail> JVBERi0xLjQKJcOkw7zDtsOfCjIgMCBvYmoKPDwvTGVuZ3RoIDMgMCBSL0Zp bHRlci9GbGF0ZURlY29kZT4+CnN0cmVhbQp4nEVMPQvCMBDd71e8WWi8S5o0 hRCw2g5uhYCDuFndBLv4901SQQ7ePd4XK8GH3mBwZra3ysC3onqsCy47vDYv 3/qkIZF1yqPrWuWQ7thPAtFIj2tgiY0JrKMENlH/oGXLjrvYZO5roK/cResD H6oylGCFYymfCow8FaGsCW+m216V5F+08ZbONCaaacYXvIksNQplbmRzdHJl YW0KZW5kb2JqCgozIDAgb2JqCjE0NgplbmRvYmoKCjUgMCBvYmoKPDwvTGVu Z3RoIDYgMCBSL0ZpbHRlci9GbGF0ZURlY29kZS9MZW5ndGgxIDIzODY4Pj4K c3RyZWFtCnic7bx7YNxFtTg+M5/Pvl+f3WQf2ednX0m2u5tsXk02TZtPmlfb JDT0mfQSkjRJ25S22bwKRUqDgKUt0ijIQ5QWtYCCsk1LTXlI4FYUsFIviIAC VasCEq1auCok+z0zu+kD0XvvX78/fu4nZ+bMmTOvM2fOnJnPbkaGRvuQFo0h Dkk9W7sTUigoIoR+hBA29WwfES1vlDgAP42Q4o0NiY1bhx553Y+Q6kmEZIc3 btmx4YXXPixHyJBAKPzQpr7u3mc31RoQWmCDOuZvAkLb7A4FpFshHdi0deSa evmvayE9AuklWwZ6ur+y/PnVkIb6UO7W7msS22XrZZA+A2lxW/fWvleuHzUi VMUjZDyVGBiGchxktbxC8xNDfYn3nn+/ENLnoA/1QMPw0I8WUDlNE46XyRVK lVqj1ekNghH9/+wjuw2gGXkAnNwdCOYy9UsAkG/qndllqY9lVyH/7ObUaS4L mL+dAYSC6E50AAXQWVyEnkVTaBl6ANWgVnQHakQvoUeRHu3ALyIe+VEdeggF sQcR1ICsWIbuQa+jK9AQ+g06jfJRE3oLm6CeepRAFhRPvQthE7oldRy41KgW fQc9jrfglagQ8CUkgsPQ8v7UFLKi/NTJ1GuQ+ir6DQ6kDqMlgP0WGVEe2oW+ gExoM3oh9TH0NIDWowfxdfhd5EVdaB9fyu9NXYUWoMfQT3ETYC1oh+w11WNo C5T6OrbiqdTbqd+h7/EY9UFNn0W3QI8n0BQp4GplB5GIctFCdBnqhtzPoNdx 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 j9EyWtlTYP2VAAqURfeArGxJRSs3A6Gw+s1w8TQY/7TpD58EOCGcEJ6jpj+z qvOyTHTVZmfTMAuKIaTAbFFiQkm0a3LWqbNHaRqQc0cpiVK+S/m0WqslvVJZ A9Bq9ZuwLHE47PjuuHXKetbKWenCq24opbFUGV9Qiq0Tut75rVYsWVutXdaE ddx6EBgV2pBbscyHQ255nj87T1eT5c6ugy4p5GqEAzptphoaS/ayBaXjWtyq xV3ahHZce1B7VivTTlgmvk438Q6qT+m9o7rqg+mP8QfpzQIPYpj8MB5qb8fG bIuFbgbF8+leQOdZkRfEn8kpbZytri6w6z02e74RG2W3fVSzpsIVCNirVnPS vY12wZ+gtrsebPdx7lFkQC6izdhuJ0wqmx+2a2uZ3dYKGg2Edp5KjmZSRMqi RJ6x8dagUiMEYcKqp6vTc3WSTdjJC8ZXRfMpn50WdlC52/lsNk/ZWoHNh8Am Ayo/K2koyvNurTZtRGEhhgU6PcLJMGuEWs1605gZP2g5Zvk+fl51wvW6Sm76 nRovUdVb1ppvxreq9hhedyg8UnEZz4znAQ9+zvy8nUgevFQ51xsTT6cibNJU L+exxONTNGzlu/gEP84neTn/vlaCTEl7QEu05+3GULhlmlqCcFMyf2UTHA/X Hda6lx728EvBX34KaVNTiAfwpKYqKiraa9ueRHauGPEomyt+V3jXcVESlLs9 M6DaHdJ87DIF9bkk6MxVB+W5RkO2iFzYLmKLCjCbArAsnSBiBweBWWMVUY4M AipqfMEY3ICHMNWeQUztk2QcJaPya9XX6q81XWMZtY06lR3tHagDjJSkcgrG uAMArPTZwxpqWKgpwcVWizmb2pG83LLS+fOtPvAtssGgzC8Dj4OgU9dftf2l XS9du3Hnj1aWXbX4wGe7r+9v5B69b/ejn/l47NC+b1//t6trqu+77oezbx38 z3O3dtE71S+kzuAB9CzSoLDkRJJcw0kqqbJMJVWXdarwAdWjKqK6Wbv5Wqr6 tO/UCBfFgsW0H34f7QVGhVJNQUFNzbMsLCiUqA77EOLD9B0GiuBrjqMCmMzb K8sKC0ZtI44R53X5iYIvORU7bN8NPJ7/c8fPnW8E5Dl5QkF+bjwYz1uQHytY l9eflygYK9A8h7DdGXI2OX+W83OH7KF8/ELgdesbgdfzXsv/fUDulPyufKXe 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 dTm324ace023ZY0bx20P4G+QR/1H8DP4B4of5LyrPON8TzyHbXKyzLTWtM+z Txzzn/UrjCJ+MnUaiQAemGzkQnTxxAQv7vKOeQnyCl7R2+ql4xr3HvQmvVPe U97T3rNenXeD620DNvzAElQpYHivTWTHaSRVmOIwSI33Rx4tbB77tURbKLBb 8C6UQOMoiabQaaSiBIK+NWy/0U5a7fiAHdsnsVYynZVjJBfkojwml+Qyea2v 9jj5AkpfhQy2THcMDc4MdpwZHJpmV/HV09ODbNmdMWU2X/VKV49r2MXd7oK1 NNgOq6iiogJX0ENkB5whYblR5UaCLQ678dljWXGZIMQx9foEqtVTh4V45pID dpFBnLnhRCWZ64G8XOpF+xRpJQS95JYFX7vxK+9gfHT3d4oiC9xGjd+/qHfh 5ffvWX9ZeSm+4rH/xPK3X8P6/S25hbnm7R73svX3f+Oj2gL67SkdQtxfZLeh +WR/RrNcFRJVEkFtVLPLTrUtQNM2dmawzXkpNrr6mALamPtiy51MvSd5KCHX W1qWF8Vent6Felkd3ijzdKLgHrAbUkA+ZD5ONH09yZD3JQMtHmX1RbHIG2rA QUcmgCBAPkAeKgWX3VDG7kzL5qM8oyvC03u4Qnp3Os3em4HZyvhF7FWEcOK5 YuFEeO5G9YRw4qLb1LZSdptaxkJoMa8UKqVVGvPUzN9Ry6lbo2b+jtrGSDZG sjGSzVZRjr2M7GVkLyN7o3PXr1F6OUwzAPmYeUbRaEX5RZ7R1CVXsYXgJb15 Mm156YWsVFghzStTV3TBYjQEDbljFeMVfLJiquJUBReW49aKrooEJUkVWFTa Qm7jJGeQjL5oyJ23zKcOuYVlfm/InTvJ6aUCf1leQU2pu6wOi3nzERslaJLR KKhzbAHVuBon1digTqgPqF9S8+pJ8pQUjCJvoMATbY12RRNRfiw6HiXJKEZR IToVPRXlo13lD+xiXlLGTZq59AUgnO/hdJ+5zc0c8rPtTplSHnTkOmU5TqxQ 2hUuJ4ZjZBU73LNtgJ4jmf80Hyxw2gBb0ze/1JWau/uVK87fB5dfciHcMvDZ mssSjiy9OibNLjJLxWrOUxcr2rzMHG+YrVzoz7YZPHZzoR6bZLfNrL+2fs0V 0rdmn1wr2pyBQF6ucBmuu/PKwtLls84rCzyBQJa6Yg238KtL0rfIGBkRksfA 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 ZG9iagoKNyAwIG9iago8PC9UeXBlL0ZvbnREZXNjcmlwdG9yL0ZvbnROYW1l L0JBQUFBQStUaW1lc05ld1JvbWFuUFNNVAovRmxhZ3MgNgovRm9udEJCb3hb LTU2OCAtMzA2IDIwMDAgMTAwN10vSXRhbGljQW5nbGUgMAovQXNjZW50IDg5 MQovRGVzY2VudCAtMjE2Ci9DYXBIZWlnaHQgMTAwNgovU3RlbVYgODAKL0Zv bnRGaWxlMiA1IDAgUj4+CmVuZG9iagoKOCAwIG9iago8PC9MZW5ndGggMjk5 L0ZpbHRlci9GbGF0ZURlY29kZT4+CnN0cmVhbQp4nF3RzW7DIAwA4DtPwbE7 VIH0J5sURerSRsphP1q2B0jB6ZAWggg95O0HdrdJOyT6ANsyJqvbY2tNyF79 pDoIfDBWe5inq1fAz3Axlsmca6PCbYV/NfaOZTG3W+YAY2uHqSxZ9hbP5uAX vjro6Qx3LHvxGryxF776qLu47q7OfcEINnDBqoprGGKdp9499yNkmLVudTw2 YVnHlL+A98UBz3EtqRU1aZhdr8D39gKsFKLiZdNUDKz+dyYLSjkP6rP3MVTG UCG291V0jt7vkjfkOnlLbpJ36PyUvCeL5AK9Q99TzW3yA3mffCDL5Ed0gTE1 1cf9I3mTfCJjbw35GC0FOfUgqf9ig5e93SpdO73Lzzi5unofR4mPhzNM0zMW ft/XTS5l4fcNcLKTFwplbmRzdHJlYW0KZW5kb2JqCgo5IDAgb2JqCjw8L1R5 cGUvRm9udC9TdWJ0eXBlL1RydWVUeXBlL0Jhc2VGb250L0JBQUFBQStUaW1l c05ld1JvbWFuUFNNVAovRmlyc3RDaGFyIDAKL0xhc3RDaGFyIDE3Ci9XaWR0 aHNbNzc3IDcyMiA0NDMgMjc3IDUwMCAyNTAgMjUwIDU1NiA3MjIgNTU2IDcy MiAyNzcgNDQzIDQ0MyA1MDAgNzc3CjUwMCAzODkgXQovRm9udERlc2NyaXB0 b3IgNyAwIFIKL1RvVW5pY29kZSA4IDAgUgo+PgplbmRvYmoKCjEwIDAgb2Jq Cjw8L0YxIDkgMCBSCj4+CmVuZG9iagoKMTEgMCBvYmoKPDwvRm9udCAxMCAw IFIKL1Byb2NTZXRbL1BERi9UZXh0XQo+PgplbmRvYmoKCjEgMCBvYmoKPDwv VHlwZS9QYWdlL1BhcmVudCA0IDAgUi9SZXNvdXJjZXMgMTEgMCBSL01lZGlh Qm94WzAgMCA1OTUgODQyXS9Hcm91cDw8L1MvVHJhbnNwYXJlbmN5L0NTL0Rl dmljZVJHQi9JIHRydWU+Pi9Db250ZW50cyAyIDAgUj4+CmVuZG9iagoKNCAw IG9iago8PC9UeXBlL1BhZ2VzCi9SZXNvdXJjZXMgMTEgMCBSCi9NZWRpYUJv eFsgMCAwIDU5NSA4NDIgXQovS2lkc1sgMSAwIFIgXQovQ291bnQgMT4+CmVu ZG9iagoKMTIgMCBvYmoKPDwvVHlwZS9DYXRhbG9nL1BhZ2VzIDQgMCBSCi9P cGVuQWN0aW9uWzEgMCBSIC9YWVogbnVsbCBudWxsIDBdCi9MYW5nKGVuLU5a KQo+PgplbmRvYmoKCjEzIDAgb2JqCjw8L0F1dGhvcjxGRUZGMDA0QjAwNjkw MDY1MDA3MjAwNjEwMDZFMDAyMDAwNTA+Ci9DcmVhdG9yPEZFRkYwMDU3MDA3 MjAwNjkwMDc0MDA2NTAwNzI+Ci9Qcm9kdWNlcjxGRUZGMDA0RjAwNzAwMDY1 MDA2RTAwNEYwMDY2MDA2NjAwNjkwMDYzMDA2NTAwMkUwMDZGMDA3MjAwNjcw MDIwMDAzMzAwMkUwMDMxPgovQ3JlYXRpb25EYXRlKEQ6MjAxMDAzMDYxMzQ0 MDUrMTMnMDAnKT4+CmVuZG9iagoKeHJlZgowIDE0CjAwMDAwMDAwMDAgNjU1 MzUgZiAKMDAwMDAxNjE4OSAwMDAwMCBuIAowMDAwMDAwMDE5IDAwMDAwIG4g CjAwMDAwMDAyMzYgMDAwMDAgbiAKMDAwMDAxNjMzMiAwMDAwMCBuIAowMDAw MDAwMjU2IDAwMDAwIG4gCjAwMDAwMTUyODQgMDAwMDAgbiAKMDAwMDAxNTMw NiAwMDAwMCBuIAowMDAwMDE1NTA0IDAwMDAwIG4gCjAwMDAwMTU4NzIgMDAw MDAgbiAKMDAwMDAxNjEwMiAwMDAwMCBuIAowMDAwMDE2MTM0IDAwMDAwIG4g CjAwMDAwMTY0MzEgMDAwMDAgbiAKMDAwMDAxNjUyOCAwMDAwMCBuIAp0cmFp bGVyCjw8L1NpemUgMTQvUm9vdCAxMiAwIFIKL0luZm8gMTMgMCBSCi9JRCBb IDw2MUFCQTJFMDI3ODBBRTMwODQ1Q0NCQjIwODNFRUYyND4KPDYxQUJBMkUw Mjc4MEFFMzA4NDVDQ0JCMjA4M0VFRjI0PiBdCi9Eb2NDaGVja3N1bSAvRTQ5 OEFCNTkyODgwQkZDRUExRTRDODBEQzE0MjI4RTcKPj4Kc3RhcnR4cmVmCjE2 NzYxCiUlRU9GCg== ----==_mimepart_5ac412352b663_135413feca083f7bc628dc-- Rendered welcome/attachments.html.erb (0.3ms) Completed 200 OK in 300ms (Views: 6.5ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.1ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.6ms) Completed 200 OK in 2ms (Views: 2.0ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.6ms) Rendered user_mailer/signup.text.erb (0.3ms) UserMailer#signup: processed outbound mail in 17.3ms Sent mail to example@example.com (2.2ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412353f58a_135413feca083f7bc6336d@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412353f038_135413feca083f7bc632b5"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412353f038_135413feca083f7bc632b5 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac412353f038_135413feca083f7bc632b5 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412353f038_135413feca083f7bc632b5-- Rendered welcome/signup.html.erb (0.2ms) Completed 200 OK in 22ms (Views: 2.0ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/newsletter?Email=example%40example.com&Name=Joe+Someone" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#newsletter as HTML Parameters: {"Email"=>"example@example.com", "Name"=>"Joe Someone"} Rendered user_mailer/newsletter.html.erb (0.5ms) UserMailer#newsletter: processed outbound mail in 15.0ms Sent mail to example@example.com (1.8ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123547ce8_135413feca083f7bc634ca@foo.local.mail> Subject: Newsletter sent Mime-Version: 1.0 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 Hello Joe Someone! This week..... ..... ..... Regards Rails Example App Rendered welcome/newsletter.html.erb (0.2ms) Completed 200 OK in 27ms (Views: 2.2ms | ActiveRecord: 0.3ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Sent mail to example@example.com (2.5ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412354be2b_135413feca083f7bc63612@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412354b825_135413feca083f7bc63592"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412354b825_135413feca083f7bc63592 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac412354b825_135413feca083f7bc63592 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412354b825_135413feca083f7bc63592-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.1ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412354fd91_135413feca083f7bc63816@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412354f758_135413feca083f7bc63738"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412354f758_135413feca083f7bc63738 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac412354f758_135413feca083f7bc63738 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412354f758_135413feca083f7bc63738-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.5ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.1ms Sent mail to example@example.com (2.5ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123553f64_135413feca083f7bc64039@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412355392f_135413feca083f7bc639d3"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412355392f_135413feca083f7bc639d3 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac412355392f_135413feca083f7bc639d3 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412355392f_135413feca083f7bc639d3-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123557a49_135413feca083f7bc642fb@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac4123557405_135413feca083f7bc6419a"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac4123557405_135413feca083f7bc6419a Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac4123557405_135413feca083f7bc6419a Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac4123557405_135413feca083f7bc6419a-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.1ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.5ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.1ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412355bb51_135413feca083f7bc64428@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412355b4eb_135413feca083f7bc6434c"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412355b4eb_135413feca083f7bc6434c Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac412355b4eb_135413feca083f7bc6434c Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412355b4eb_135413feca083f7bc6434c-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.5ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.3ms Sent mail to example@example.com (2.7ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123564ab4_135413feca083f7bc646cb@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412356441b_135413feca083f7bc645f7"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412356441b_135413feca083f7bc645f7 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac412356441b_135413feca083f7bc645f7 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412356441b_135413feca083f7bc645f7-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 6ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.1ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.1ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123568702_135413feca083f7bc648a2@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac41235680ce_135413feca083f7bc6478f"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac41235680ce_135413feca083f7bc6478f Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the text part ----==_mimepart_5ac41235680ce_135413feca083f7bc6478f Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello ! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac41235680ce_135413feca083f7bc6478f-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412356c62e_135413feca083f7bc65068@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412356c01b_135413feca083f7bc6497e"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412356c01b_135413feca083f7bc6497e Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac412356c01b_135413feca083f7bc6497e Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412356c01b_135413feca083f7bc6497e-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms) Started GET "/confirm" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#confirm as HTML Rendered welcome/confirm.html.erb (0.6ms) Completed 200 OK in 8ms (Views: 8.2ms | ActiveRecord: 0.0ms)  (0.1ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.5ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123574d71_135413feca083f7bc65246@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412357476b_135413feca083f7bc65129"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412357476b_135413feca083f7bc65129 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac412357476b_135413feca083f7bc65129 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412357476b_135413feca083f7bc65129-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms) Started GET "/confirm" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#confirm as HTML Rendered welcome/confirm.html.erb (0.0ms) Completed 200 OK in 0ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.3ms Sent mail to example@example.com (3.0ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac412357d642_135413feca083f7bc654d@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412357cf6f_135413feca083f7bc653ce"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412357cf6f_135413feca083f7bc653ce Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac412357cf6f_135413feca083f7bc653ce Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412357cf6f_135413feca083f7bc653ce-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 6ms (Views: 0.3ms | ActiveRecord: 0.0ms) Started GET "/confirm" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#confirm as HTML Rendered welcome/confirm.html.erb (0.0ms) Completed 200 OK in 0ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.0ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.5ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.3ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 3.0ms Sent mail to example@example.com (3.3ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123583459_135413feca083f7bc65666@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac4123582bcb_135413feca083f7bc655e9"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac4123582bcb_135413feca083f7bc655e9 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac4123582bcb_135413feca083f7bc655e9 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac4123582bcb_135413feca083f7bc655e9-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 7ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started GET "/confirm" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#confirm as HTML Rendered welcome/confirm.html.erb (0.0ms) Completed 200 OK in 0ms (Views: 0.2ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (0.1ms) begin transaction  (0.0ms) commit transaction  (0.0ms) begin transaction Started GET "/" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#index as HTML Rendered welcome/index.html.erb (0.2ms) Completed 200 OK in 1ms (Views: 0.4ms | ActiveRecord: 0.0ms) Started POST "/welcome/signup" for 127.0.0.1 at 2018-04-03 19:45:57 -0400 Processing by WelcomeController#signup as HTML Parameters: {"utf8"=>"✓", "Name"=>"Joe Someone", "Email"=>"example@example.com", "commit"=>"Sign up"} Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Sent mail to example@example.com (2.6ms) Date: Tue, 03 Apr 2018 19:45:57 -0400 From: admin@example.com To: example@example.com Message-ID: <5ac4123588678_135413feca083f7bc65859@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412358804e_135413feca083f7bc6574b"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:45:56 -0400 ----==_mimepart_5ac412358804e_135413feca083f7bc6574b Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the text part ----==_mimepart_5ac412358804e_135413feca083f7bc6574b Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Joe Someone! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412358804e_135413feca083f7bc6574b-- Rendered welcome/signup.html.erb (0.0ms) Completed 200 OK in 5ms (Views: 0.3ms | ActiveRecord: 0.0ms)  (0.0ms) rollback transaction  (3.9ms) CREATE TABLE "delayed_jobs" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "priority" integer DEFAULT 0, "attempts" integer DEFAULT 0, "handler" text, "last_error" text, "run_at" datetime, "locked_at" datetime, "failed_at" datetime, "locked_by" varchar, "created_at" datetime NOT NULL, "updated_at" datetime NOT NULL, "queue" varchar)   (1.5ms) CREATE TABLE "users" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "email" varchar, "name" varchar)  (1.6ms) CREATE TABLE "schema_migrations" ("version" varchar NOT NULL)   (0.1ms) select sqlite_version(*)  (1.6ms) CREATE UNIQUE INDEX "unique_schema_migrations" ON "schema_migrations" ("version")  (0.2ms) SELECT version FROM "schema_migrations"  (1.4ms) INSERT INTO "schema_migrations" (version) VALUES ('20141119224309')  (1.3ms) INSERT INTO "schema_migrations" (version) VALUES ('20090125013728')  (1.2ms) INSERT INTO "schema_migrations" (version) VALUES ('20090908054656') Processing by WelcomeController#signup as HTML Parameters: {"Email"=>"email@example.com", "Name"=>"Jimmy Bean"} Rendered user_mailer/signup.html.erb (1.4ms) Rendered user_mailer/signup.text.erb (0.6ms) UserMailer#signup: processed outbound mail in 212.8ms Sent mail to email@example.com (5.3ms) Date: Tue, 03 Apr 2018 19:46:01 -0400 From: admin@example.com To: email@example.com Message-ID: <5ac4123990ac6_135473fe41643f7bc835fe@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac412398fdd5_135473fe41643f7bc8344a"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:46:01 -0400 ----==_mimepart_5ac412398fdd5_135473fe41643f7bc8344a Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Jimmy Bean! Click here to confirm your account! This is the text part ----==_mimepart_5ac412398fdd5_135473fe41643f7bc8344a Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Jimmy Bean! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac412398fdd5_135473fe41643f7bc8344a-- Template rendering was prevented by rspec-rails. Use `render_views` to verify rendered view contents if necessary. Rendered welcome/signup.html.erb (0.2ms) Completed 200 OK in 236ms (Views: 13.2ms | ActiveRecord: 0.0ms) Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.7ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.7ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.5ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.4ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.4ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.5ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.5ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.2ms) UserMailer#signup: processed outbound mail in 2.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Processing by WelcomeController#signup as HTML Parameters: {"Email"=>"email@example.com", "Name"=>"Jimmy Bean"} Rendered user_mailer/signup.html.erb (1.6ms) Rendered user_mailer/signup.text.erb (0.5ms) UserMailer#signup: processed outbound mail in 238.6ms Sent mail to email@example.com (6.8ms) Date: Tue, 03 Apr 2018 19:46:03 -0400 From: admin@example.com To: email@example.com Message-ID: <5ac4123bb2fc1_1354b3ff3bac3f79c90757@foo.local.mail> Subject: Account confirmation Mime-Version: 1.0 Content-Type: multipart/alternative; boundary="--==_mimepart_5ac4123bb1f14_1354b3ff3bac3f79c90672"; charset=UTF-8 Content-Transfer-Encoding: 7bit sent-on: 2018-04-03 19:46:03 -0400 ----==_mimepart_5ac4123bb1f14_1354b3ff3bac3f79c90672 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Jimmy Bean! Click here to confirm your account! This is the text part ----==_mimepart_5ac4123bb1f14_1354b3ff3bac3f79c90672 Content-Type: text/html; charset=UTF-8 Content-Transfer-Encoding: 7bit Hello Jimmy Bean! Click here to confirm your account! This is the HTML part ----==_mimepart_5ac4123bb1f14_1354b3ff3bac3f79c90672-- Template rendering was prevented by rspec-rails. Use `render_views` to verify rendered view contents if necessary. Rendered welcome/signup.html.erb (0.4ms) Completed 200 OK in 262ms (Views: 11.5ms | ActiveRecord: 0.0ms) Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.2ms) UserMailer#signup: processed outbound mail in 2.3ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 2.0ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.9ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.8ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.7ms ActiveRecord::SchemaMigration Load (0.1ms) SELECT "schema_migrations".* FROM "schema_migrations" Rendered user_mailer/signup.html.erb (1.8ms) Rendered user_mailer/signup.text.erb (0.5ms) UserMailer#signup: processed outbound mail in 240.6ms Rendered user_mailer/signup.html.erb (0.2ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.9ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.6ms Rendered user_mailer/signup.html.erb (0.1ms) Rendered user_mailer/signup.text.erb (0.1ms) UserMailer#signup: processed outbound mail in 1.7ms email_spec-2.2.0/examples/rails4_root/log/development.log0000644000004100000410000000000013317714226023525 0ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/db/0000755000004100000410000000000013317714226020316 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/db/schema.rb0000644000004100000410000000237313317714226022110 0ustar www-datawww-data# encoding: UTF-8 # 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 that you check this file into your version control system. ActiveRecord::Schema.define(version: 20141119224309) do create_table "delayed_jobs", force: :cascade do |t| t.integer "priority", default: 0 t.integer "attempts", default: 0 t.text "handler" t.text "last_error" t.datetime "run_at" t.datetime "locked_at" t.datetime "failed_at" t.string "locked_by" t.datetime "created_at" t.datetime "updated_at" t.string "queue" end create_table "users", force: :cascade do |t| t.string "email" t.string "name" end end email_spec-2.2.0/examples/rails4_root/db/migrate/0000755000004100000410000000000013317714226021746 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/db/migrate/20141119224309_add_queue_to_delayed_jobs.rb0000644000004100000410000000027313317714226031105 0ustar www-datawww-dataclass AddQueueToDelayedJobs < ActiveRecord::Migration def self.up add_column :delayed_jobs, :queue, :string end def self.down remove_column :delayed_jobs, :queue end end email_spec-2.2.0/examples/rails4_root/db/migrate/20090125013728_create_users.rb0000644000004100000410000000026513317714226026431 0ustar www-datawww-dataclass CreateUsers < ActiveRecord::Migration def self.up create_table :users do |t| t.string :email, :name end end def self.down drop_table :users end end email_spec-2.2.0/examples/rails4_root/db/migrate/20090908054656_create_delayed_jobs.rb0000644000004100000410000000210513317714226027725 0ustar www-datawww-dataclass CreateDelayedJobs < ActiveRecord::Migration def self.up create_table :delayed_jobs, :force => true do |table| table.integer :priority, :default => 0 # Allows some jobs to jump to the front of the queue table.integer :attempts, :default => 0 # Provides for retries, but still fail eventually. table.text :handler # YAML-encoded string of the object that will do work table.text :last_error # reason for last failure (See Note below) table.datetime :run_at # When to run. Could be Time.now for immediately, or sometime in the future. table.datetime :locked_at # Set when a client is working on this object table.datetime :failed_at # Set when all retries have failed (actually, by default, the record is deleted instead) table.string :locked_by # Who is working on this object (if locked) table.timestamps null: false end end def self.down drop_table :delayed_jobs end end email_spec-2.2.0/examples/rails4_root/db/test.sqlite30000644000004100000410000006000013317714226022577 0ustar www-datawww-dataSQLite format 3@ .  5 G =//indexunique_schema_migrationsschema_migrationsCREATE UNIQUE INDEX "unique_schema_migrations" ON "schema_migrations" ("version")l//tableschema_migrationsschema_migrationsCREATE TABLE "schema_migrations" ("version" varchar NOT NULL)~[tableusersusersCREATE TABLE "users" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "email" varchar, "name" varchar)P++Ytablesqlite_sequencesqlite_sequenceCREATE TABLE sqlite_sequence(name,seq)v%%/tabledelayed_jobsdelayed_jobsCREATE TABLE "delayed_jobs" ("id" INTEGER PRIMARY KEY AUTOINCREMENT NOT NULL, "priority" integer DEFAULT 0, "attempts" integer DEFAULT 0, "handler" text, "last_error" text, "run_at" datetime, "locked_at" datetime, "failed_at" datetime, "locked_by" varchar, "created_at" datetime NOT NULL, "updated_at" datetime NOT NULL, "queue" varchar)    )20090908054656)20090125013728)20141119224309 )20090908054656)20090125013728) 20141119224309email_spec-2.2.0/examples/rails4_root/db/seeds.rb0000644000004100000410000000054113317714226021746 0ustar www-datawww-data# This file should contain all the record creation needed to seed the database with its default values. # The data can then be loaded with the rake db:seed (or created alongside the db with db:setup). # # Examples: # # cities = City.create([{ :name => 'Chicago' }, { :name => 'Copenhagen' }]) # Mayor.create(:name => 'Daley', :city => cities.first) email_spec-2.2.0/examples/rails4_root/attachments/0000755000004100000410000000000013317714226022244 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/attachments/image.png0000644000004100000410000000574113317714226024043 0ustar www-datawww-dataPNG  IHDRddpT pHYs   IDATx]o\G?~?b;64DiJB[&g $*!@ !QPm8BM;ftZ߻wg93gnK|ƳOO#=c{loOիݔ{]]t~jinr~l5[Z[f=9oR^^F0 \h?gU1wR?{}5A/ <=ePLg])O՘P,YR5 cMdG芋/-eSot+NYOs+_RϟC_Tښjl/>Ͽu_e)?0*R('ˏC ʈ{' A2 ()kn*KTSVxWu7S40x=pʀ`hCDYbh0vs%@GbhTDP̝)Y`-fA (w}dQU̝IYeMM$xvs%@fPt/N呲 d F,%@P oɮWU̝uyyYYYU-ثOIJKUM)BK!t'UYV2HR66.<6)0exKs#}kG(UQeVVD4ׇnply a> stream xEL= 0WYhK4nYMRA=+`pfz .;6/!uʣZOHk` 6QeˎdkE2`c) < o^_4&i,5 endstream endobj 3 0 obj 146 endobj 5 0 obj <> stream x{`E8>3Ͼ_dg_Il^M6MOW$44I۔ټ EJ-(CMKMyHVR/UZ*$=3{_?~'g̙33gΜnFF!I=[R(("~6l-o8? [yݏId7nٱ>,GȐ@(Цg7Z`:oB[!شuzk!=%[zՐPkee>iq[־W5"T#d<rdBC}@ЏP9MJZ # y2KoeeW!i. NtY\EShzՠVtjD/G/"QzԀXA+ :Qz z@O a%uԨ}=%$k* %QڅLh3z!14֣u]E]h_M]Oq`-h5ch :۩ߡAME@')jerBtϠq.T^jq>L9N}xA` ._kз&4Uރt"b%VjۏAG)܄~;$VSR)4Ags8<Fx7?"+F؋N@?Kr=ٕZz(yPCh;} fYt DT}ٵ/lsbr^ uY@ 4bFQ/+F߉'u"'^2H܋/2Yj 7Gk&A_>f0Wd%w3Xӳ()@A[ ?b !7ak899se\ kn~ߐ-uVtnI)uVBCTA66]K3C76З/ayS&zfa/Z Zw3 {#}<%>O>OI-i ANW;vqc^)Y1w&Ąmq/Jŝn4Q I &D?=f+It©Q;#%fJΒܓS\`% ]9ݙ̢m y` b^=(:ȋoiCCq[P܍K^-`? ěQ!րBv~a!>_V ]EXYS(9`u@+d/؊/Z t%9 G"XA(%HsELN"Xw6aj=yԟ ܭHݘڇ:S@C`&|[N|) ؏~^{6<߁/=?Y5Sdy@Ba=?ܖpSd2r8%`z]z0j),Bg e@wH B\+d^%U/ZX2^Q>(VXr+z.ÞcZLFi5jR!(Ro]I>׿dI@辈Еp)ORbl⥜pnsbABUшX'$^wyi0|:^( 6ՉI%'o[U֨k}hVkufW&HN%d {9h$k{ȿ8i3T˚Ik ֌OG#S{o}E[nmn]zۅ$Tnm}q[oirbmziu$Iko4|+imےfhPcJ_O)]Ťʿؿi.$Z;aKS^ܻMV;uhGr$1Ҝh`LސAy ckZq^ȿ!)Г6?}hoOCd/GRU۵W@'eA/߿ҝȃTK+p89oUE-(qKE#'IҟD@|d^YzzH$.oKE1p{tќj36sx(;U2-"/ߴt6~oWFM.I+e0' HjToź6J?Y_ߵ1U9H{#U{iMKr %(0`!)t-IjYh2ubхb1%+×\{ڽt%Mݫ$޽ ~aoz(ڸL=ϑl Wjra ߲r]qj v-n#$.Q)Yj9L{*HOZA0QeOC>A! _laê˄sU-3Upcb^;yM},GL:]U8ڿ6J:h2cYYdut2dhfj Aʢ,اپ;Vh25+ -*j*Wf!  jS0=,aL TQ,O5nsJ[%&C<Wqx}"xj?pxA#y ~?p,ZPQht(MY>">,bO$d97}e]_-¹!*|ap\PtcR8)0h^ci^n^_5} l̶ZvpOR>Y"uj1X,,ZujO$y227gf/*y"*ז! D ߴɛcV/mL'kWg{..b}Eͩ3J.ajwuGWZHkGzN1 7rH.`jhV6p"F4*BPͦ)OJwM&@4T=;AwtLS])< Z8,gr̖DDh'-*i#R@H*`v) M)`hsRN'_q8) 5h![(I?o2䩓׎bWI蟔cutW9Y[,3R.+UzjuiH~~)#WPڲtZîJP)2Rr|vpacY o_,SKQ=Bc&©_:--j Udj=dɹ'o'ߑsl)FS 2Hc*qZb;liWLqSym3o6;^1੉8DLd5i)Ne|QHiw <`eo,粳ǔ8Wrm%'(%%DTVʕқyL1osH/E=wV'*]d  V 3Mw i!R!#LQmĀVhstѢ 4؁kێ&d}p(laǑZZ)("iHLJN)KIzU,sDc\aF>Yr0 ˭rGJM%s "{{w91pٍؗ 9=WHhd=cŋߪl TH7ΰAv2/Wqd9[+ (橑rBdGV{G) _|Z[2ZS` ElIE+7p48!?Ltfg0 !lQbBIkr֩GisG)RKZ%RYj,q8uzY«n(T_PV,Y[]քuzڐ[̇Cny?;OWή.)j:mKZܪ]ڄv\{P{V+NX&N7O齣b0joln^@Yğ)m.=6{e}TثVsҽvlqQd@.n'L*kk  vJfRDʢD֠R#aªsuM WE)vPl6OZ͇&*?+i(n6mDa!:=0kZzӘ?h9f>~^uJn/Q[֚oƷ^w(-/S_]/]qV]gooxоo_kosv;/,ҠD\I*L%Uuգ*YZłŴ~FRMAAMͳ,,(k+ Fm#u/9;l =u|rC-0ͦ,)`ٹns !p=k#+gE31:K.ț̂B#@5DNq /|gfGk;xg ˽ܖ]b$w{/{wl9Eװ}WNbT읯Ud!wws׹??R|brL6&߯دT*ԪyDj'qS:.ɽPJHxn#GJ-/#A(|!O`"U0 BR(CrG+ ďIHϦ(gŦň.6E an3gKL -,L(FCp':e%F~ˢz.yu @#?22ٲ˂k.:mI㳯vS( IʮBN!n7qD.v;IR @ UInΠtZ\ȓc`4%b8M8e3y#}@7fRo0[i͆,n;NK^mFGbm,x"&iݝ&[y".!Tg@q2CԽnX+nso6l6 cn^naiϽ{{ Oُ_4 ׄqWo#*Cxܘ nJv,Nâ$ 5n ng p^?IĝM:PZpTJ-RtNK*!qĎ,wc$y_ҋrV\LǴ.L uyj ,)ݡ/wvې0 w ;OT)<vUnJ/_K%{pߜgW)Y+g޽<^])T[vrY0x#XG3v#;'R?x&Ad hm$5gaYCBi2ףtqfQ,UQF l~;&9O΀ZNȜޜ+g1!Ǫ޿p@>л+vHhl4$'CLI}Wsde?*hO!CFV˙9d ngNutL 'AH;q'!ծ{R-X( b'z(1Se+a%qQ溉*(RP|(+J*z\.M4LZ-ˆmjc6!VTExT^>J+f@⯄OҦKᩩ7JbZ9DO|!15g v%C4.RUfY^5WJ{\J^!"Rܤ^YZT^[tZFͪ7i ,7Zt)KPiUA(Zv -=SŵxZY&h[DKˉ,ڮU6ziė:m6жFl{LG ;hmk`*]ATjKKASވ(H[Q y)x6HƂ8(PNf"/Edte9)9Ͽ@w eJ5&z*'=mQw!yn DMv#}tF yzB`1}\K4-NKM&jV'v&,ϧb5*UKn+rX{eu VRL3w̼{=uM?3Ћ]Y%k/[4AkJl(wE׭hi1糛kYسTF׌<7>lκqGUN+' 32X^sk/q[˛/EEujeȀpZW˓^"wT\`PO氉O2ZůR9ڜ14=xJ<~#SFƶu9ۆ{MemoGG3伫<|Od>N4}=ɐ%-eEjAG& @>@*PL<+{Bzw:ޛEUpbDxFpԶRvZBh1*UQ˩[fH6F1VQed/#{sׯQz9L3yFhEE%W%y2my酬TX!+SWtb4 c|bT֊%IXTBn$ghȝ̧e~oȝ;Y^AMy%h(slո'ؠN_RI"omvEQ~,:%(FQ!:=]b^RM tNRt:e9NP.'cd;ܳm#4,p[7ԕ+_r!2ٚ,:&.2KjS+ڼo\϶uYFVn::yH!i(H2KZh),_2 Xl\Z<wV7ق 8z!]̭l^ͼ0P8!Yz])V^55<0[aF װT K48ckXʩ!EdȟYnyy[3=Qb!,-CzؘOVWiU bwrl`~ҟv8*Ni"`Đ6I>>+4搻qr[2CnpWuG5!w "꼖UuPy+"ظf-`D(LPYV]01'ĤHI\&C@E'ʓ夜,-kk͞22BPBZ`;m)mjk$xLޛ/5(-}I-˙tTuY}_oO5kO6gZ. h|VNvg9`_S>lzmy"4?MLۤkn*YsymMKz٪^w)selQs\#F/[)j]P]lW)vggy45nv&*xo@*[ 61WHrEg;ǂʻ"C L ;%+!-Yd1!%3IJf(-A0¾c3O#' r)XaŪpb[(}М Qۦ%BI1jHTEXGn|iAR b=B-),L>Bx"!P"݄ fB ]1vRpցtmJf)j(-,da& q.btRjh\1PY] e]ֲDx,ccJɓeH was)-!!:B22/Bnu(ʛWs98 zP+qR ʄ%%#T ZC]z,4J8B$ľ >Uv{d|0:Ln1A6+bϡ7Ŧ-E)Z< K*Q5-WoBn(!mZSu쎵5w~vaqJkū-Yo )"4J؍MDokQ]6R+󴔞5ց4PP),;dx ˂yݸǍAю ;4DGtGG!uuj0|+K <%?6ȼ"ES _%ȂunqwФTq bR@Fyy%O@'::E/CHNL&ӻ-].^ᎀL|$Q2V"7LbQ E^w#Su#-3'1}1_"Qs*ܩ_,&ŕrDQݣW{@c #"K:K ]!%pujC"7z z`%<;)c.hfS3~7H?@B_cGlK!, cK޷rI.[$ ː08$/s 7VJW7z!( ]yHDUR˜8Jz=T)>>4~wÆ z|i\$:XFX{s:HO(W#MKeV'knI/bq,6_\Ol60#' %G5[+7nW m-XտoXjqm+6n+_l1_oh bEb~K'/f+(B&k]ٲPwoā t4b6qVoWt@Mz {F JVѠnm6QZ-};Pgo6"KFŭ;>t#ow֯n!H Q\gEe!ֳe É- JCpA8-;طu=-umsܟ%K<7 By&L] XTCj۶ t_(t;U10:{SϦ-OfՏuݛAe.oΖm4e04qq{#F77r-T>..i:ώӼ/F~!KZZΤ^כp!655 88ZŽ穮1sgP.: #(` S#$O䇊ӌ )GŽ=aq&/ +ȑyk[{{4:_P|F] #:ȽIG~/pσh'tbw yccǎŨfvb;_!M/g@ݙ.; 6Lz;7ʍd#Ը!_p ;F@-C5vNxlv?v㮛.:ㆡaz3 p a;'!87A8pA!nDl㑸T\D-q上_HT!gbc}GTZJ;bwcສF az&ǹV%]d%cu4WZTEQ0g ?Ks WTzx³X)t)1PPh4h syL~eB8Cx4L'= Y7BA1Qu:2|J?jeौLLDCNDv@_F]3@}"/ND@"H;yjHO(o ړLK-7 ODh:$AG{$V֜gB~VY(b=6됏 Ph翫G`}_? [_{~rkRdy~$^3ᙊL*!$yK1ϣoY!?Tzȓh+x dGy4'1dKUИT%fwUm{n]~]M|[>]ZMDm5Ѷh[HV,Mmh1C۾"$űk8`҄ۓZfEk54 V'ɲ]x?l/Fad;7<<a4} G?T#0f?0+=ãD010-6JSc endstream endobj 6 0 obj 14943 endobj 7 0 obj <> endobj 8 0 obj <> stream x]n ;O;T'EFa?ZH=vI;$2&ckM^:|0V{Wp1ɜkm5e1[ck,Y: w,{}6pkb=#dnu<6aYǔq-5i]eT w& J91TP!Ut7:yKnw'CSmy| Gt15#y|"co -9 b*];8zG3L3~M.e p endstream endobj 9 0 obj <> endobj 10 0 obj <> endobj 11 0 obj <> endobj 1 0 obj <>/Contents 2 0 R>> endobj 4 0 obj <> endobj 12 0 obj <> endobj 13 0 obj < /Creator /Producer /CreationDate(D:20100306134405+13'00')>> endobj xref 0 14 0000000000 65535 f 0000016189 00000 n 0000000019 00000 n 0000000236 00000 n 0000016332 00000 n 0000000256 00000 n 0000015284 00000 n 0000015306 00000 n 0000015504 00000 n 0000015872 00000 n 0000016102 00000 n 0000016134 00000 n 0000016431 00000 n 0000016528 00000 n trailer < <61ABA2E02780AE30845CCBB2083EEF24> ] /DocChecksum /E498AB592880BFCEA1E4C80DC14228E7 >> startxref 16761 %%EOF email_spec-2.2.0/examples/rails4_root/Rakefile0000644000004100000410000000052313317714226021376 0ustar www-datawww-data# Add your own tasks in files placed in lib/tasks ending in .rake, # for example lib/tasks/capistrano.rake, and they will automatically be available to Rake. require File.expand_path('../config/application', __FILE__) require 'rake' require 'rake/testtask' require 'rdoc/task' require 'delayed/tasks' Rails4Root::Application.load_tasks email_spec-2.2.0/examples/rails4_root/lib/0000755000004100000410000000000013317714226020477 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/lib/tasks/0000755000004100000410000000000013317714226021624 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/lib/tasks/cucumber.rake0000644000004100000410000000453113317714226024300 0ustar www-datawww-data# IMPORTANT: This file is generated by cucumber-rails - edit at your own peril. # It is recommended to regenerate this file in the future when you upgrade to a # newer version of cucumber-rails. Consider adding your own code to a new file # instead of editing this one. Cucumber will automatically load all features/**/*.rb # files. unless ARGV.any? {|a| a =~ /^gems/} # Don't load anything when running the gems:* tasks vendored_cucumber_bin = Dir["#{Rails.root}/vendor/{gems,plugins}/cucumber*/bin/cucumber"].first $LOAD_PATH.unshift(File.dirname(vendored_cucumber_bin) + '/../lib') unless vendored_cucumber_bin.nil? begin require 'cucumber/rake/task' namespace :cucumber do Cucumber::Rake::Task.new({:ok => 'db:test:prepare'}, 'Run features that should pass') do |t| t.binary = vendored_cucumber_bin # If nil, the gem's binary is used. t.fork = true # You may get faster startup if you set this to false t.profile = 'default' end Cucumber::Rake::Task.new({:wip => 'db:test:prepare'}, 'Run features that are being worked on') do |t| t.binary = vendored_cucumber_bin t.fork = true # You may get faster startup if you set this to false t.profile = 'wip' end Cucumber::Rake::Task.new({:rerun => 'db:test:prepare'}, 'Record failing features and run only them if any exist') do |t| t.binary = vendored_cucumber_bin t.fork = true # You may get faster startup if you set this to false t.profile = 'rerun' end desc 'Run all features' task :all => [:ok, :wip] task :statsetup do require 'rails/code_statistics' ::STATS_DIRECTORIES << %w(Cucumber\ features features) if File.exist?('features') ::CodeStatistics::TEST_TYPES << "Cucumber features" if File.exist?('features') end end desc 'Alias for cucumber:ok' task :cucumber => 'cucumber:ok' task :default => :cucumber task :features => :cucumber do STDERR.puts "*** The 'features' task is deprecated. See rake -T cucumber ***" end # In case we don't have ActiveRecord, append a no-op task that we can depend upon. task 'db:test:prepare' do end task :stats => 'cucumber:statsetup' rescue LoadError desc 'cucumber rake task not available (cucumber not installed)' task :cucumber do abort 'Cucumber rake task is not available. Be sure to install cucumber as a gem or plugin' end end end email_spec-2.2.0/examples/rails4_root/lib/tasks/rspec.rake0000644000004100000410000000527513317714226023615 0ustar www-datawww-databegin require 'rspec/core' require 'rspec/core/rake_task' rescue MissingSourceFile module RSpec module Core class RakeTask def initialize(name) task name do # if rspec-rails is a configured gem, this will output helpful material and exit ... require File.expand_path(File.dirname(__FILE__) + "/../../config/environment") # ... otherwise, do this: raise <<-MSG #{"*" * 80} * You are trying to run an rspec rake task defined in * #{__FILE__}, * but rspec can not be found in vendor/gems, vendor/plugins or system gems. #{"*" * 80} MSG end end end end end end Rake.application.instance_variable_get('@tasks').delete('default') spec_prereq = File.exist?(File.join(Rails.root, 'config', 'database.yml')) ? "db:test:prepare" : :noop task :noop do end task :default => :spec task :stats => "spec:statsetup" desc "Run all specs in spec directory (excluding plugin specs)" RSpec::Core::RakeTask.new(:spec => spec_prereq) namespace :spec do [:requests, :models, :controllers, :views, :helpers, :mailers, :lib].each do |sub| desc "Run the code examples in spec/#{sub}" RSpec::Core::RakeTask.new(sub => spec_prereq) do |t| t.pattern = "./spec/#{sub}/**/*_spec.rb" end end task :statsetup do require 'rails/code_statistics' ::STATS_DIRECTORIES << %w(Model\ specs spec/models) if File.exist?('spec/models') ::STATS_DIRECTORIES << %w(View\ specs spec/views) if File.exist?('spec/views') ::STATS_DIRECTORIES << %w(Controller\ specs spec/controllers) if File.exist?('spec/controllers') ::STATS_DIRECTORIES << %w(Helper\ specs spec/helpers) if File.exist?('spec/helpers') ::STATS_DIRECTORIES << %w(Library\ specs spec/lib) if File.exist?('spec/lib') ::STATS_DIRECTORIES << %w(Mailer\ specs spec/mailers) if File.exist?('spec/mailers') ::STATS_DIRECTORIES << %w(Routing\ specs spec/routing) if File.exist?('spec/routing') ::STATS_DIRECTORIES << %w(Request\ specs spec/requests) if File.exist?('spec/requests') ::CodeStatistics::TEST_TYPES << "Model specs" if File.exist?('spec/models') ::CodeStatistics::TEST_TYPES << "View specs" if File.exist?('spec/views') ::CodeStatistics::TEST_TYPES << "Controller specs" if File.exist?('spec/controllers') ::CodeStatistics::TEST_TYPES << "Helper specs" if File.exist?('spec/helpers') ::CodeStatistics::TEST_TYPES << "Library specs" if File.exist?('spec/lib') ::CodeStatistics::TEST_TYPES << "Mailer specs" if File.exist?('spec/mailer') ::CodeStatistics::TEST_TYPES << "Routing specs" if File.exist?('spec/routing') ::CodeStatistics::TEST_TYPES << "Request specs" if File.exist?('spec/requests') end end email_spec-2.2.0/examples/rails4_root/lib/notifier_job.rb0000644000004100000410000000023113317714226023471 0ustar www-datawww-dataclass NotifierJob < Struct.new(:notifier_method,:username,:name) def perform UserMailer.send(notifier_method,username, name).deliver_now end end email_spec-2.2.0/examples/rails4_root/config/0000755000004100000410000000000013317714226021176 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/config/boot.rb0000644000004100000410000000051213317714226022464 0ustar www-datawww-data# Use locked gems if present. begin require File.expand_path('../../.bundle/environment', __FILE__) rescue LoadError # Otherwise, use RubyGems. require 'rubygems' # And set up the gems listed in the Gemfile. if File.exist?(File.expand_path('../../Gemfile', __FILE__)) require 'bundler' Bundler.setup end end email_spec-2.2.0/examples/rails4_root/config/routes.rb0000644000004100000410000000051013317714226023040 0ustar www-datawww-dataRails4Root::Application.routes.draw do root :to => "welcome#index" get "/confirm" => "welcome#confirm", :as => :confirm_account get "/newsletter" => "welcome#newsletter", :as => :request_newsletter get "/attachments" => "welcome#attachments", :as => :request_attachments post "/welcome/signup" => "welcome#signup" end email_spec-2.2.0/examples/rails4_root/config/secrets.yml0000644000004100000410000000170413317714226023373 0ustar www-datawww-data# Be sure to restart your server when you modify this file. # Your secret key is used for verifying the integrity of signed cookies. # If you change this key, all old signed cookies will become invalid! # Make sure the secret is at least 30 characters and all random, # no regular words or you'll be exposed to dictionary attacks. # You can use `rake secret` to generate a secure secret key. # Make sure the secrets in this file are kept private # if you're sharing your code publicly. development: secret_key_base: c74ed16a9b39b24630690494ff39823193ae23274ffc6a4960554635220056e2674288a77afb2c2127b70763f737d3866b8869b3d38b9c55ce4e36ff919cd430 test: secret_key_base: 4edb6ae3a2e2953fc86c07b7b6e0afbef95e9100b04c7b34423b2f70feb9c9992d455bf593082c1e1351745e19be693430921040ec29a2bb1ad28ccc657c3ef2 # Do not keep production secrets in the repository, # instead read values from the environment. production: secret_key_base: <%= ENV["SECRET_KEY_BASE"] %> email_spec-2.2.0/examples/rails4_root/config/initializers/0000755000004100000410000000000013317714226023704 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/config/initializers/backtrace_silencers.rb0000644000004100000410000000062413317714226030221 0ustar www-datawww-data# Be sure to restart your server when you modify this file. # You can add backtrace silencers for libraries that you're using but don't wish to see in your backtraces. # Rails.backtrace_cleaner.add_silencer { |line| line =~ /my_noisy_library/ } # You can also remove all the silencers if you're trying to debug a problem that might stem from framework code. # Rails.backtrace_cleaner.remove_silencers! email_spec-2.2.0/examples/rails4_root/config/initializers/inflections.rb0000644000004100000410000000057113317714226026551 0ustar www-datawww-data# Be sure to restart your server when you modify this file. # Add new inflection rules using the following format # (all these examples are active by default): # ActiveSupport::Inflector.inflections do |inflect| # inflect.plural /^(ox)$/i, '\1en' # inflect.singular /^(ox)en/i, '\1' # inflect.irregular 'person', 'people' # inflect.uncountable %w( fish sheep ) # end email_spec-2.2.0/examples/rails4_root/config/initializers/session_store.rb0000644000004100000410000000063413317714226027133 0ustar www-datawww-data# Be sure to restart your server when you modify this file. Rails.application.config.session_store :cookie_store, { :key => '_rails4_root_session', } # Use the database for sessions instead of the cookie-based default, # which shouldn't be used to store highly confidential information # (create the session table with "rake db:sessions:create") # Rails.application.config.session_store :active_record_store email_spec-2.2.0/examples/rails4_root/config/initializers/notifier_job.rb0000644000004100000410000000002713317714226026701 0ustar www-datawww-datarequire 'notifier_job' email_spec-2.2.0/examples/rails4_root/config/initializers/rspec_generator.rb0000644000004100000410000000020613317714226027411 0ustar www-datawww-dataRails4Root::Application.configure do config.generators do |g| g.integration_tool :rspec g.test_framework :rspec end end email_spec-2.2.0/examples/rails4_root/config/initializers/delayed_job.rb0000644000004100000410000000005113317714226026466 0ustar www-datawww-dataDelayed::Worker.backend = :active_record email_spec-2.2.0/examples/rails4_root/config/initializers/cookie_verification_secret.rb0000644000004100000410000000076013317714226031614 0ustar www-datawww-data# Be sure to restart your server when you modify this file. # Your secret key for verifying the integrity of signed cookies. # If you change this key, all old signed cookies will become invalid! # Make sure the secret is at least 30 characters and all random, # no regular words or you'll be exposed to dictionary attacks. Rails.application.config.secret_token = 'c63e0079364c42f86ef0256e8eb5e8f6993b30f1e64e4374f5211ddf4efdcd0895282c6383bd58dc817716b7ecc15f52de122e2df27fa326266d4163e4e5fe7d' email_spec-2.2.0/examples/rails4_root/config/initializers/mime_types.rb0000644000004100000410000000031513317714226026403 0ustar www-datawww-data# Be sure to restart your server when you modify this file. # Add new mime types for use in respond_to blocks: # Mime::Type.register "text/richtext", :rtf # Mime::Type.register_alias "text/html", :iphone email_spec-2.2.0/examples/rails4_root/config/application.rb0000644000004100000410000000363113317714226024031 0ustar www-datawww-datarequire File.expand_path('../boot', __FILE__) require 'rails/all' # If you have a Gemfile, require the gems listed there, including any gems # you've limited to :test, :development, or :production. Bundler.require(:default, Rails.env) if defined?(Bundler) module Rails4Root class Application < Rails::Application # Settings in config/environments/* take precedence over those specified here. # Application configuration should go into files in config/initializers # -- all .rb files in that directory are automatically loaded. # Add additional load paths for your own custom dirs # config.load_paths += %W( #{config.root}/extras ) # Only load the plugins named here, in the order given (default is alphabetical). # :all can be used as a placeholder for all plugins not explicitly named # config.plugins = [ :exception_notification, :ssl_requirement, :all ] # Activate observers that should always be running # config.active_record.observers = :cacher, :garbage_collector, :forum_observer # Set Time.zone default to the specified zone and make Active Record auto-convert to this zone. # Run "rake -D time" for a list of tasks for finding time zone names. Default is UTC. # config.time_zone = 'Central Time (US & Canada)' # The default locale is :en and all translations from config/locales/*.rb,yml are auto loaded. # config.i18n.load_path += Dir[Rails.root.join('my', 'locales', '*.{rb,yml}').to_s] # config.i18n.default_locale = :de # Configure generators values. Many other options are available, be sure to check the documentation. # config.generators do |g| # g.orm :active_record # g.template_engine :erb # g.test_framework :test_unit, :fixture => true # end # Configure sensitive parameters which will be filtered from the log file. config.filter_parameters << :password config.eager_load = false end end email_spec-2.2.0/examples/rails4_root/config/locales/0000755000004100000410000000000013317714226022620 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/config/locales/en.yml0000644000004100000410000000032513317714226023745 0ustar www-datawww-data# Sample localization file for English. Add more files in this directory for other locales. # See http://github.com/svenfuchs/rails-i18n/tree/master/rails%2Flocale for starting points. en: hello: "Hello world" email_spec-2.2.0/examples/rails4_root/config/cucumber.yml0000644000004100000410000000070513317714226023530 0ustar www-datawww-data<% rerun = File.file?('rerun.txt') ? IO.read('rerun.txt') : "" rerun_opts = rerun.to_s.strip.empty? ? "--format #{ENV['CUCUMBER_FORMAT'] || 'progress'} features" : "--format #{ENV['CUCUMBER_FORMAT'] || 'pretty'} #{rerun}" std_opts = "--format #{ENV['CUCUMBER_FORMAT'] || 'pretty'} --strict --tags ~@wip" %> default: <%= std_opts %> features wip: --tags @wip:3 --wip features rerun: <%= rerun_opts %> --format rerun --out rerun.txt --strict --tags ~@wip email_spec-2.2.0/examples/rails4_root/config/environment.rb0000644000004100000410000000033613317714226024071 0ustar www-datawww-data# Load the rails application require File.expand_path('../application', __FILE__) # Initialize the rails application Rails4Root::Application.initialize! ActionMailer::Base.default_url_options = { :host => 'example.com' } email_spec-2.2.0/examples/rails4_root/config/environments/0000755000004100000410000000000013317714226023725 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/config/environments/test.rb0000644000004100000410000000257313317714226025240 0ustar www-datawww-dataRails4Root::Application.configure do # Settings specified here will take precedence over those in config/environment.rb # The test environment is used exclusively to run your application's # test suite. You never need to work with it otherwise. Remember that # your test database is "scratch space" for the test suite and is wiped # and recreated between test runs. Don't rely on the data there! config.cache_classes = true # Log error messages when you accidentally call methods on nil. config.whiny_nils = true # Show full error reports and disable caching config.consider_all_requests_local = true config.action_controller.perform_caching = false # Disable request forgery protection in test environment config.action_controller.allow_forgery_protection = false # Tell Action Mailer not to deliver emails to the real world. # The :test delivery method accumulates sent emails in the # ActionMailer::Base.deliveries array. config.action_mailer.delivery_method = :test # Use SQL instead of Active Record's schema dumper when creating the test database. # This is necessary if your schema can't be completely dumped by the schema dumper, # like if you have constraints or database-specific column types # config.active_record.schema_format = :sql config.active_support.deprecation = :stderr config.active_support.test_order = :sorted end email_spec-2.2.0/examples/rails4_root/config/environments/production.rb0000644000004100000410000000271213317714226026442 0ustar www-datawww-dataRails4Root::Application.configure do # Settings specified here will take precedence over those in config/environment.rb # The production environment is meant for finished, "live" apps. # Code is not reloaded between requests config.cache_classes = true # Full error reports are disabled and caching is turned on config.consider_all_requests_local = false config.action_controller.perform_caching = true # Specifies the header that your server uses for sending files config.action_dispatch.x_sendfile_header = "X-Sendfile" # For nginx: # config.action_dispatch.x_sendfile_header = 'X-Accel-Redirect' # If you have no front-end server that supports something like X-Sendfile, # just comment this out and Rails will serve the files # See everything in the log (default is :info) # config.log_level = :debug # Use a different logger for distributed setups # config.logger = SyslogLogger.new # Use a different cache store in production # config.cache_store = :mem_cache_store # Disable Rails's static asset server # In production, Apache or nginx will already do this config.serve_static_assets = false # Enable serving of images, stylesheets, and javascripts from an asset server # config.action_controller.asset_host = "http://assets.example.com" # Disable delivery errors, bad email addresses will be ignored # config.action_mailer.raise_delivery_errors = false # Enable threaded mode # config.threadsafe! end email_spec-2.2.0/examples/rails4_root/config/environments/development.rb0000644000004100000410000000140713317714226026576 0ustar www-datawww-dataRails4Root::Application.configure do # Settings specified here will take precedence over those in config/environment.rb # In the development environment your application's code is reloaded on # every request. This slows down response time but is perfect for development # since you don't have to restart the webserver when you make code changes. config.cache_classes = false # Log error messages when you accidentally call methods on nil. config.whiny_nils = true # Show full error reports and disable caching config.consider_all_requests_local = true config.action_controller.perform_caching = false # Don't care if the mailer can't send config.action_mailer.raise_delivery_errors = false config.active_support.deprecation = :stderr end email_spec-2.2.0/examples/rails4_root/config/database.yml0000644000004100000410000000106613317714226023470 0ustar www-datawww-data# SQLite version 3.x # gem install sqlite3-ruby (not necessary on OS X Leopard) development: adapter: sqlite3 database: db/development.sqlite3 pool: 5 timeout: 5000 # Warning: The database defined as "test" will be erased and # re-generated from your development database when you run "rake". # Do not set this db to the same as development or production. test: &TEST adapter: sqlite3 database: db/test.sqlite3 pool: 5 timeout: 5000 production: adapter: sqlite3 database: db/production.sqlite3 pool: 5 timeout: 5000 cucumber: <<: *TESTemail_spec-2.2.0/examples/rails4_root/doc/0000755000004100000410000000000013317714226020476 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/doc/README_FOR_APP0000644000004100000410000000032313317714226022562 0ustar www-datawww-dataUse this README file to introduce your application and point to useful places in the API for learning more. Run "rake doc:app" to generate API documentation for your models, controllers, helpers, and libraries. email_spec-2.2.0/examples/rails4_root/Gemfile0000644000004100000410000000066013317714226021226 0ustar www-datawww-datasource 'http://rubygems.org' gem "rake", "~> 10.3.2" gem 'rails', '4.2.8' gem 'sqlite3' gem 'delayed_job', '~> 4.0.6' gem 'delayed_job_active_record' gem 'mimetype-fu', :require => 'mimetype_fu' gem "minitest-rails" group :test do gem "cucumber-rails", :require => false gem "database_cleaner" gem 'capybara' gem "rspec-rails" gem "minitest-matchers" gem "email_spec", :path => "../../", :require => "email_spec" end email_spec-2.2.0/examples/rails4_root/features/0000755000004100000410000000000013317714226021547 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/features/example.feature0000644000004100000410000000731513317714226024565 0ustar www-datawww-dataFeature: EmailSpec Example -- Prevent Bots from creating accounts In order to help alleviate email testing in apps As an email-spec contributor I want new users of the library to easily adopt email-spec in their app by following this example In order to prevent bots from setting up new accounts As a site manager I want new users to verify their email address with a confirmation link Background: Given no emails have been sent And I am a real person wanting to sign up for an account And I am on the homepage And I submit my registration information Scenario: First person signup (as myself) with three ways of opening email Then I should receive an email And I should have 1 email # Opening email #1 When I open the email Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body # Opening email #2 When I open the email with subject "Account confirmation" Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body # Opening email #3 When I open the email with subject /Account confirmation/ Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body When I follow "Click here to confirm your account!" in the email Then I should see "Confirm your new account" And "foo@bar.com" should have no emails Scenario: Third person signup (emails sent to others) with three ways of opening email Then "foo@bar.com" should have no emails And "example@example.com" should receive an email And "example@example.com" should have 1 email # Opening email #1 When they open the email Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body # Opening email #2 When "example@example.com" opens the email with subject "Account confirmation" Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body # Opening email #3 When "example@example.com" opens the email with subject /Account confirmation/ Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body When they follow "Click here to confirm your account!" in the email Then they should see "Confirm your new account" Scenario: Declarative First Person signup Then I should receive an email with a link to a confirmation page Scenario: Declarative First Person signup Then they should receive an email with a link to a confirmation page Scenario: Checking for text in different parts Then I should receive an email And I should have 1 email # Opening email #1 When I open the email Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body # Opening email #2 When I open the email with text "This is the HTML part" Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body # Opening email #3 When I open the email with text /This is the HTML part/ Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body email_spec-2.2.0/examples/rails4_root/features/attachments.feature0000644000004100000410000000217313317714226025442 0ustar www-datawww-dataFeature: Attachment testing support In order for developers to test attachments in emails I want to be able to provide working steps which inspect attachments Scenario: Email with Attachments Given no emails have been sent And I go to request attachments be sent to me Then I should receive an email When I open the email Then I should see 2 attachments with the email And there should be an attachment named "image.png" And there should be an attachment named "document.pdf" And attachment 1 should be named "image.png" And attachment 2 should be named "document.pdf" And there should be an attachment of type "image/png" And there should be an attachment of type "application/pdf" And attachment 1 should be of type "image/png" And attachment 2 should be of type "application/pdf" And all attachments should not be blank Scenario: Email without Attachments Given no emails have been sent And I am on the homepage And I submit my registration information Then I should receive an email When I open the email Then I should see no attachments with the email email_spec-2.2.0/examples/rails4_root/features/step_definitions/0000755000004100000410000000000013317714226025115 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/features/step_definitions/web_steps.rb0000644000004100000410000001367413317714226027450 0ustar www-datawww-data# IMPORTANT: This file is generated by cucumber-rails - edit at your own peril. # It is recommended to regenerate this file in the future when you upgrade to a # newer version of cucumber-rails. Consider adding your own code to a new file # instead of editing this one. Cucumber will automatically load all features/**/*.rb # files. require 'uri' require 'cgi' require File.expand_path(File.join(File.dirname(__FILE__), "..", "support", "paths")) module WithinHelpers def with_scope(locator) locator ? within(locator) { yield } : yield end end World(WithinHelpers) Given /^(?:|I )am on (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )go to (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )press "([^"]*)"(?: within "([^"]*)")?$/ do |button, selector| with_scope(selector) do click_button(button) end end When /^(?:|I )follow "([^"]*)"(?: within "([^"]*)")?$/ do |link, selector| with_scope(selector) do click_link(link) end end When /^(?:|I )fill in "([^"]*)" with "([^"]*)"(?: within "([^"]*)")?$/ do |field, value, selector| with_scope(selector) do fill_in(field, :with => value) end end When /^(?:|I )fill in "([^"]*)" for "([^"]*)"(?: within "([^"]*)")?$/ do |value, field, selector| with_scope(selector) do fill_in(field, :with => value) end end # Use this to fill in an entire form with data from a table. Example: # # When I fill in the following: # | Account Number | 5002 | # | Expiry date | 2009-11-01 | # | Note | Nice guy | # | Wants Email? | | # # TODO: Add support for checkbox, select og option # based on naming conventions. # When /^(?:|I )fill in the following(?: within "([^"]*)")?:$/ do |selector, fields| with_scope(selector) do fields.rows_hash.each do |name, value| When %{I fill in "#{name}" with "#{value}"} end end end When /^(?:|I )select "([^"]*)" from "([^"]*)"(?: within "([^"]*)")?$/ do |value, field, selector| with_scope(selector) do select(value, :from => field) end end When /^(?:|I )check "([^"]*)"(?: within "([^"]*)")?$/ do |field, selector| with_scope(selector) do check(field) end end When /^(?:|I )uncheck "([^"]*)"(?: within "([^"]*)")?$/ do |field, selector| with_scope(selector) do uncheck(field) end end When /^(?:|I )choose "([^"]*)"(?: within "([^"]*)")?$/ do |field, selector| with_scope(selector) do choose(field) end end When /^(?:|I )attach the file "([^"]*)" to "([^"]*)"(?: within "([^"]*)")?$/ do |path, field, selector| with_scope(selector) do attach_file(field, path) end end Then /^(?:|I )should see JSON:$/ do |expected_json| require 'json' expected = JSON.pretty_generate(JSON.parse(expected_json)) actual = JSON.pretty_generate(JSON.parse(response.body)) expect(expected).to eql actual end Then /^(?:|I )should see "([^"]*)"(?: within "([^"]*)")?$/ do |text, selector| with_scope(selector) do if self.respond_to? :expect expect(page).to have_content(text) else assert page.has_content?(text) end end end Then /^(?:|I )should see \/([^\/]*)\/(?: within "([^"]*)")?$/ do |regexp, selector| regexp = Regexp.new(regexp) with_scope(selector) do if self.respond_to? :expect expect(page).to have_xpath('//*', :text => regexp) else assert page.has_xpath?('//*', :text => regexp) end end end Then /^(?:|I )should not see "([^"]*)"(?: within "([^"]*)")?$/ do |text, selector| with_scope(selector) do if self.respond_to? :expect expect(page).to have_no_content(text) else assert page.has_no_content?(text) end end end Then /^(?:|I )should not see \/([^\/]*)\/(?: within "([^"]*)")?$/ do |regexp, selector| regexp = Regexp.new(regexp) with_scope(selector) do if self.respond_to? :expect expect(page).to have_no_xpath('//*', :text => regexp) else assert page.has_no_xpath?('//*', :text => regexp) end end end Then /^the "([^"]*)" field(?: within "([^"]*)")? should contain "([^"]*)"$/ do |field, selector, value| with_scope(selector) do field = find_field(field) field_value = (field.tag_name == 'textarea') ? field.text : field.value if self.respond_to? :expect expect(field_value).to match /#{value}/ else assert_match(/#{value}/, field_value) end end end Then /^the "([^"]*)" field(?: within "([^"]*)")? should not contain "([^"]*)"$/ do |field, selector, value| with_scope(selector) do field = find_field(field) field_value = (field.tag_name == 'textarea') ? field.text : field.value if self.respond_to? :expect expect(field_value).to_not match /#{value}/ else assert_no_match(/#{value}/, field_value) end end end Then /^the "([^"]*)" checkbox(?: within "([^"]*)")? should be checked$/ do |label, selector| with_scope(selector) do field_checked = find_field(label)['checked'] if self.respond_to? :expect expect(field_checked).to be true else assert field_checked end end end Then /^the "([^"]*)" checkbox(?: within "([^"]*)")? should not be checked$/ do |label, selector| with_scope(selector) do field_checked = find_field(label)['checked'] if self.respond_to? :expect expect(field_checked).to be false else assert !field_checked end end end Then /^(?:|I )should be on (.+)$/ do |page_name| current_path = URI.parse(current_url).path if self.respond_to? :expect expect(current_path).to eql path_to(page_name) else assert_equal path_to(page_name), current_path end end Then /^(?:|I )should have the following query string:$/ do |expected_pairs| query = URI.parse(current_url).query actual_params = query ? CGI.parse(query) : {} expected_params = {} expected_pairs.rows_hash.each_pair{|k,v| expected_params[k] = v.split(',')} if self.respond_to? :expect expect(actual_params).to eql expected_params else assert_equal expected_params, actual_params end end Then /^show me the page$/ do save_and_open_page end email_spec-2.2.0/examples/rails4_root/features/step_definitions/user_steps.rb0000755000004100000410000000163213317714226027643 0ustar www-datawww-dataGiven "I am a real person wanting to sign up for an account" do # no-op.. for documentation purposes only! end When /^I submit my registration information$/ do fill_in "Name", :with => 'Joe Someone' fill_in "Email", :with => 'example@example.com' click_button "Sign up" end Then /^(?:I|they) should receive an email with a link to a confirmation page$/ do expect(unread_emails_for(current_email_address).size).to eql 1 # this call will store the email and you can access it with current_email open_last_email_for(last_email_address) expect(current_email).to have_subject(/Account confirmation/) expect(current_email).to have_body_text('Joe Someone') click_email_link_matching /confirm/ expect(page).to have_content("Confirm your new account") end # Basically aliases "I should see [text]", but for third person Then /^they should see "([^\"]*)"$/ do |text| step "I should see \"#{text}\"" end email_spec-2.2.0/examples/rails4_root/features/step_definitions/email_steps.rb0000644000004100000410000001577213317714226027763 0ustar www-datawww-data# Commonly used email steps # # To add your own steps make a custom_email_steps.rb # The provided methods are: # # last_email_address # reset_mailer # open_last_email # visit_in_email # unread_emails_for # mailbox_for # current_email # open_email # read_emails_for # find_email # # General form for email scenarios are: # - clear the email queue (done automatically by email_spec) # - execute steps that sends an email # - check the user received an/no/[0-9] emails # - open the email # - inspect the email contents # - interact with the email (e.g. click links) # # The Cucumber steps below are setup in this order. module EmailHelpers def current_email_address # Replace with your a way to find your current email. e.g @current_user.email # last_email_address will return the last email address used by email spec to find an email. # Note that last_email_address will be reset after each Scenario. last_email_address || "example@example.com" end end World(EmailHelpers) # # Reset the e-mail queue within a scenario. # This is done automatically before each scenario. # Given /^(?:a clear email queue|no emails have been sent)$/ do reset_mailer end # # Check how many emails have been sent/received # Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails?$/ do |address, amount| expect(unread_emails_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should have (an|no|\d+) emails?$/ do |address, amount| expect(mailbox_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject "([^"]*?)"$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(Regexp.escape(subject)) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject \/([^"]*?)\/$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(subject) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive an email with the following body:$/ do |address, expected_body| open_email(address, :with_text => expected_body) end # # Accessing emails # # Opens the most recently received email When /^(?:I|they|"([^"]*?)") opens? the email$/ do |address| open_email(address) end When /^(?:I|they|"([^"]*?)") opens? the email with subject "([^"]*?)"$/ do |address, subject| open_email(address, :with_subject => subject) end When /^(?:I|they|"([^"]*?)") opens? the email with subject \/([^"]*?)\/$/ do |address, subject| open_email(address, :with_subject => Regexp.new(subject)) end When /^(?:I|they|"([^"]*?)") opens? the email with text "([^"]*?)"$/ do |address, text| open_email(address, :with_text => text) end When /^(?:I|they|"([^"]*?)") opens? the email with text \/([^"]*?)\/$/ do |address, text| open_email(address, :with_text => Regexp.new(text)) end # # Inspect the Email Contents # Then /^(?:I|they) should see "([^"]*?)" in the email subject$/ do |text| expect(current_email).to have_subject(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).to have_subject(Regexp.new(text)) end Then /^(?:I|they) should not see "([^"]*?)" in the email subject$/ do |text| expect(current_email).not_to have_subject(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).not_to have_subject(Regexp.new(text)) end Then /^(?:I|they) should see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).to include(text) end Then /^(?:I|they) should not see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to include(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).to match Regexp.new(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to match Regexp.new(text) end Then /^(?:I|they) should see the email delivered from "([^"]*?)"$/ do |text| expect(current_email).to be_delivered_from(text) end Then /^(?:I|they) should see the email reply to "([^"]*?)"$/ do |text| expect(current_email).to have_reply_to(text) end Then /^(?:I|they) should see "([^\"]*)" in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, text) end Then /^(?:I|they) should see \/([^\"]*)\/ in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, Regexp.new(text)) end Then /^I should see it is a multi\-part email$/ do expect(current_email).to be_multipart end Then /^(?:I|they) should see "([^"]*?)" in the email html part body$/ do |text| expect(current_email.html_part.body.to_s).to include(text) end Then /^(?:I|they) should see "([^"]*?)" in the email text part body$/ do |text| expect(current_email.text_part.body.to_s).to include(text) end # # Inspect the Email Attachments # Then /^(?:I|they) should see (an|no|\d+) attachments? with the email$/ do |amount| expect(current_email_attachments.size).to eql parse_email_count(amount) end Then /^there should be (an|no|\d+) attachments? named "([^"]*?)"$/ do |amount, filename| expect(current_email_attachments.select { |a| a.filename == filename }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be named "([^"]*?)"$/ do |index, filename| expect(current_email_attachments[(index.to_i - 1)].filename).to eql filename end Then /^there should be (an|no|\d+) attachments? of type "([^"]*?)"$/ do |amount, content_type| expect(current_email_attachments.select { |a| a.content_type.include?(content_type) }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be of type "([^"]*?)"$/ do |index, content_type| expect(current_email_attachments[(index.to_i - 1)].content_type).to include(content_type) end Then /^all attachments should not be blank$/ do current_email_attachments.each do |attachment| expect(attachment.read.size).to_not eql 0 end end Then /^show me a list of email attachments$/ do EmailSpec::EmailViewer::save_and_open_email_attachments_list(current_email) end # # Interact with Email Contents # When /^(?:I|they|"([^"]*?)") follows? "([^"]*?)" in the email$/ do |address, link| visit_in_email(link, address) end When /^(?:I|they) click the first link in the email$/ do click_first_link_in_email end # # Debugging # These only work with Rails and OSx ATM since EmailViewer uses RAILS_ROOT and OSx's 'open' command. # Patches accepted. ;) # Then /^save and open current email$/ do EmailSpec::EmailViewer::save_and_open_email(current_email) end Then /^save and open all text emails$/ do EmailSpec::EmailViewer::save_and_open_all_text_emails end Then /^save and open all html emails$/ do EmailSpec::EmailViewer::save_and_open_all_html_emails end Then /^save and open all raw emails$/ do EmailSpec::EmailViewer::save_and_open_all_raw_emails end email_spec-2.2.0/examples/rails4_root/features/delayed_job.feature0000644000004100000410000000077413317714226025375 0ustar www-datawww-dataFeature: Delayed Job support In order for developers using delayed_job to test emails I want to be able to provide a compatibility layer, which should run all delayed jobs before checking email boxes In order to populate deliveries done via send_later Scenario: Newsletter Given no emails have been sent And I go to request a newsletter Then I should receive an email And I should have 1 email When I open the email Then I should see "Newsletter sent" in the email subject email_spec-2.2.0/examples/rails4_root/features/errors.feature0000644000004100000410000000301013317714226024432 0ustar www-datawww-dataFeature: Email-spec errors example In order to help alleviate email testing in apps As a email-spec contributor I a newcomer Should be able to easily determine where I have gone wrong These scenarios should fail with helpful messages Background: Given I am on the homepage And no emails have been sent When I fill in "Email" with "example@example.com" And I press "Sign up" Scenario: I fail to open an email with incorrect subject Then I should receive an email When "example@example.com" opens the email with subject "no email" Scenario: I fail to open an email with incorrect subject Then I should receive an email When "example@example.com" opens the email with subject /no email/ Scenario: I fail to open an email with incorrect text Then I should receive an email When "example@example.com" opens the email with text "no email" Scenario: I fail to open an email with incorrect text Then I should receive an email When "example@example.com" opens the email with text /no email/ Scenario: I fail to receive an email with the expected link Then I should receive an email When I open the email When I follow "link that doesn't exist" in the email Scenario: I attempt to operate on an email that is not opened Then I should receive an email When I follow "confirm" in the email Scenario: I attempt to check out an unopened email Then I should see "confirm" in the email body And I should see "Account confirmation" in the email subject email_spec-2.2.0/examples/rails4_root/features/support/0000755000004100000410000000000013317714226023263 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/features/support/env_ext.rb0000644000004100000410000000013613317714226025260 0ustar www-datawww-data# email testing in cucumber require 'email_spec/cucumber' Delayed::Worker.delay_jobs = false email_spec-2.2.0/examples/rails4_root/features/support/paths.rb0000644000004100000410000000154113317714226024730 0ustar www-datawww-datamodule NavigationHelpers # Maps a name to a path. Used by the # # When /^I go to (.+)$/ do |page_name| # # step definition in web_steps.rb # def path_to(page_name) case page_name when /the home\s?page/ '/' # Add more mappings here. # Here is an example that pulls values out of the Regexp: # # when /^(.*)'s profile page$/i # user_profile_path(User.find_by_login($1)) when /request a newsletter/ request_newsletter_url('Name' => 'Joe Someone', 'Email' => 'example@example.com') when /request attachments be sent to me/ request_attachments_url('Name' => 'Joe Someone', 'Email' => 'example@example.com') else raise "Can't find mapping from \"#{page_name}\" to a path.\n" + "Now, go and add a mapping in #{__FILE__}" end end end World(NavigationHelpers) email_spec-2.2.0/examples/rails4_root/features/support/env.rb0000644000004100000410000000522413317714226024403 0ustar www-datawww-data# IMPORTANT: This file is generated by cucumber-rails - edit at your own peril. # It is recommended to regenerate this file in the future when you upgrade to a # newer version of cucumber-rails. Consider adding your own code to a new file # instead of editing this one. Cucumber will automatically load all features/**/*.rb # files. require 'cucumber/rails' # Capybara defaults to XPath selectors rather than Webrat's default of CSS3. In # order to ease the transition to Capybara we set the default here. If you'd # prefer to use XPath just remove this line and adjust any selectors in your # steps to use the XPath syntax. Capybara.default_selector = :css # By default, any exception happening in your Rails application will bubble up # to Cucumber so that your scenario will fail. This is a different from how # your application behaves in the production environment, where an error page will # be rendered instead. # # Sometimes we want to override this default behaviour and allow Rails to rescue # exceptions and display an error page (just like when the app is running in production). # Typical scenarios where you want to do this is when you test your error pages. # There are two ways to allow Rails to rescue exceptions: # # 1) Tag your scenario (or feature) with @allow-rescue # # 2) Set the value below to true. Beware that doing this globally is not # recommended as it will mask a lot of errors for you! # ActionController::Base.allow_rescue = false # Remove/comment out the lines below if your app doesn't have a database. # For some databases (like MongoDB and CouchDB) you may need to use :truncation instead. begin DatabaseCleaner.strategy = :transaction rescue NameError raise "You need to add database_cleaner to your Gemfile (in the :test group) if you wish to use it." end # You may also want to configure DatabaseCleaner to use different strategies for certain features and scenarios. # See the DatabaseCleaner documentation for details. Example: # # Before('@no-txn,@selenium,@culerity,@celerity,@javascript') do # # { :except => [:widgets] } may not do what you expect here # # as tCucumber::Rails::Database.javascript_strategy overrides # # this setting. # DatabaseCleaner.strategy = :truncation # end # # Before('~@no-txn', '~@selenium', '~@culerity', '~@celerity', '~@javascript') do # DatabaseCleaner.strategy = :transaction # end # # Possible values are :truncation and :transaction # The :transaction strategy is faster, but might give you threading problems. # See https://github.com/cucumber/cucumber-rails/blob/master/features/choose_javascript_database_strategy.feature Cucumber::Rails::Database.javascript_strategy = :truncation email_spec-2.2.0/examples/rails4_root/README0000644000004100000410000000025113317714226020607 0ustar www-datawww-data== Running features in this app 1. Make sure you have latest bundler gem. 2. cd into rails4_root and run `bundle install` 3. run script/rails generate email_spec:steps email_spec-2.2.0/examples/rails4_root/public/0000755000004100000410000000000013317714226021207 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/public/javascripts/0000755000004100000410000000000013317714226023540 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/public/javascripts/effects.js0000644000004100000410000011310713317714226025520 0ustar www-datawww-data// script.aculo.us effects.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009 // Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us) // Contributors: // Justin Palmer (http://encytemedia.com/) // Mark Pilgrim (http://diveintomark.org/) // Martin Bialasinki // // 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/ // converts rgb() and #xxx to #xxxxxx format, // returns self (or first argument) if not convertable String.prototype.parseColor = function() { var color = '#'; if (this.slice(0,4) == 'rgb(') { var cols = this.slice(4,this.length-1).split(','); var i=0; do { color += parseInt(cols[i]).toColorPart() } while (++i<3); } else { if (this.slice(0,1) == '#') { if (this.length==4) for(var i=1;i<4;i++) color += (this.charAt(i) + this.charAt(i)).toLowerCase(); if (this.length==7) color = this.toLowerCase(); } } return (color.length==7 ? color : (arguments[0] || this)); }; /*--------------------------------------------------------------------------*/ Element.collectTextNodes = function(element) { return $A($(element).childNodes).collect( function(node) { return (node.nodeType==3 ? node.nodeValue : (node.hasChildNodes() ? Element.collectTextNodes(node) : '')); }).flatten().join(''); }; Element.collectTextNodesIgnoreClass = function(element, className) { return $A($(element).childNodes).collect( function(node) { return (node.nodeType==3 ? node.nodeValue : ((node.hasChildNodes() && !Element.hasClassName(node,className)) ? Element.collectTextNodesIgnoreClass(node, className) : '')); }).flatten().join(''); }; Element.setContentZoom = function(element, percent) { element = $(element); element.setStyle({fontSize: (percent/100) + 'em'}); if (Prototype.Browser.WebKit) window.scrollBy(0,0); return element; }; Element.getInlineOpacity = function(element){ return $(element).style.opacity || ''; }; Element.forceRerendering = function(element) { try { element = $(element); var n = document.createTextNode(' '); element.appendChild(n); element.removeChild(n); } catch(e) { } }; /*--------------------------------------------------------------------------*/ var Effect = { _elementDoesNotExistError: { name: 'ElementDoesNotExistError', message: 'The specified DOM element does not exist, but is required for this effect to operate' }, Transitions: { linear: Prototype.K, sinoidal: function(pos) { return (-Math.cos(pos*Math.PI)/2) + .5; }, reverse: function(pos) { return 1-pos; }, flicker: function(pos) { var pos = ((-Math.cos(pos*Math.PI)/4) + .75) + Math.random()/4; return pos > 1 ? 1 : pos; }, wobble: function(pos) { return (-Math.cos(pos*Math.PI*(9*pos))/2) + .5; }, pulse: function(pos, pulses) { return (-Math.cos((pos*((pulses||5)-.5)*2)*Math.PI)/2) + .5; }, spring: function(pos) { return 1 - (Math.cos(pos * 4.5 * Math.PI) * Math.exp(-pos * 6)); }, none: function(pos) { return 0; }, full: function(pos) { return 1; } }, DefaultOptions: { duration: 1.0, // seconds fps: 100, // 100= assume 66fps max. sync: false, // true for combining from: 0.0, to: 1.0, delay: 0.0, queue: 'parallel' }, tagifyText: function(element) { var tagifyStyle = 'position:relative'; if (Prototype.Browser.IE) tagifyStyle += ';zoom:1'; element = $(element); $A(element.childNodes).each( function(child) { if (child.nodeType==3) { child.nodeValue.toArray().each( function(character) { element.insertBefore( new Element('span', {style: tagifyStyle}).update( character == ' ' ? String.fromCharCode(160) : character), child); }); Element.remove(child); } }); }, multiple: function(element, effect) { var elements; if (((typeof element == 'object') || Object.isFunction(element)) && (element.length)) elements = element; else elements = $(element).childNodes; var options = Object.extend({ speed: 0.1, delay: 0.0 }, arguments[2] || { }); var masterDelay = options.delay; $A(elements).each( function(element, index) { new effect(element, Object.extend(options, { delay: index * options.speed + masterDelay })); }); }, PAIRS: { 'slide': ['SlideDown','SlideUp'], 'blind': ['BlindDown','BlindUp'], 'appear': ['Appear','Fade'] }, toggle: function(element, effect, options) { element = $(element); effect = (effect || 'appear').toLowerCase(); return Effect[ Effect.PAIRS[ effect ][ element.visible() ? 1 : 0 ] ](element, Object.extend({ queue: { position:'end', scope:(element.id || 'global'), limit: 1 } }, options || {})); } }; Effect.DefaultOptions.transition = Effect.Transitions.sinoidal; /* ------------- core effects ------------- */ Effect.ScopedQueue = Class.create(Enumerable, { initialize: function() { this.effects = []; this.interval = null; }, _each: function(iterator) { this.effects._each(iterator); }, add: function(effect) { var timestamp = new Date().getTime(); var position = Object.isString(effect.options.queue) ? effect.options.queue : effect.options.queue.position; switch(position) { case 'front': // move unstarted effects after this effect this.effects.findAll(function(e){ return e.state=='idle' }).each( function(e) { e.startOn += effect.finishOn; e.finishOn += effect.finishOn; }); break; case 'with-last': timestamp = this.effects.pluck('startOn').max() || timestamp; break; case 'end': // start effect after last queued effect has finished timestamp = this.effects.pluck('finishOn').max() || timestamp; break; } effect.startOn += timestamp; effect.finishOn += timestamp; if (!effect.options.queue.limit || (this.effects.length < effect.options.queue.limit)) this.effects.push(effect); if (!this.interval) this.interval = setInterval(this.loop.bind(this), 15); }, remove: function(effect) { this.effects = this.effects.reject(function(e) { return e==effect }); if (this.effects.length == 0) { clearInterval(this.interval); this.interval = null; } }, loop: function() { var timePos = new Date().getTime(); for(var i=0, len=this.effects.length;i= this.startOn) { if (timePos >= this.finishOn) { this.render(1.0); this.cancel(); this.event('beforeFinish'); if (this.finish) this.finish(); this.event('afterFinish'); return; } var pos = (timePos - this.startOn) / this.totalTime, frame = (pos * this.totalFrames).round(); if (frame > this.currentFrame) { this.render(pos); this.currentFrame = frame; } } }, cancel: function() { if (!this.options.sync) Effect.Queues.get(Object.isString(this.options.queue) ? 'global' : this.options.queue.scope).remove(this); this.state = 'finished'; }, event: function(eventName) { if (this.options[eventName + 'Internal']) this.options[eventName + 'Internal'](this); if (this.options[eventName]) this.options[eventName](this); }, inspect: function() { var data = $H(); for(property in this) if (!Object.isFunction(this[property])) data.set(property, this[property]); return '#'; } }); Effect.Parallel = Class.create(Effect.Base, { initialize: function(effects) { this.effects = effects || []; this.start(arguments[1]); }, update: function(position) { this.effects.invoke('render', position); }, finish: function(position) { this.effects.each( function(effect) { effect.render(1.0); effect.cancel(); effect.event('beforeFinish'); if (effect.finish) effect.finish(position); effect.event('afterFinish'); }); } }); Effect.Tween = Class.create(Effect.Base, { initialize: function(object, from, to) { object = Object.isString(object) ? $(object) : object; var args = $A(arguments), method = args.last(), options = args.length == 5 ? args[3] : null; this.method = Object.isFunction(method) ? method.bind(object) : Object.isFunction(object[method]) ? object[method].bind(object) : function(value) { object[method] = value }; this.start(Object.extend({ from: from, to: to }, options || { })); }, update: function(position) { this.method(position); } }); Effect.Event = Class.create(Effect.Base, { initialize: function() { this.start(Object.extend({ duration: 0 }, arguments[0] || { })); }, update: Prototype.emptyFunction }); Effect.Opacity = Class.create(Effect.Base, { initialize: function(element) { this.element = $(element); if (!this.element) throw(Effect._elementDoesNotExistError); // make this work on IE on elements without 'layout' if (Prototype.Browser.IE && (!this.element.currentStyle.hasLayout)) this.element.setStyle({zoom: 1}); var options = Object.extend({ from: this.element.getOpacity() || 0.0, to: 1.0 }, arguments[1] || { }); this.start(options); }, update: function(position) { this.element.setOpacity(position); } }); Effect.Move = Class.create(Effect.Base, { initialize: function(element) { this.element = $(element); if (!this.element) throw(Effect._elementDoesNotExistError); var options = Object.extend({ x: 0, y: 0, mode: 'relative' }, arguments[1] || { }); this.start(options); }, setup: function() { this.element.makePositioned(); this.originalLeft = parseFloat(this.element.getStyle('left') || '0'); this.originalTop = parseFloat(this.element.getStyle('top') || '0'); if (this.options.mode == 'absolute') { this.options.x = this.options.x - this.originalLeft; this.options.y = this.options.y - this.originalTop; } }, update: function(position) { this.element.setStyle({ left: (this.options.x * position + this.originalLeft).round() + 'px', top: (this.options.y * position + this.originalTop).round() + 'px' }); } }); // for backwards compatibility Effect.MoveBy = function(element, toTop, toLeft) { return new Effect.Move(element, Object.extend({ x: toLeft, y: toTop }, arguments[3] || { })); }; Effect.Scale = Class.create(Effect.Base, { initialize: function(element, percent) { this.element = $(element); if (!this.element) throw(Effect._elementDoesNotExistError); var options = Object.extend({ scaleX: true, scaleY: true, scaleContent: true, scaleFromCenter: false, scaleMode: 'box', // 'box' or 'contents' or { } with provided values scaleFrom: 100.0, scaleTo: percent }, arguments[2] || { }); this.start(options); }, setup: function() { this.restoreAfterFinish = this.options.restoreAfterFinish || false; this.elementPositioning = this.element.getStyle('position'); this.originalStyle = { }; ['top','left','width','height','fontSize'].each( function(k) { this.originalStyle[k] = this.element.style[k]; }.bind(this)); this.originalTop = this.element.offsetTop; this.originalLeft = this.element.offsetLeft; var fontSize = this.element.getStyle('font-size') || '100%'; ['em','px','%','pt'].each( function(fontSizeType) { if (fontSize.indexOf(fontSizeType)>0) { this.fontSize = parseFloat(fontSize); this.fontSizeType = fontSizeType; } }.bind(this)); this.factor = (this.options.scaleTo - this.options.scaleFrom)/100; this.dims = null; if (this.options.scaleMode=='box') this.dims = [this.element.offsetHeight, this.element.offsetWidth]; if (/^content/.test(this.options.scaleMode)) this.dims = [this.element.scrollHeight, this.element.scrollWidth]; if (!this.dims) this.dims = [this.options.scaleMode.originalHeight, this.options.scaleMode.originalWidth]; }, update: function(position) { var currentScale = (this.options.scaleFrom/100.0) + (this.factor * position); if (this.options.scaleContent && this.fontSize) this.element.setStyle({fontSize: this.fontSize * currentScale + this.fontSizeType }); this.setDimensions(this.dims[0] * currentScale, this.dims[1] * currentScale); }, finish: function(position) { if (this.restoreAfterFinish) this.element.setStyle(this.originalStyle); }, setDimensions: function(height, width) { var d = { }; if (this.options.scaleX) d.width = width.round() + 'px'; if (this.options.scaleY) d.height = height.round() + 'px'; if (this.options.scaleFromCenter) { var topd = (height - this.dims[0])/2; var leftd = (width - this.dims[1])/2; if (this.elementPositioning == 'absolute') { if (this.options.scaleY) d.top = this.originalTop-topd + 'px'; if (this.options.scaleX) d.left = this.originalLeft-leftd + 'px'; } else { if (this.options.scaleY) d.top = -topd + 'px'; if (this.options.scaleX) d.left = -leftd + 'px'; } } this.element.setStyle(d); } }); Effect.Highlight = Class.create(Effect.Base, { initialize: function(element) { this.element = $(element); if (!this.element) throw(Effect._elementDoesNotExistError); var options = Object.extend({ startcolor: '#ffff99' }, arguments[1] || { }); this.start(options); }, setup: function() { // Prevent executing on elements not in the layout flow if (this.element.getStyle('display')=='none') { this.cancel(); return; } // Disable background image during the effect this.oldStyle = { }; if (!this.options.keepBackgroundImage) { this.oldStyle.backgroundImage = this.element.getStyle('background-image'); this.element.setStyle({backgroundImage: 'none'}); } if (!this.options.endcolor) this.options.endcolor = this.element.getStyle('background-color').parseColor('#ffffff'); if (!this.options.restorecolor) this.options.restorecolor = this.element.getStyle('background-color'); // init color calculations this._base = $R(0,2).map(function(i){ return parseInt(this.options.startcolor.slice(i*2+1,i*2+3),16) }.bind(this)); this._delta = $R(0,2).map(function(i){ return parseInt(this.options.endcolor.slice(i*2+1,i*2+3),16)-this._base[i] }.bind(this)); }, update: function(position) { this.element.setStyle({backgroundColor: $R(0,2).inject('#',function(m,v,i){ return m+((this._base[i]+(this._delta[i]*position)).round().toColorPart()); }.bind(this)) }); }, finish: function() { this.element.setStyle(Object.extend(this.oldStyle, { backgroundColor: this.options.restorecolor })); } }); Effect.ScrollTo = function(element) { var options = arguments[1] || { }, scrollOffsets = document.viewport.getScrollOffsets(), elementOffsets = $(element).cumulativeOffset(); if (options.offset) elementOffsets[1] += options.offset; return new Effect.Tween(null, scrollOffsets.top, elementOffsets[1], options, function(p){ scrollTo(scrollOffsets.left, p.round()); } ); }; /* ------------- combination effects ------------- */ Effect.Fade = function(element) { element = $(element); var oldOpacity = element.getInlineOpacity(); var options = Object.extend({ from: element.getOpacity() || 1.0, to: 0.0, afterFinishInternal: function(effect) { if (effect.options.to!=0) return; effect.element.hide().setStyle({opacity: oldOpacity}); } }, arguments[1] || { }); return new Effect.Opacity(element,options); }; Effect.Appear = function(element) { element = $(element); var options = Object.extend({ from: (element.getStyle('display') == 'none' ? 0.0 : element.getOpacity() || 0.0), to: 1.0, // force Safari to render floated elements properly afterFinishInternal: function(effect) { effect.element.forceRerendering(); }, beforeSetup: function(effect) { effect.element.setOpacity(effect.options.from).show(); }}, arguments[1] || { }); return new Effect.Opacity(element,options); }; Effect.Puff = function(element) { element = $(element); var oldStyle = { opacity: element.getInlineOpacity(), position: element.getStyle('position'), top: element.style.top, left: element.style.left, width: element.style.width, height: element.style.height }; return new Effect.Parallel( [ new Effect.Scale(element, 200, { sync: true, scaleFromCenter: true, scaleContent: true, restoreAfterFinish: true }), new Effect.Opacity(element, { sync: true, to: 0.0 } ) ], Object.extend({ duration: 1.0, beforeSetupInternal: function(effect) { Position.absolutize(effect.effects[0].element); }, afterFinishInternal: function(effect) { effect.effects[0].element.hide().setStyle(oldStyle); } }, arguments[1] || { }) ); }; Effect.BlindUp = function(element) { element = $(element); element.makeClipping(); return new Effect.Scale(element, 0, Object.extend({ scaleContent: false, scaleX: false, restoreAfterFinish: true, afterFinishInternal: function(effect) { effect.element.hide().undoClipping(); } }, arguments[1] || { }) ); }; Effect.BlindDown = function(element) { element = $(element); var elementDimensions = element.getDimensions(); return new Effect.Scale(element, 100, Object.extend({ scaleContent: false, scaleX: false, scaleFrom: 0, scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width}, restoreAfterFinish: true, afterSetup: function(effect) { effect.element.makeClipping().setStyle({height: '0px'}).show(); }, afterFinishInternal: function(effect) { effect.element.undoClipping(); } }, arguments[1] || { })); }; Effect.SwitchOff = function(element) { element = $(element); var oldOpacity = element.getInlineOpacity(); return new Effect.Appear(element, Object.extend({ duration: 0.4, from: 0, transition: Effect.Transitions.flicker, afterFinishInternal: function(effect) { new Effect.Scale(effect.element, 1, { duration: 0.3, scaleFromCenter: true, scaleX: false, scaleContent: false, restoreAfterFinish: true, beforeSetup: function(effect) { effect.element.makePositioned().makeClipping(); }, afterFinishInternal: function(effect) { effect.element.hide().undoClipping().undoPositioned().setStyle({opacity: oldOpacity}); } }); } }, arguments[1] || { })); }; Effect.DropOut = function(element) { element = $(element); var oldStyle = { top: element.getStyle('top'), left: element.getStyle('left'), opacity: element.getInlineOpacity() }; return new Effect.Parallel( [ new Effect.Move(element, {x: 0, y: 100, sync: true }), new Effect.Opacity(element, { sync: true, to: 0.0 }) ], Object.extend( { duration: 0.5, beforeSetup: function(effect) { effect.effects[0].element.makePositioned(); }, afterFinishInternal: function(effect) { effect.effects[0].element.hide().undoPositioned().setStyle(oldStyle); } }, arguments[1] || { })); }; Effect.Shake = function(element) { element = $(element); var options = Object.extend({ distance: 20, duration: 0.5 }, arguments[1] || {}); var distance = parseFloat(options.distance); var split = parseFloat(options.duration) / 10.0; var oldStyle = { top: element.getStyle('top'), left: element.getStyle('left') }; return new Effect.Move(element, { x: distance, y: 0, duration: split, afterFinishInternal: function(effect) { new Effect.Move(effect.element, { x: -distance*2, y: 0, duration: split*2, afterFinishInternal: function(effect) { new Effect.Move(effect.element, { x: distance*2, y: 0, duration: split*2, afterFinishInternal: function(effect) { new Effect.Move(effect.element, { x: -distance*2, y: 0, duration: split*2, afterFinishInternal: function(effect) { new Effect.Move(effect.element, { x: distance*2, y: 0, duration: split*2, afterFinishInternal: function(effect) { new Effect.Move(effect.element, { x: -distance, y: 0, duration: split, afterFinishInternal: function(effect) { effect.element.undoPositioned().setStyle(oldStyle); }}); }}); }}); }}); }}); }}); }; Effect.SlideDown = function(element) { element = $(element).cleanWhitespace(); // SlideDown need to have the content of the element wrapped in a container element with fixed height! var oldInnerBottom = element.down().getStyle('bottom'); var elementDimensions = element.getDimensions(); return new Effect.Scale(element, 100, Object.extend({ scaleContent: false, scaleX: false, scaleFrom: window.opera ? 0 : 1, scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width}, restoreAfterFinish: true, afterSetup: function(effect) { effect.element.makePositioned(); effect.element.down().makePositioned(); if (window.opera) effect.element.setStyle({top: ''}); effect.element.makeClipping().setStyle({height: '0px'}).show(); }, afterUpdateInternal: function(effect) { effect.element.down().setStyle({bottom: (effect.dims[0] - effect.element.clientHeight) + 'px' }); }, afterFinishInternal: function(effect) { effect.element.undoClipping().undoPositioned(); effect.element.down().undoPositioned().setStyle({bottom: oldInnerBottom}); } }, arguments[1] || { }) ); }; Effect.SlideUp = function(element) { element = $(element).cleanWhitespace(); var oldInnerBottom = element.down().getStyle('bottom'); var elementDimensions = element.getDimensions(); return new Effect.Scale(element, window.opera ? 0 : 1, Object.extend({ scaleContent: false, scaleX: false, scaleMode: 'box', scaleFrom: 100, scaleMode: {originalHeight: elementDimensions.height, originalWidth: elementDimensions.width}, restoreAfterFinish: true, afterSetup: function(effect) { effect.element.makePositioned(); effect.element.down().makePositioned(); if (window.opera) effect.element.setStyle({top: ''}); effect.element.makeClipping().show(); }, afterUpdateInternal: function(effect) { effect.element.down().setStyle({bottom: (effect.dims[0] - effect.element.clientHeight) + 'px' }); }, afterFinishInternal: function(effect) { effect.element.hide().undoClipping().undoPositioned(); effect.element.down().undoPositioned().setStyle({bottom: oldInnerBottom}); } }, arguments[1] || { }) ); }; // Bug in opera makes the TD containing this element expand for a instance after finish Effect.Squish = function(element) { return new Effect.Scale(element, window.opera ? 1 : 0, { restoreAfterFinish: true, beforeSetup: function(effect) { effect.element.makeClipping(); }, afterFinishInternal: function(effect) { effect.element.hide().undoClipping(); } }); }; Effect.Grow = function(element) { element = $(element); var options = Object.extend({ direction: 'center', moveTransition: Effect.Transitions.sinoidal, scaleTransition: Effect.Transitions.sinoidal, opacityTransition: Effect.Transitions.full }, arguments[1] || { }); var oldStyle = { top: element.style.top, left: element.style.left, height: element.style.height, width: element.style.width, opacity: element.getInlineOpacity() }; var dims = element.getDimensions(); var initialMoveX, initialMoveY; var moveX, moveY; switch (options.direction) { case 'top-left': initialMoveX = initialMoveY = moveX = moveY = 0; break; case 'top-right': initialMoveX = dims.width; initialMoveY = moveY = 0; moveX = -dims.width; break; case 'bottom-left': initialMoveX = moveX = 0; initialMoveY = dims.height; moveY = -dims.height; break; case 'bottom-right': initialMoveX = dims.width; initialMoveY = dims.height; moveX = -dims.width; moveY = -dims.height; break; case 'center': initialMoveX = dims.width / 2; initialMoveY = dims.height / 2; moveX = -dims.width / 2; moveY = -dims.height / 2; break; } return new Effect.Move(element, { x: initialMoveX, y: initialMoveY, duration: 0.01, beforeSetup: function(effect) { effect.element.hide().makeClipping().makePositioned(); }, afterFinishInternal: function(effect) { new Effect.Parallel( [ new Effect.Opacity(effect.element, { sync: true, to: 1.0, from: 0.0, transition: options.opacityTransition }), new Effect.Move(effect.element, { x: moveX, y: moveY, sync: true, transition: options.moveTransition }), new Effect.Scale(effect.element, 100, { scaleMode: { originalHeight: dims.height, originalWidth: dims.width }, sync: true, scaleFrom: window.opera ? 1 : 0, transition: options.scaleTransition, restoreAfterFinish: true}) ], Object.extend({ beforeSetup: function(effect) { effect.effects[0].element.setStyle({height: '0px'}).show(); }, afterFinishInternal: function(effect) { effect.effects[0].element.undoClipping().undoPositioned().setStyle(oldStyle); } }, options) ); } }); }; Effect.Shrink = function(element) { element = $(element); var options = Object.extend({ direction: 'center', moveTransition: Effect.Transitions.sinoidal, scaleTransition: Effect.Transitions.sinoidal, opacityTransition: Effect.Transitions.none }, arguments[1] || { }); var oldStyle = { top: element.style.top, left: element.style.left, height: element.style.height, width: element.style.width, opacity: element.getInlineOpacity() }; var dims = element.getDimensions(); var moveX, moveY; switch (options.direction) { case 'top-left': moveX = moveY = 0; break; case 'top-right': moveX = dims.width; moveY = 0; break; case 'bottom-left': moveX = 0; moveY = dims.height; break; case 'bottom-right': moveX = dims.width; moveY = dims.height; break; case 'center': moveX = dims.width / 2; moveY = dims.height / 2; break; } return new Effect.Parallel( [ new Effect.Opacity(element, { sync: true, to: 0.0, from: 1.0, transition: options.opacityTransition }), new Effect.Scale(element, window.opera ? 1 : 0, { sync: true, transition: options.scaleTransition, restoreAfterFinish: true}), new Effect.Move(element, { x: moveX, y: moveY, sync: true, transition: options.moveTransition }) ], Object.extend({ beforeStartInternal: function(effect) { effect.effects[0].element.makePositioned().makeClipping(); }, afterFinishInternal: function(effect) { effect.effects[0].element.hide().undoClipping().undoPositioned().setStyle(oldStyle); } }, options) ); }; Effect.Pulsate = function(element) { element = $(element); var options = arguments[1] || { }, oldOpacity = element.getInlineOpacity(), transition = options.transition || Effect.Transitions.linear, reverser = function(pos){ return 1 - transition((-Math.cos((pos*(options.pulses||5)*2)*Math.PI)/2) + .5); }; return new Effect.Opacity(element, Object.extend(Object.extend({ duration: 2.0, from: 0, afterFinishInternal: function(effect) { effect.element.setStyle({opacity: oldOpacity}); } }, options), {transition: reverser})); }; Effect.Fold = function(element) { element = $(element); var oldStyle = { top: element.style.top, left: element.style.left, width: element.style.width, height: element.style.height }; element.makeClipping(); return new Effect.Scale(element, 5, Object.extend({ scaleContent: false, scaleX: false, afterFinishInternal: function(effect) { new Effect.Scale(element, 1, { scaleContent: false, scaleY: false, afterFinishInternal: function(effect) { effect.element.hide().undoClipping().setStyle(oldStyle); } }); }}, arguments[1] || { })); }; Effect.Morph = Class.create(Effect.Base, { initialize: function(element) { this.element = $(element); if (!this.element) throw(Effect._elementDoesNotExistError); var options = Object.extend({ style: { } }, arguments[1] || { }); if (!Object.isString(options.style)) this.style = $H(options.style); else { if (options.style.include(':')) this.style = options.style.parseStyle(); else { this.element.addClassName(options.style); this.style = $H(this.element.getStyles()); this.element.removeClassName(options.style); var css = this.element.getStyles(); this.style = this.style.reject(function(style) { return style.value == css[style.key]; }); options.afterFinishInternal = function(effect) { effect.element.addClassName(effect.options.style); effect.transforms.each(function(transform) { effect.element.style[transform.style] = ''; }); }; } } this.start(options); }, setup: function(){ function parseColor(color){ if (!color || ['rgba(0, 0, 0, 0)','transparent'].include(color)) color = '#ffffff'; color = color.parseColor(); return $R(0,2).map(function(i){ return parseInt( color.slice(i*2+1,i*2+3), 16 ); }); } this.transforms = this.style.map(function(pair){ var property = pair[0], value = pair[1], unit = null; if (value.parseColor('#zzzzzz') != '#zzzzzz') { value = value.parseColor(); unit = 'color'; } else if (property == 'opacity') { value = parseFloat(value); if (Prototype.Browser.IE && (!this.element.currentStyle.hasLayout)) this.element.setStyle({zoom: 1}); } else if (Element.CSS_LENGTH.test(value)) { var components = value.match(/^([\+\-]?[0-9\.]+)(.*)$/); value = parseFloat(components[1]); unit = (components.length == 3) ? components[2] : null; } var originalValue = this.element.getStyle(property); return { style: property.camelize(), originalValue: unit=='color' ? parseColor(originalValue) : parseFloat(originalValue || 0), targetValue: unit=='color' ? parseColor(value) : value, unit: unit }; }.bind(this)).reject(function(transform){ return ( (transform.originalValue == transform.targetValue) || ( transform.unit != 'color' && (isNaN(transform.originalValue) || isNaN(transform.targetValue)) ) ); }); }, update: function(position) { var style = { }, transform, i = this.transforms.length; while(i--) style[(transform = this.transforms[i]).style] = transform.unit=='color' ? '#'+ (Math.round(transform.originalValue[0]+ (transform.targetValue[0]-transform.originalValue[0])*position)).toColorPart() + (Math.round(transform.originalValue[1]+ (transform.targetValue[1]-transform.originalValue[1])*position)).toColorPart() + (Math.round(transform.originalValue[2]+ (transform.targetValue[2]-transform.originalValue[2])*position)).toColorPart() : (transform.originalValue + (transform.targetValue - transform.originalValue) * position).toFixed(3) + (transform.unit === null ? '' : transform.unit); this.element.setStyle(style, true); } }); Effect.Transform = Class.create({ initialize: function(tracks){ this.tracks = []; this.options = arguments[1] || { }; this.addTracks(tracks); }, addTracks: function(tracks){ tracks.each(function(track){ track = $H(track); var data = track.values().first(); this.tracks.push($H({ ids: track.keys().first(), effect: Effect.Morph, options: { style: data } })); }.bind(this)); return this; }, play: function(){ return new Effect.Parallel( this.tracks.map(function(track){ var ids = track.get('ids'), effect = track.get('effect'), options = track.get('options'); var elements = [$(ids) || $$(ids)].flatten(); return elements.map(function(e){ return new effect(e, Object.extend({ sync:true }, options)) }); }).flatten(), this.options ); } }); Element.CSS_PROPERTIES = $w( 'backgroundColor backgroundPosition borderBottomColor borderBottomStyle ' + 'borderBottomWidth borderLeftColor borderLeftStyle borderLeftWidth ' + 'borderRightColor borderRightStyle borderRightWidth borderSpacing ' + 'borderTopColor borderTopStyle borderTopWidth bottom clip color ' + 'fontSize fontWeight height left letterSpacing lineHeight ' + 'marginBottom marginLeft marginRight marginTop markerOffset maxHeight '+ 'maxWidth minHeight minWidth opacity outlineColor outlineOffset ' + 'outlineWidth paddingBottom paddingLeft paddingRight paddingTop ' + 'right textIndent top width wordSpacing zIndex'); Element.CSS_LENGTH = /^(([\+\-]?[0-9\.]+)(em|ex|px|in|cm|mm|pt|pc|\%))|0$/; String.__parseStyleElement = document.createElement('div'); String.prototype.parseStyle = function(){ var style, styleRules = $H(); if (Prototype.Browser.WebKit) style = new Element('div',{style:this}).style; else { String.__parseStyleElement.innerHTML = '
'; style = String.__parseStyleElement.childNodes[0].style; } Element.CSS_PROPERTIES.each(function(property){ if (style[property]) styleRules.set(property, style[property]); }); if (Prototype.Browser.IE && this.include('opacity')) styleRules.set('opacity', this.match(/opacity:\s*((?:0|1)?(?:\.\d*)?)/)[1]); return styleRules; }; if (document.defaultView && document.defaultView.getComputedStyle) { Element.getStyles = function(element) { var css = document.defaultView.getComputedStyle($(element), null); return Element.CSS_PROPERTIES.inject({ }, function(styles, property) { styles[property] = css[property]; return styles; }); }; } else { Element.getStyles = function(element) { element = $(element); var css = element.currentStyle, styles; styles = Element.CSS_PROPERTIES.inject({ }, function(results, property) { results[property] = css[property]; return results; }); if (!styles.opacity) styles.opacity = element.getOpacity(); return styles; }; } Effect.Methods = { morph: function(element, style) { element = $(element); new Effect.Morph(element, Object.extend({ style: style }, arguments[2] || { })); return element; }, visualEffect: function(element, effect, options) { element = $(element); var s = effect.dasherize().camelize(), klass = s.charAt(0).toUpperCase() + s.substring(1); new Effect[klass](element, options); return element; }, highlight: function(element, options) { element = $(element); new Effect.Highlight(element, options); return element; } }; $w('fade appear grow shrink fold blindUp blindDown slideUp slideDown '+ 'pulsate shake puff squish switchOff dropOut').each( function(effect) { Effect.Methods[effect] = function(element, options){ element = $(element); Effect[effect.charAt(0).toUpperCase() + effect.substring(1)](element, options); return element; }; } ); $w('getInlineOpacity forceRerendering setContentZoom collectTextNodes collectTextNodesIgnoreClass getStyles').each( function(f) { Effect.Methods[f] = Element[f]; } ); Element.addMethods(Effect.Methods);email_spec-2.2.0/examples/rails4_root/public/javascripts/prototype.js0000644000004100000410000042111613317714226026150 0ustar www-datawww-data/* Prototype JavaScript framework, version 1.6.1 * (c) 2005-2009 Sam Stephenson * * Prototype is freely distributable under the terms of an MIT-style license. * For details, see the Prototype web site: http://www.prototypejs.org/ * *--------------------------------------------------------------------------*/ var Prototype = { Version: '1.6.1', Browser: (function(){ var ua = navigator.userAgent; var isOpera = Object.prototype.toString.call(window.opera) == '[object Opera]'; return { IE: !!window.attachEvent && !isOpera, Opera: isOpera, WebKit: ua.indexOf('AppleWebKit/') > -1, Gecko: ua.indexOf('Gecko') > -1 && ua.indexOf('KHTML') === -1, MobileSafari: /Apple.*Mobile.*Safari/.test(ua) } })(), BrowserFeatures: { XPath: !!document.evaluate, SelectorsAPI: !!document.querySelector, ElementExtensions: (function() { var constructor = window.Element || window.HTMLElement; return !!(constructor && constructor.prototype); })(), SpecificElementExtensions: (function() { if (typeof window.HTMLDivElement !== 'undefined') return true; var div = document.createElement('div'); var form = document.createElement('form'); var isSupported = false; if (div['__proto__'] && (div['__proto__'] !== form['__proto__'])) { isSupported = true; } div = form = null; return isSupported; })() }, ScriptFragment: ']*>([\\S\\s]*?)<\/script>', JSONFilter: /^\/\*-secure-([\s\S]*)\*\/\s*$/, emptyFunction: function() { }, K: function(x) { return x } }; if (Prototype.Browser.MobileSafari) Prototype.BrowserFeatures.SpecificElementExtensions = false; var Abstract = { }; var Try = { these: function() { var returnValue; for (var i = 0, length = arguments.length; i < length; i++) { var lambda = arguments[i]; try { returnValue = lambda(); break; } catch (e) { } } return returnValue; } }; /* Based on Alex Arnell's inheritance implementation. */ var Class = (function() { function subclass() {}; function create() { var parent = null, properties = $A(arguments); if (Object.isFunction(properties[0])) parent = properties.shift(); function klass() { this.initialize.apply(this, arguments); } Object.extend(klass, Class.Methods); klass.superclass = parent; klass.subclasses = []; if (parent) { subclass.prototype = parent.prototype; klass.prototype = new subclass; parent.subclasses.push(klass); } for (var i = 0; i < properties.length; i++) klass.addMethods(properties[i]); if (!klass.prototype.initialize) klass.prototype.initialize = Prototype.emptyFunction; klass.prototype.constructor = klass; return klass; } function addMethods(source) { var ancestor = this.superclass && this.superclass.prototype; var properties = Object.keys(source); if (!Object.keys({ toString: true }).length) { if (source.toString != Object.prototype.toString) properties.push("toString"); if (source.valueOf != Object.prototype.valueOf) properties.push("valueOf"); } for (var i = 0, length = properties.length; i < length; i++) { var property = properties[i], value = source[property]; if (ancestor && Object.isFunction(value) && value.argumentNames().first() == "$super") { var method = value; value = (function(m) { return function() { return ancestor[m].apply(this, arguments); }; })(property).wrap(method); value.valueOf = method.valueOf.bind(method); value.toString = method.toString.bind(method); } this.prototype[property] = value; } return this; } return { create: create, Methods: { addMethods: addMethods } }; })(); (function() { var _toString = Object.prototype.toString; function extend(destination, source) { for (var property in source) destination[property] = source[property]; return destination; } function inspect(object) { try { if (isUndefined(object)) return 'undefined'; if (object === null) return 'null'; return object.inspect ? object.inspect() : String(object); } catch (e) { if (e instanceof RangeError) return '...'; throw e; } } function toJSON(object) { var type = typeof object; switch (type) { case 'undefined': case 'function': case 'unknown': return; case 'boolean': return object.toString(); } if (object === null) return 'null'; if (object.toJSON) return object.toJSON(); if (isElement(object)) return; var results = []; for (var property in object) { var value = toJSON(object[property]); if (!isUndefined(value)) results.push(property.toJSON() + ': ' + value); } return '{' + results.join(', ') + '}'; } function toQueryString(object) { return $H(object).toQueryString(); } function toHTML(object) { return object && object.toHTML ? object.toHTML() : String.interpret(object); } function keys(object) { var results = []; for (var property in object) results.push(property); return results; } function values(object) { var results = []; for (var property in object) results.push(object[property]); return results; } function clone(object) { return extend({ }, object); } function isElement(object) { return !!(object && object.nodeType == 1); } function isArray(object) { return _toString.call(object) == "[object Array]"; } function isHash(object) { return object instanceof Hash; } function isFunction(object) { return typeof object === "function"; } function isString(object) { return _toString.call(object) == "[object String]"; } function isNumber(object) { return _toString.call(object) == "[object Number]"; } function isUndefined(object) { return typeof object === "undefined"; } extend(Object, { extend: extend, inspect: inspect, toJSON: toJSON, toQueryString: toQueryString, toHTML: toHTML, keys: keys, values: values, clone: clone, isElement: isElement, isArray: isArray, isHash: isHash, isFunction: isFunction, isString: isString, isNumber: isNumber, isUndefined: isUndefined }); })(); Object.extend(Function.prototype, (function() { var slice = Array.prototype.slice; function update(array, args) { var arrayLength = array.length, length = args.length; while (length--) array[arrayLength + length] = args[length]; return array; } function merge(array, args) { array = slice.call(array, 0); return update(array, args); } function argumentNames() { var names = this.toString().match(/^[\s\(]*function[^(]*\(([^)]*)\)/)[1] .replace(/\/\/.*?[\r\n]|\/\*(?:.|[\r\n])*?\*\//g, '') .replace(/\s+/g, '').split(','); return names.length == 1 && !names[0] ? [] : names; } function bind(context) { if (arguments.length < 2 && Object.isUndefined(arguments[0])) return this; var __method = this, args = slice.call(arguments, 1); return function() { var a = merge(args, arguments); return __method.apply(context, a); } } function bindAsEventListener(context) { var __method = this, args = slice.call(arguments, 1); return function(event) { var a = update([event || window.event], args); return __method.apply(context, a); } } function curry() { if (!arguments.length) return this; var __method = this, args = slice.call(arguments, 0); return function() { var a = merge(args, arguments); return __method.apply(this, a); } } function delay(timeout) { var __method = this, args = slice.call(arguments, 1); timeout = timeout * 1000 return window.setTimeout(function() { return __method.apply(__method, args); }, timeout); } function defer() { var args = update([0.01], arguments); return this.delay.apply(this, args); } function wrap(wrapper) { var __method = this; return function() { var a = update([__method.bind(this)], arguments); return wrapper.apply(this, a); } } function methodize() { if (this._methodized) return this._methodized; var __method = this; return this._methodized = function() { var a = update([this], arguments); return __method.apply(null, a); }; } return { argumentNames: argumentNames, bind: bind, bindAsEventListener: bindAsEventListener, curry: curry, delay: delay, defer: defer, wrap: wrap, methodize: methodize } })()); Date.prototype.toJSON = function() { return '"' + this.getUTCFullYear() + '-' + (this.getUTCMonth() + 1).toPaddedString(2) + '-' + this.getUTCDate().toPaddedString(2) + 'T' + this.getUTCHours().toPaddedString(2) + ':' + this.getUTCMinutes().toPaddedString(2) + ':' + this.getUTCSeconds().toPaddedString(2) + 'Z"'; }; RegExp.prototype.match = RegExp.prototype.test; RegExp.escape = function(str) { return String(str).replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1'); }; var PeriodicalExecuter = Class.create({ initialize: function(callback, frequency) { this.callback = callback; this.frequency = frequency; this.currentlyExecuting = false; this.registerCallback(); }, registerCallback: function() { this.timer = setInterval(this.onTimerEvent.bind(this), this.frequency * 1000); }, execute: function() { this.callback(this); }, stop: function() { if (!this.timer) return; clearInterval(this.timer); this.timer = null; }, onTimerEvent: function() { if (!this.currentlyExecuting) { try { this.currentlyExecuting = true; this.execute(); this.currentlyExecuting = false; } catch(e) { this.currentlyExecuting = false; throw e; } } } }); Object.extend(String, { interpret: function(value) { return value == null ? '' : String(value); }, specialChar: { '\b': '\\b', '\t': '\\t', '\n': '\\n', '\f': '\\f', '\r': '\\r', '\\': '\\\\' } }); Object.extend(String.prototype, (function() { function prepareReplacement(replacement) { if (Object.isFunction(replacement)) return replacement; var template = new Template(replacement); return function(match) { return template.evaluate(match) }; } function gsub(pattern, replacement) { var result = '', source = this, match; replacement = prepareReplacement(replacement); if (Object.isString(pattern)) pattern = RegExp.escape(pattern); if (!(pattern.length || pattern.source)) { replacement = replacement(''); return replacement + source.split('').join(replacement) + replacement; } while (source.length > 0) { if (match = source.match(pattern)) { result += source.slice(0, match.index); result += String.interpret(replacement(match)); source = source.slice(match.index + match[0].length); } else { result += source, source = ''; } } return result; } function sub(pattern, replacement, count) { replacement = prepareReplacement(replacement); count = Object.isUndefined(count) ? 1 : count; return this.gsub(pattern, function(match) { if (--count < 0) return match[0]; return replacement(match); }); } function scan(pattern, iterator) { this.gsub(pattern, iterator); return String(this); } function truncate(length, truncation) { length = length || 30; truncation = Object.isUndefined(truncation) ? '...' : truncation; return this.length > length ? this.slice(0, length - truncation.length) + truncation : String(this); } function strip() { return this.replace(/^\s+/, '').replace(/\s+$/, ''); } function stripTags() { return this.replace(/<\w+(\s+("[^"]*"|'[^']*'|[^>])+)?>|<\/\w+>/gi, ''); } function stripScripts() { return this.replace(new RegExp(Prototype.ScriptFragment, 'img'), ''); } function extractScripts() { var matchAll = new RegExp(Prototype.ScriptFragment, 'img'); var matchOne = new RegExp(Prototype.ScriptFragment, 'im'); return (this.match(matchAll) || []).map(function(scriptTag) { return (scriptTag.match(matchOne) || ['', ''])[1]; }); } function evalScripts() { return this.extractScripts().map(function(script) { return eval(script) }); } function escapeHTML() { return this.replace(/&/g,'&').replace(//g,'>'); } function unescapeHTML() { return this.stripTags().replace(/</g,'<').replace(/>/g,'>').replace(/&/g,'&'); } function toQueryParams(separator) { var match = this.strip().match(/([^?#]*)(#.*)?$/); if (!match) return { }; return match[1].split(separator || '&').inject({ }, function(hash, pair) { if ((pair = pair.split('='))[0]) { var key = decodeURIComponent(pair.shift()); var value = pair.length > 1 ? pair.join('=') : pair[0]; if (value != undefined) value = decodeURIComponent(value); if (key in hash) { if (!Object.isArray(hash[key])) hash[key] = [hash[key]]; hash[key].push(value); } else hash[key] = value; } return hash; }); } function toArray() { return this.split(''); } function succ() { return this.slice(0, this.length - 1) + String.fromCharCode(this.charCodeAt(this.length - 1) + 1); } function times(count) { return count < 1 ? '' : new Array(count + 1).join(this); } function camelize() { var parts = this.split('-'), len = parts.length; if (len == 1) return parts[0]; var camelized = this.charAt(0) == '-' ? parts[0].charAt(0).toUpperCase() + parts[0].substring(1) : parts[0]; for (var i = 1; i < len; i++) camelized += parts[i].charAt(0).toUpperCase() + parts[i].substring(1); return camelized; } function capitalize() { return this.charAt(0).toUpperCase() + this.substring(1).toLowerCase(); } function underscore() { return this.replace(/::/g, '/') .replace(/([A-Z]+)([A-Z][a-z])/g, '$1_$2') .replace(/([a-z\d])([A-Z])/g, '$1_$2') .replace(/-/g, '_') .toLowerCase(); } function dasherize() { return this.replace(/_/g, '-'); } function inspect(useDoubleQuotes) { var escapedString = this.replace(/[\x00-\x1f\\]/g, function(character) { if (character in String.specialChar) { return String.specialChar[character]; } return '\\u00' + character.charCodeAt().toPaddedString(2, 16); }); if (useDoubleQuotes) return '"' + escapedString.replace(/"/g, '\\"') + '"'; return "'" + escapedString.replace(/'/g, '\\\'') + "'"; } function toJSON() { return this.inspect(true); } function unfilterJSON(filter) { return this.replace(filter || Prototype.JSONFilter, '$1'); } function isJSON() { var str = this; if (str.blank()) return false; str = this.replace(/\\./g, '@').replace(/"[^"\\\n\r]*"/g, ''); return (/^[,:{}\[\]0-9.\-+Eaeflnr-u \n\r\t]*$/).test(str); } function evalJSON(sanitize) { var json = this.unfilterJSON(); try { if (!sanitize || json.isJSON()) return eval('(' + json + ')'); } catch (e) { } throw new SyntaxError('Badly formed JSON string: ' + this.inspect()); } function include(pattern) { return this.indexOf(pattern) > -1; } function startsWith(pattern) { return this.indexOf(pattern) === 0; } function endsWith(pattern) { var d = this.length - pattern.length; return d >= 0 && this.lastIndexOf(pattern) === d; } function empty() { return this == ''; } function blank() { return /^\s*$/.test(this); } function interpolate(object, pattern) { return new Template(this, pattern).evaluate(object); } return { gsub: gsub, sub: sub, scan: scan, truncate: truncate, strip: String.prototype.trim ? String.prototype.trim : strip, stripTags: stripTags, stripScripts: stripScripts, extractScripts: extractScripts, evalScripts: evalScripts, escapeHTML: escapeHTML, unescapeHTML: unescapeHTML, toQueryParams: toQueryParams, parseQuery: toQueryParams, toArray: toArray, succ: succ, times: times, camelize: camelize, capitalize: capitalize, underscore: underscore, dasherize: dasherize, inspect: inspect, toJSON: toJSON, unfilterJSON: unfilterJSON, isJSON: isJSON, evalJSON: evalJSON, include: include, startsWith: startsWith, endsWith: endsWith, empty: empty, blank: blank, interpolate: interpolate }; })()); var Template = Class.create({ initialize: function(template, pattern) { this.template = template.toString(); this.pattern = pattern || Template.Pattern; }, evaluate: function(object) { if (object && Object.isFunction(object.toTemplateReplacements)) object = object.toTemplateReplacements(); return this.template.gsub(this.pattern, function(match) { if (object == null) return (match[1] + ''); var before = match[1] || ''; if (before == '\\') return match[2]; var ctx = object, expr = match[3]; var pattern = /^([^.[]+|\[((?:.*?[^\\])?)\])(\.|\[|$)/; match = pattern.exec(expr); if (match == null) return before; while (match != null) { var comp = match[1].startsWith('[') ? match[2].replace(/\\\\]/g, ']') : match[1]; ctx = ctx[comp]; if (null == ctx || '' == match[3]) break; expr = expr.substring('[' == match[3] ? match[1].length : match[0].length); match = pattern.exec(expr); } return before + String.interpret(ctx); }); } }); Template.Pattern = /(^|.|\r|\n)(#\{(.*?)\})/; var $break = { }; var Enumerable = (function() { function each(iterator, context) { var index = 0; try { this._each(function(value) { iterator.call(context, value, index++); }); } catch (e) { if (e != $break) throw e; } return this; } function eachSlice(number, iterator, context) { var index = -number, slices = [], array = this.toArray(); if (number < 1) return array; while ((index += number) < array.length) slices.push(array.slice(index, index+number)); return slices.collect(iterator, context); } function all(iterator, context) { iterator = iterator || Prototype.K; var result = true; this.each(function(value, index) { result = result && !!iterator.call(context, value, index); if (!result) throw $break; }); return result; } function any(iterator, context) { iterator = iterator || Prototype.K; var result = false; this.each(function(value, index) { if (result = !!iterator.call(context, value, index)) throw $break; }); return result; } function collect(iterator, context) { iterator = iterator || Prototype.K; var results = []; this.each(function(value, index) { results.push(iterator.call(context, value, index)); }); return results; } function detect(iterator, context) { var result; this.each(function(value, index) { if (iterator.call(context, value, index)) { result = value; throw $break; } }); return result; } function findAll(iterator, context) { var results = []; this.each(function(value, index) { if (iterator.call(context, value, index)) results.push(value); }); return results; } function grep(filter, iterator, context) { iterator = iterator || Prototype.K; var results = []; if (Object.isString(filter)) filter = new RegExp(RegExp.escape(filter)); this.each(function(value, index) { if (filter.match(value)) results.push(iterator.call(context, value, index)); }); return results; } function include(object) { if (Object.isFunction(this.indexOf)) if (this.indexOf(object) != -1) return true; var found = false; this.each(function(value) { if (value == object) { found = true; throw $break; } }); return found; } function inGroupsOf(number, fillWith) { fillWith = Object.isUndefined(fillWith) ? null : fillWith; return this.eachSlice(number, function(slice) { while(slice.length < number) slice.push(fillWith); return slice; }); } function inject(memo, iterator, context) { this.each(function(value, index) { memo = iterator.call(context, memo, value, index); }); return memo; } function invoke(method) { var args = $A(arguments).slice(1); return this.map(function(value) { return value[method].apply(value, args); }); } function max(iterator, context) { iterator = iterator || Prototype.K; var result; this.each(function(value, index) { value = iterator.call(context, value, index); if (result == null || value >= result) result = value; }); return result; } function min(iterator, context) { iterator = iterator || Prototype.K; var result; this.each(function(value, index) { value = iterator.call(context, value, index); if (result == null || value < result) result = value; }); return result; } function partition(iterator, context) { iterator = iterator || Prototype.K; var trues = [], falses = []; this.each(function(value, index) { (iterator.call(context, value, index) ? trues : falses).push(value); }); return [trues, falses]; } function pluck(property) { var results = []; this.each(function(value) { results.push(value[property]); }); return results; } function reject(iterator, context) { var results = []; this.each(function(value, index) { if (!iterator.call(context, value, index)) results.push(value); }); return results; } function sortBy(iterator, context) { return this.map(function(value, index) { return { value: value, criteria: iterator.call(context, value, index) }; }).sort(function(left, right) { var a = left.criteria, b = right.criteria; return a < b ? -1 : a > b ? 1 : 0; }).pluck('value'); } function toArray() { return this.map(); } function zip() { var iterator = Prototype.K, args = $A(arguments); if (Object.isFunction(args.last())) iterator = args.pop(); var collections = [this].concat(args).map($A); return this.map(function(value, index) { return iterator(collections.pluck(index)); }); } function size() { return this.toArray().length; } function inspect() { return '#'; } return { each: each, eachSlice: eachSlice, all: all, every: all, any: any, some: any, collect: collect, map: collect, detect: detect, findAll: findAll, select: findAll, filter: findAll, grep: grep, include: include, member: include, inGroupsOf: inGroupsOf, inject: inject, invoke: invoke, max: max, min: min, partition: partition, pluck: pluck, reject: reject, sortBy: sortBy, toArray: toArray, entries: toArray, zip: zip, size: size, inspect: inspect, find: detect }; })(); function $A(iterable) { if (!iterable) return []; if ('toArray' in Object(iterable)) return iterable.toArray(); var length = iterable.length || 0, results = new Array(length); while (length--) results[length] = iterable[length]; return results; } function $w(string) { if (!Object.isString(string)) return []; string = string.strip(); return string ? string.split(/\s+/) : []; } Array.from = $A; (function() { var arrayProto = Array.prototype, slice = arrayProto.slice, _each = arrayProto.forEach; // use native browser JS 1.6 implementation if available function each(iterator) { for (var i = 0, length = this.length; i < length; i++) iterator(this[i]); } if (!_each) _each = each; function clear() { this.length = 0; return this; } function first() { return this[0]; } function last() { return this[this.length - 1]; } function compact() { return this.select(function(value) { return value != null; }); } function flatten() { return this.inject([], function(array, value) { if (Object.isArray(value)) return array.concat(value.flatten()); array.push(value); return array; }); } function without() { var values = slice.call(arguments, 0); return this.select(function(value) { return !values.include(value); }); } function reverse(inline) { return (inline !== false ? this : this.toArray())._reverse(); } function uniq(sorted) { return this.inject([], function(array, value, index) { if (0 == index || (sorted ? array.last() != value : !array.include(value))) array.push(value); return array; }); } function intersect(array) { return this.uniq().findAll(function(item) { return array.detect(function(value) { return item === value }); }); } function clone() { return slice.call(this, 0); } function size() { return this.length; } function inspect() { return '[' + this.map(Object.inspect).join(', ') + ']'; } function toJSON() { var results = []; this.each(function(object) { var value = Object.toJSON(object); if (!Object.isUndefined(value)) results.push(value); }); return '[' + results.join(', ') + ']'; } function indexOf(item, i) { i || (i = 0); var length = this.length; if (i < 0) i = length + i; for (; i < length; i++) if (this[i] === item) return i; return -1; } function lastIndexOf(item, i) { i = isNaN(i) ? this.length : (i < 0 ? this.length + i : i) + 1; var n = this.slice(0, i).reverse().indexOf(item); return (n < 0) ? n : i - n - 1; } function concat() { var array = slice.call(this, 0), item; for (var i = 0, length = arguments.length; i < length; i++) { item = arguments[i]; if (Object.isArray(item) && !('callee' in item)) { for (var j = 0, arrayLength = item.length; j < arrayLength; j++) array.push(item[j]); } else { array.push(item); } } return array; } Object.extend(arrayProto, Enumerable); if (!arrayProto._reverse) arrayProto._reverse = arrayProto.reverse; Object.extend(arrayProto, { _each: _each, clear: clear, first: first, last: last, compact: compact, flatten: flatten, without: without, reverse: reverse, uniq: uniq, intersect: intersect, clone: clone, toArray: clone, size: size, inspect: inspect, toJSON: toJSON }); var CONCAT_ARGUMENTS_BUGGY = (function() { return [].concat(arguments)[0][0] !== 1; })(1,2) if (CONCAT_ARGUMENTS_BUGGY) arrayProto.concat = concat; if (!arrayProto.indexOf) arrayProto.indexOf = indexOf; if (!arrayProto.lastIndexOf) arrayProto.lastIndexOf = lastIndexOf; })(); function $H(object) { return new Hash(object); }; var Hash = Class.create(Enumerable, (function() { function initialize(object) { this._object = Object.isHash(object) ? object.toObject() : Object.clone(object); } function _each(iterator) { for (var key in this._object) { var value = this._object[key], pair = [key, value]; pair.key = key; pair.value = value; iterator(pair); } } function set(key, value) { return this._object[key] = value; } function get(key) { if (this._object[key] !== Object.prototype[key]) return this._object[key]; } function unset(key) { var value = this._object[key]; delete this._object[key]; return value; } function toObject() { return Object.clone(this._object); } function keys() { return this.pluck('key'); } function values() { return this.pluck('value'); } function index(value) { var match = this.detect(function(pair) { return pair.value === value; }); return match && match.key; } function merge(object) { return this.clone().update(object); } function update(object) { return new Hash(object).inject(this, function(result, pair) { result.set(pair.key, pair.value); return result; }); } function toQueryPair(key, value) { if (Object.isUndefined(value)) return key; return key + '=' + encodeURIComponent(String.interpret(value)); } function toQueryString() { return this.inject([], function(results, pair) { var key = encodeURIComponent(pair.key), values = pair.value; if (values && typeof values == 'object') { if (Object.isArray(values)) return results.concat(values.map(toQueryPair.curry(key))); } else results.push(toQueryPair(key, values)); return results; }).join('&'); } function inspect() { return '#'; } function toJSON() { return Object.toJSON(this.toObject()); } function clone() { return new Hash(this); } return { initialize: initialize, _each: _each, set: set, get: get, unset: unset, toObject: toObject, toTemplateReplacements: toObject, keys: keys, values: values, index: index, merge: merge, update: update, toQueryString: toQueryString, inspect: inspect, toJSON: toJSON, clone: clone }; })()); Hash.from = $H; Object.extend(Number.prototype, (function() { function toColorPart() { return this.toPaddedString(2, 16); } function succ() { return this + 1; } function times(iterator, context) { $R(0, this, true).each(iterator, context); return this; } function toPaddedString(length, radix) { var string = this.toString(radix || 10); return '0'.times(length - string.length) + string; } function toJSON() { return isFinite(this) ? this.toString() : 'null'; } function abs() { return Math.abs(this); } function round() { return Math.round(this); } function ceil() { return Math.ceil(this); } function floor() { return Math.floor(this); } return { toColorPart: toColorPart, succ: succ, times: times, toPaddedString: toPaddedString, toJSON: toJSON, abs: abs, round: round, ceil: ceil, floor: floor }; })()); function $R(start, end, exclusive) { return new ObjectRange(start, end, exclusive); } var ObjectRange = Class.create(Enumerable, (function() { function initialize(start, end, exclusive) { this.start = start; this.end = end; this.exclusive = exclusive; } function _each(iterator) { var value = this.start; while (this.include(value)) { iterator(value); value = value.succ(); } } function include(value) { if (value < this.start) return false; if (this.exclusive) return value < this.end; return value <= this.end; } return { initialize: initialize, _each: _each, include: include }; })()); var Ajax = { getTransport: function() { return Try.these( function() {return new XMLHttpRequest()}, function() {return new ActiveXObject('Msxml2.XMLHTTP')}, function() {return new ActiveXObject('Microsoft.XMLHTTP')} ) || false; }, activeRequestCount: 0 }; Ajax.Responders = { responders: [], _each: function(iterator) { this.responders._each(iterator); }, register: function(responder) { if (!this.include(responder)) this.responders.push(responder); }, unregister: function(responder) { this.responders = this.responders.without(responder); }, dispatch: function(callback, request, transport, json) { this.each(function(responder) { if (Object.isFunction(responder[callback])) { try { responder[callback].apply(responder, [request, transport, json]); } catch (e) { } } }); } }; Object.extend(Ajax.Responders, Enumerable); Ajax.Responders.register({ onCreate: function() { Ajax.activeRequestCount++ }, onComplete: function() { Ajax.activeRequestCount-- } }); Ajax.Base = Class.create({ initialize: function(options) { this.options = { method: 'post', asynchronous: true, contentType: 'application/x-www-form-urlencoded', encoding: 'UTF-8', parameters: '', evalJSON: true, evalJS: true }; Object.extend(this.options, options || { }); this.options.method = this.options.method.toLowerCase(); if (Object.isString(this.options.parameters)) this.options.parameters = this.options.parameters.toQueryParams(); else if (Object.isHash(this.options.parameters)) this.options.parameters = this.options.parameters.toObject(); } }); Ajax.Request = Class.create(Ajax.Base, { _complete: false, initialize: function($super, url, options) { $super(options); this.transport = Ajax.getTransport(); this.request(url); }, request: function(url) { this.url = url; this.method = this.options.method; var params = Object.clone(this.options.parameters); if (!['get', 'post'].include(this.method)) { params['_method'] = this.method; this.method = 'post'; } this.parameters = params; if (params = Object.toQueryString(params)) { if (this.method == 'get') this.url += (this.url.include('?') ? '&' : '?') + params; else if (/Konqueror|Safari|KHTML/.test(navigator.userAgent)) params += '&_='; } try { var response = new Ajax.Response(this); if (this.options.onCreate) this.options.onCreate(response); Ajax.Responders.dispatch('onCreate', this, response); this.transport.open(this.method.toUpperCase(), this.url, this.options.asynchronous); if (this.options.asynchronous) this.respondToReadyState.bind(this).defer(1); this.transport.onreadystatechange = this.onStateChange.bind(this); this.setRequestHeaders(); this.body = this.method == 'post' ? (this.options.postBody || params) : null; this.transport.send(this.body); /* Force Firefox to handle ready state 4 for synchronous requests */ if (!this.options.asynchronous && this.transport.overrideMimeType) this.onStateChange(); } catch (e) { this.dispatchException(e); } }, onStateChange: function() { var readyState = this.transport.readyState; if (readyState > 1 && !((readyState == 4) && this._complete)) this.respondToReadyState(this.transport.readyState); }, setRequestHeaders: function() { var headers = { 'X-Requested-With': 'XMLHttpRequest', 'X-Prototype-Version': Prototype.Version, 'Accept': 'text/javascript, text/html, application/xml, text/xml, */*' }; if (this.method == 'post') { headers['Content-type'] = this.options.contentType + (this.options.encoding ? '; charset=' + this.options.encoding : ''); /* Force "Connection: close" for older Mozilla browsers to work * around a bug where XMLHttpRequest sends an incorrect * Content-length header. See Mozilla Bugzilla #246651. */ if (this.transport.overrideMimeType && (navigator.userAgent.match(/Gecko\/(\d{4})/) || [0,2005])[1] < 2005) headers['Connection'] = 'close'; } if (typeof this.options.requestHeaders == 'object') { var extras = this.options.requestHeaders; if (Object.isFunction(extras.push)) for (var i = 0, length = extras.length; i < length; i += 2) headers[extras[i]] = extras[i+1]; else $H(extras).each(function(pair) { headers[pair.key] = pair.value }); } for (var name in headers) this.transport.setRequestHeader(name, headers[name]); }, success: function() { var status = this.getStatus(); return !status || (status >= 200 && status < 300); }, getStatus: function() { try { return this.transport.status || 0; } catch (e) { return 0 } }, respondToReadyState: function(readyState) { var state = Ajax.Request.Events[readyState], response = new Ajax.Response(this); if (state == 'Complete') { try { this._complete = true; (this.options['on' + response.status] || this.options['on' + (this.success() ? 'Success' : 'Failure')] || Prototype.emptyFunction)(response, response.headerJSON); } catch (e) { this.dispatchException(e); } var contentType = response.getHeader('Content-type'); if (this.options.evalJS == 'force' || (this.options.evalJS && this.isSameOrigin() && contentType && contentType.match(/^\s*(text|application)\/(x-)?(java|ecma)script(;.*)?\s*$/i))) this.evalResponse(); } try { (this.options['on' + state] || Prototype.emptyFunction)(response, response.headerJSON); Ajax.Responders.dispatch('on' + state, this, response, response.headerJSON); } catch (e) { this.dispatchException(e); } if (state == 'Complete') { this.transport.onreadystatechange = Prototype.emptyFunction; } }, isSameOrigin: function() { var m = this.url.match(/^\s*https?:\/\/[^\/]*/); return !m || (m[0] == '#{protocol}//#{domain}#{port}'.interpolate({ protocol: location.protocol, domain: document.domain, port: location.port ? ':' + location.port : '' })); }, getHeader: function(name) { try { return this.transport.getResponseHeader(name) || null; } catch (e) { return null; } }, evalResponse: function() { try { return eval((this.transport.responseText || '').unfilterJSON()); } catch (e) { this.dispatchException(e); } }, dispatchException: function(exception) { (this.options.onException || Prototype.emptyFunction)(this, exception); Ajax.Responders.dispatch('onException', this, exception); } }); Ajax.Request.Events = ['Uninitialized', 'Loading', 'Loaded', 'Interactive', 'Complete']; Ajax.Response = Class.create({ initialize: function(request){ this.request = request; var transport = this.transport = request.transport, readyState = this.readyState = transport.readyState; if((readyState > 2 && !Prototype.Browser.IE) || readyState == 4) { this.status = this.getStatus(); this.statusText = this.getStatusText(); this.responseText = String.interpret(transport.responseText); this.headerJSON = this._getHeaderJSON(); } if(readyState == 4) { var xml = transport.responseXML; this.responseXML = Object.isUndefined(xml) ? null : xml; this.responseJSON = this._getResponseJSON(); } }, status: 0, statusText: '', getStatus: Ajax.Request.prototype.getStatus, getStatusText: function() { try { return this.transport.statusText || ''; } catch (e) { return '' } }, getHeader: Ajax.Request.prototype.getHeader, getAllHeaders: function() { try { return this.getAllResponseHeaders(); } catch (e) { return null } }, getResponseHeader: function(name) { return this.transport.getResponseHeader(name); }, getAllResponseHeaders: function() { return this.transport.getAllResponseHeaders(); }, _getHeaderJSON: function() { var json = this.getHeader('X-JSON'); if (!json) return null; json = decodeURIComponent(escape(json)); try { return json.evalJSON(this.request.options.sanitizeJSON || !this.request.isSameOrigin()); } catch (e) { this.request.dispatchException(e); } }, _getResponseJSON: function() { var options = this.request.options; if (!options.evalJSON || (options.evalJSON != 'force' && !(this.getHeader('Content-type') || '').include('application/json')) || this.responseText.blank()) return null; try { return this.responseText.evalJSON(options.sanitizeJSON || !this.request.isSameOrigin()); } catch (e) { this.request.dispatchException(e); } } }); Ajax.Updater = Class.create(Ajax.Request, { initialize: function($super, container, url, options) { this.container = { success: (container.success || container), failure: (container.failure || (container.success ? null : container)) }; options = Object.clone(options); var onComplete = options.onComplete; options.onComplete = (function(response, json) { this.updateContent(response.responseText); if (Object.isFunction(onComplete)) onComplete(response, json); }).bind(this); $super(url, options); }, updateContent: function(responseText) { var receiver = this.container[this.success() ? 'success' : 'failure'], options = this.options; if (!options.evalScripts) responseText = responseText.stripScripts(); if (receiver = $(receiver)) { if (options.insertion) { if (Object.isString(options.insertion)) { var insertion = { }; insertion[options.insertion] = responseText; receiver.insert(insertion); } else options.insertion(receiver, responseText); } else receiver.update(responseText); } } }); Ajax.PeriodicalUpdater = Class.create(Ajax.Base, { initialize: function($super, container, url, options) { $super(options); this.onComplete = this.options.onComplete; this.frequency = (this.options.frequency || 2); this.decay = (this.options.decay || 1); this.updater = { }; this.container = container; this.url = url; this.start(); }, start: function() { this.options.onComplete = this.updateComplete.bind(this); this.onTimerEvent(); }, stop: function() { this.updater.options.onComplete = undefined; clearTimeout(this.timer); (this.onComplete || Prototype.emptyFunction).apply(this, arguments); }, updateComplete: function(response) { if (this.options.decay) { this.decay = (response.responseText == this.lastText ? this.decay * this.options.decay : 1); this.lastText = response.responseText; } this.timer = this.onTimerEvent.bind(this).delay(this.decay * this.frequency); }, onTimerEvent: function() { this.updater = new Ajax.Updater(this.container, this.url, this.options); } }); function $(element) { if (arguments.length > 1) { for (var i = 0, elements = [], length = arguments.length; i < length; i++) elements.push($(arguments[i])); return elements; } if (Object.isString(element)) element = document.getElementById(element); return Element.extend(element); } if (Prototype.BrowserFeatures.XPath) { document._getElementsByXPath = function(expression, parentElement) { var results = []; var query = document.evaluate(expression, $(parentElement) || document, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); for (var i = 0, length = query.snapshotLength; i < length; i++) results.push(Element.extend(query.snapshotItem(i))); return results; }; } /*--------------------------------------------------------------------------*/ if (!window.Node) var Node = { }; if (!Node.ELEMENT_NODE) { Object.extend(Node, { ELEMENT_NODE: 1, ATTRIBUTE_NODE: 2, TEXT_NODE: 3, CDATA_SECTION_NODE: 4, ENTITY_REFERENCE_NODE: 5, ENTITY_NODE: 6, PROCESSING_INSTRUCTION_NODE: 7, COMMENT_NODE: 8, DOCUMENT_NODE: 9, DOCUMENT_TYPE_NODE: 10, DOCUMENT_FRAGMENT_NODE: 11, NOTATION_NODE: 12 }); } (function(global) { var SETATTRIBUTE_IGNORES_NAME = (function(){ var elForm = document.createElement("form"); var elInput = document.createElement("input"); var root = document.documentElement; elInput.setAttribute("name", "test"); elForm.appendChild(elInput); root.appendChild(elForm); var isBuggy = elForm.elements ? (typeof elForm.elements.test == "undefined") : null; root.removeChild(elForm); elForm = elInput = null; return isBuggy; })(); var element = global.Element; global.Element = function(tagName, attributes) { attributes = attributes || { }; tagName = tagName.toLowerCase(); var cache = Element.cache; if (SETATTRIBUTE_IGNORES_NAME && attributes.name) { tagName = '<' + tagName + ' name="' + attributes.name + '">'; delete attributes.name; return Element.writeAttribute(document.createElement(tagName), attributes); } if (!cache[tagName]) cache[tagName] = Element.extend(document.createElement(tagName)); return Element.writeAttribute(cache[tagName].cloneNode(false), attributes); }; Object.extend(global.Element, element || { }); if (element) global.Element.prototype = element.prototype; })(this); Element.cache = { }; Element.idCounter = 1; Element.Methods = { visible: function(element) { return $(element).style.display != 'none'; }, toggle: function(element) { element = $(element); Element[Element.visible(element) ? 'hide' : 'show'](element); return element; }, hide: function(element) { element = $(element); element.style.display = 'none'; return element; }, show: function(element) { element = $(element); element.style.display = ''; return element; }, remove: function(element) { element = $(element); element.parentNode.removeChild(element); return element; }, update: (function(){ var SELECT_ELEMENT_INNERHTML_BUGGY = (function(){ var el = document.createElement("select"), isBuggy = true; el.innerHTML = ""; if (el.options && el.options[0]) { isBuggy = el.options[0].nodeName.toUpperCase() !== "OPTION"; } el = null; return isBuggy; })(); var TABLE_ELEMENT_INNERHTML_BUGGY = (function(){ try { var el = document.createElement("table"); if (el && el.tBodies) { el.innerHTML = "test"; var isBuggy = typeof el.tBodies[0] == "undefined"; el = null; return isBuggy; } } catch (e) { return true; } })(); var SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING = (function () { var s = document.createElement("script"), isBuggy = false; try { s.appendChild(document.createTextNode("")); isBuggy = !s.firstChild || s.firstChild && s.firstChild.nodeType !== 3; } catch (e) { isBuggy = true; } s = null; return isBuggy; })(); function update(element, content) { element = $(element); if (content && content.toElement) content = content.toElement(); if (Object.isElement(content)) return element.update().insert(content); content = Object.toHTML(content); var tagName = element.tagName.toUpperCase(); if (tagName === 'SCRIPT' && SCRIPT_ELEMENT_REJECTS_TEXTNODE_APPENDING) { element.text = content; return element; } if (SELECT_ELEMENT_INNERHTML_BUGGY || TABLE_ELEMENT_INNERHTML_BUGGY) { if (tagName in Element._insertionTranslations.tags) { while (element.firstChild) { element.removeChild(element.firstChild); } Element._getContentFromAnonymousElement(tagName, content.stripScripts()) .each(function(node) { element.appendChild(node) }); } else { element.innerHTML = content.stripScripts(); } } else { element.innerHTML = content.stripScripts(); } content.evalScripts.bind(content).defer(); return element; } return update; })(), replace: function(element, content) { element = $(element); if (content && content.toElement) content = content.toElement(); else if (!Object.isElement(content)) { content = Object.toHTML(content); var range = element.ownerDocument.createRange(); range.selectNode(element); content.evalScripts.bind(content).defer(); content = range.createContextualFragment(content.stripScripts()); } element.parentNode.replaceChild(content, element); return element; }, insert: function(element, insertions) { element = $(element); if (Object.isString(insertions) || Object.isNumber(insertions) || Object.isElement(insertions) || (insertions && (insertions.toElement || insertions.toHTML))) insertions = {bottom:insertions}; var content, insert, tagName, childNodes; for (var position in insertions) { content = insertions[position]; position = position.toLowerCase(); insert = Element._insertionTranslations[position]; if (content && content.toElement) content = content.toElement(); if (Object.isElement(content)) { insert(element, content); continue; } content = Object.toHTML(content); tagName = ((position == 'before' || position == 'after') ? element.parentNode : element).tagName.toUpperCase(); childNodes = Element._getContentFromAnonymousElement(tagName, content.stripScripts()); if (position == 'top' || position == 'after') childNodes.reverse(); childNodes.each(insert.curry(element)); content.evalScripts.bind(content).defer(); } return element; }, wrap: function(element, wrapper, attributes) { element = $(element); if (Object.isElement(wrapper)) $(wrapper).writeAttribute(attributes || { }); else if (Object.isString(wrapper)) wrapper = new Element(wrapper, attributes); else wrapper = new Element('div', wrapper); if (element.parentNode) element.parentNode.replaceChild(wrapper, element); wrapper.appendChild(element); return wrapper; }, inspect: function(element) { element = $(element); var result = '<' + element.tagName.toLowerCase(); $H({'id': 'id', 'className': 'class'}).each(function(pair) { var property = pair.first(), attribute = pair.last(); var value = (element[property] || '').toString(); if (value) result += ' ' + attribute + '=' + value.inspect(true); }); return result + '>'; }, recursivelyCollect: function(element, property) { element = $(element); var elements = []; while (element = element[property]) if (element.nodeType == 1) elements.push(Element.extend(element)); return elements; }, ancestors: function(element) { return Element.recursivelyCollect(element, 'parentNode'); }, descendants: function(element) { return Element.select(element, "*"); }, firstDescendant: function(element) { element = $(element).firstChild; while (element && element.nodeType != 1) element = element.nextSibling; return $(element); }, immediateDescendants: function(element) { if (!(element = $(element).firstChild)) return []; while (element && element.nodeType != 1) element = element.nextSibling; if (element) return [element].concat($(element).nextSiblings()); return []; }, previousSiblings: function(element) { return Element.recursivelyCollect(element, 'previousSibling'); }, nextSiblings: function(element) { return Element.recursivelyCollect(element, 'nextSibling'); }, siblings: function(element) { element = $(element); return Element.previousSiblings(element).reverse() .concat(Element.nextSiblings(element)); }, match: function(element, selector) { if (Object.isString(selector)) selector = new Selector(selector); return selector.match($(element)); }, up: function(element, expression, index) { element = $(element); if (arguments.length == 1) return $(element.parentNode); var ancestors = Element.ancestors(element); return Object.isNumber(expression) ? ancestors[expression] : Selector.findElement(ancestors, expression, index); }, down: function(element, expression, index) { element = $(element); if (arguments.length == 1) return Element.firstDescendant(element); return Object.isNumber(expression) ? Element.descendants(element)[expression] : Element.select(element, expression)[index || 0]; }, previous: function(element, expression, index) { element = $(element); if (arguments.length == 1) return $(Selector.handlers.previousElementSibling(element)); var previousSiblings = Element.previousSiblings(element); return Object.isNumber(expression) ? previousSiblings[expression] : Selector.findElement(previousSiblings, expression, index); }, next: function(element, expression, index) { element = $(element); if (arguments.length == 1) return $(Selector.handlers.nextElementSibling(element)); var nextSiblings = Element.nextSiblings(element); return Object.isNumber(expression) ? nextSiblings[expression] : Selector.findElement(nextSiblings, expression, index); }, select: function(element) { var args = Array.prototype.slice.call(arguments, 1); return Selector.findChildElements(element, args); }, adjacent: function(element) { var args = Array.prototype.slice.call(arguments, 1); return Selector.findChildElements(element.parentNode, args).without(element); }, identify: function(element) { element = $(element); var id = Element.readAttribute(element, 'id'); if (id) return id; do { id = 'anonymous_element_' + Element.idCounter++ } while ($(id)); Element.writeAttribute(element, 'id', id); return id; }, readAttribute: function(element, name) { element = $(element); if (Prototype.Browser.IE) { var t = Element._attributeTranslations.read; if (t.values[name]) return t.values[name](element, name); if (t.names[name]) name = t.names[name]; if (name.include(':')) { return (!element.attributes || !element.attributes[name]) ? null : element.attributes[name].value; } } return element.getAttribute(name); }, writeAttribute: function(element, name, value) { element = $(element); var attributes = { }, t = Element._attributeTranslations.write; if (typeof name == 'object') attributes = name; else attributes[name] = Object.isUndefined(value) ? true : value; for (var attr in attributes) { name = t.names[attr] || attr; value = attributes[attr]; if (t.values[attr]) name = t.values[attr](element, value); if (value === false || value === null) element.removeAttribute(name); else if (value === true) element.setAttribute(name, name); else element.setAttribute(name, value); } return element; }, getHeight: function(element) { return Element.getDimensions(element).height; }, getWidth: function(element) { return Element.getDimensions(element).width; }, classNames: function(element) { return new Element.ClassNames(element); }, hasClassName: function(element, className) { if (!(element = $(element))) return; var elementClassName = element.className; return (elementClassName.length > 0 && (elementClassName == className || new RegExp("(^|\\s)" + className + "(\\s|$)").test(elementClassName))); }, addClassName: function(element, className) { if (!(element = $(element))) return; if (!Element.hasClassName(element, className)) element.className += (element.className ? ' ' : '') + className; return element; }, removeClassName: function(element, className) { if (!(element = $(element))) return; element.className = element.className.replace( new RegExp("(^|\\s+)" + className + "(\\s+|$)"), ' ').strip(); return element; }, toggleClassName: function(element, className) { if (!(element = $(element))) return; return Element[Element.hasClassName(element, className) ? 'removeClassName' : 'addClassName'](element, className); }, cleanWhitespace: function(element) { element = $(element); var node = element.firstChild; while (node) { var nextNode = node.nextSibling; if (node.nodeType == 3 && !/\S/.test(node.nodeValue)) element.removeChild(node); node = nextNode; } return element; }, empty: function(element) { return $(element).innerHTML.blank(); }, descendantOf: function(element, ancestor) { element = $(element), ancestor = $(ancestor); if (element.compareDocumentPosition) return (element.compareDocumentPosition(ancestor) & 8) === 8; if (ancestor.contains) return ancestor.contains(element) && ancestor !== element; while (element = element.parentNode) if (element == ancestor) return true; return false; }, scrollTo: function(element) { element = $(element); var pos = Element.cumulativeOffset(element); window.scrollTo(pos[0], pos[1]); return element; }, getStyle: function(element, style) { element = $(element); style = style == 'float' ? 'cssFloat' : style.camelize(); var value = element.style[style]; if (!value || value == 'auto') { var css = document.defaultView.getComputedStyle(element, null); value = css ? css[style] : null; } if (style == 'opacity') return value ? parseFloat(value) : 1.0; return value == 'auto' ? null : value; }, getOpacity: function(element) { return $(element).getStyle('opacity'); }, setStyle: function(element, styles) { element = $(element); var elementStyle = element.style, match; if (Object.isString(styles)) { element.style.cssText += ';' + styles; return styles.include('opacity') ? element.setOpacity(styles.match(/opacity:\s*(\d?\.?\d*)/)[1]) : element; } for (var property in styles) if (property == 'opacity') element.setOpacity(styles[property]); else elementStyle[(property == 'float' || property == 'cssFloat') ? (Object.isUndefined(elementStyle.styleFloat) ? 'cssFloat' : 'styleFloat') : property] = styles[property]; return element; }, setOpacity: function(element, value) { element = $(element); element.style.opacity = (value == 1 || value === '') ? '' : (value < 0.00001) ? 0 : value; return element; }, getDimensions: function(element) { element = $(element); var display = Element.getStyle(element, 'display'); if (display != 'none' && display != null) // Safari bug return {width: element.offsetWidth, height: element.offsetHeight}; var els = element.style; var originalVisibility = els.visibility; var originalPosition = els.position; var originalDisplay = els.display; els.visibility = 'hidden'; if (originalPosition != 'fixed') // Switching fixed to absolute causes issues in Safari els.position = 'absolute'; els.display = 'block'; var originalWidth = element.clientWidth; var originalHeight = element.clientHeight; els.display = originalDisplay; els.position = originalPosition; els.visibility = originalVisibility; return {width: originalWidth, height: originalHeight}; }, makePositioned: function(element) { element = $(element); var pos = Element.getStyle(element, 'position'); if (pos == 'static' || !pos) { element._madePositioned = true; element.style.position = 'relative'; if (Prototype.Browser.Opera) { element.style.top = 0; element.style.left = 0; } } return element; }, undoPositioned: function(element) { element = $(element); if (element._madePositioned) { element._madePositioned = undefined; element.style.position = element.style.top = element.style.left = element.style.bottom = element.style.right = ''; } return element; }, makeClipping: function(element) { element = $(element); if (element._overflow) return element; element._overflow = Element.getStyle(element, 'overflow') || 'auto'; if (element._overflow !== 'hidden') element.style.overflow = 'hidden'; return element; }, undoClipping: function(element) { element = $(element); if (!element._overflow) return element; element.style.overflow = element._overflow == 'auto' ? '' : element._overflow; element._overflow = null; return element; }, cumulativeOffset: function(element) { var valueT = 0, valueL = 0; do { valueT += element.offsetTop || 0; valueL += element.offsetLeft || 0; element = element.offsetParent; } while (element); return Element._returnOffset(valueL, valueT); }, positionedOffset: function(element) { var valueT = 0, valueL = 0; do { valueT += element.offsetTop || 0; valueL += element.offsetLeft || 0; element = element.offsetParent; if (element) { if (element.tagName.toUpperCase() == 'BODY') break; var p = Element.getStyle(element, 'position'); if (p !== 'static') break; } } while (element); return Element._returnOffset(valueL, valueT); }, absolutize: function(element) { element = $(element); if (Element.getStyle(element, 'position') == 'absolute') return element; var offsets = Element.positionedOffset(element); var top = offsets[1]; var left = offsets[0]; var width = element.clientWidth; var height = element.clientHeight; element._originalLeft = left - parseFloat(element.style.left || 0); element._originalTop = top - parseFloat(element.style.top || 0); element._originalWidth = element.style.width; element._originalHeight = element.style.height; element.style.position = 'absolute'; element.style.top = top + 'px'; element.style.left = left + 'px'; element.style.width = width + 'px'; element.style.height = height + 'px'; return element; }, relativize: function(element) { element = $(element); if (Element.getStyle(element, 'position') == 'relative') return element; element.style.position = 'relative'; var top = parseFloat(element.style.top || 0) - (element._originalTop || 0); var left = parseFloat(element.style.left || 0) - (element._originalLeft || 0); element.style.top = top + 'px'; element.style.left = left + 'px'; element.style.height = element._originalHeight; element.style.width = element._originalWidth; return element; }, cumulativeScrollOffset: function(element) { var valueT = 0, valueL = 0; do { valueT += element.scrollTop || 0; valueL += element.scrollLeft || 0; element = element.parentNode; } while (element); return Element._returnOffset(valueL, valueT); }, getOffsetParent: function(element) { if (element.offsetParent) return $(element.offsetParent); if (element == document.body) return $(element); while ((element = element.parentNode) && element != document.body) if (Element.getStyle(element, 'position') != 'static') return $(element); return $(document.body); }, viewportOffset: function(forElement) { var valueT = 0, valueL = 0; var element = forElement; do { valueT += element.offsetTop || 0; valueL += element.offsetLeft || 0; if (element.offsetParent == document.body && Element.getStyle(element, 'position') == 'absolute') break; } while (element = element.offsetParent); element = forElement; do { if (!Prototype.Browser.Opera || (element.tagName && (element.tagName.toUpperCase() == 'BODY'))) { valueT -= element.scrollTop || 0; valueL -= element.scrollLeft || 0; } } while (element = element.parentNode); return Element._returnOffset(valueL, valueT); }, clonePosition: function(element, source) { var options = Object.extend({ setLeft: true, setTop: true, setWidth: true, setHeight: true, offsetTop: 0, offsetLeft: 0 }, arguments[2] || { }); source = $(source); var p = Element.viewportOffset(source); element = $(element); var delta = [0, 0]; var parent = null; if (Element.getStyle(element, 'position') == 'absolute') { parent = Element.getOffsetParent(element); delta = Element.viewportOffset(parent); } if (parent == document.body) { delta[0] -= document.body.offsetLeft; delta[1] -= document.body.offsetTop; } if (options.setLeft) element.style.left = (p[0] - delta[0] + options.offsetLeft) + 'px'; if (options.setTop) element.style.top = (p[1] - delta[1] + options.offsetTop) + 'px'; if (options.setWidth) element.style.width = source.offsetWidth + 'px'; if (options.setHeight) element.style.height = source.offsetHeight + 'px'; return element; } }; Object.extend(Element.Methods, { getElementsBySelector: Element.Methods.select, childElements: Element.Methods.immediateDescendants }); Element._attributeTranslations = { write: { names: { className: 'class', htmlFor: 'for' }, values: { } } }; if (Prototype.Browser.Opera) { Element.Methods.getStyle = Element.Methods.getStyle.wrap( function(proceed, element, style) { switch (style) { case 'left': case 'top': case 'right': case 'bottom': if (proceed(element, 'position') === 'static') return null; case 'height': case 'width': if (!Element.visible(element)) return null; var dim = parseInt(proceed(element, style), 10); if (dim !== element['offset' + style.capitalize()]) return dim + 'px'; var properties; if (style === 'height') { properties = ['border-top-width', 'padding-top', 'padding-bottom', 'border-bottom-width']; } else { properties = ['border-left-width', 'padding-left', 'padding-right', 'border-right-width']; } return properties.inject(dim, function(memo, property) { var val = proceed(element, property); return val === null ? memo : memo - parseInt(val, 10); }) + 'px'; default: return proceed(element, style); } } ); Element.Methods.readAttribute = Element.Methods.readAttribute.wrap( function(proceed, element, attribute) { if (attribute === 'title') return element.title; return proceed(element, attribute); } ); } else if (Prototype.Browser.IE) { Element.Methods.getOffsetParent = Element.Methods.getOffsetParent.wrap( function(proceed, element) { element = $(element); try { element.offsetParent } catch(e) { return $(document.body) } var position = element.getStyle('position'); if (position !== 'static') return proceed(element); element.setStyle({ position: 'relative' }); var value = proceed(element); element.setStyle({ position: position }); return value; } ); $w('positionedOffset viewportOffset').each(function(method) { Element.Methods[method] = Element.Methods[method].wrap( function(proceed, element) { element = $(element); try { element.offsetParent } catch(e) { return Element._returnOffset(0,0) } var position = element.getStyle('position'); if (position !== 'static') return proceed(element); var offsetParent = element.getOffsetParent(); if (offsetParent && offsetParent.getStyle('position') === 'fixed') offsetParent.setStyle({ zoom: 1 }); element.setStyle({ position: 'relative' }); var value = proceed(element); element.setStyle({ position: position }); return value; } ); }); Element.Methods.cumulativeOffset = Element.Methods.cumulativeOffset.wrap( function(proceed, element) { try { element.offsetParent } catch(e) { return Element._returnOffset(0,0) } return proceed(element); } ); Element.Methods.getStyle = function(element, style) { element = $(element); style = (style == 'float' || style == 'cssFloat') ? 'styleFloat' : style.camelize(); var value = element.style[style]; if (!value && element.currentStyle) value = element.currentStyle[style]; if (style == 'opacity') { if (value = (element.getStyle('filter') || '').match(/alpha\(opacity=(.*)\)/)) if (value[1]) return parseFloat(value[1]) / 100; return 1.0; } if (value == 'auto') { if ((style == 'width' || style == 'height') && (element.getStyle('display') != 'none')) return element['offset' + style.capitalize()] + 'px'; return null; } return value; }; Element.Methods.setOpacity = function(element, value) { function stripAlpha(filter){ return filter.replace(/alpha\([^\)]*\)/gi,''); } element = $(element); var currentStyle = element.currentStyle; if ((currentStyle && !currentStyle.hasLayout) || (!currentStyle && element.style.zoom == 'normal')) element.style.zoom = 1; var filter = element.getStyle('filter'), style = element.style; if (value == 1 || value === '') { (filter = stripAlpha(filter)) ? style.filter = filter : style.removeAttribute('filter'); return element; } else if (value < 0.00001) value = 0; style.filter = stripAlpha(filter) + 'alpha(opacity=' + (value * 100) + ')'; return element; }; Element._attributeTranslations = (function(){ var classProp = 'className'; var forProp = 'for'; var el = document.createElement('div'); el.setAttribute(classProp, 'x'); if (el.className !== 'x') { el.setAttribute('class', 'x'); if (el.className === 'x') { classProp = 'class'; } } el = null; el = document.createElement('label'); el.setAttribute(forProp, 'x'); if (el.htmlFor !== 'x') { el.setAttribute('htmlFor', 'x'); if (el.htmlFor === 'x') { forProp = 'htmlFor'; } } el = null; return { read: { names: { 'class': classProp, 'className': classProp, 'for': forProp, 'htmlFor': forProp }, values: { _getAttr: function(element, attribute) { return element.getAttribute(attribute); }, _getAttr2: function(element, attribute) { return element.getAttribute(attribute, 2); }, _getAttrNode: function(element, attribute) { var node = element.getAttributeNode(attribute); return node ? node.value : ""; }, _getEv: (function(){ var el = document.createElement('div'); el.onclick = Prototype.emptyFunction; var value = el.getAttribute('onclick'); var f; if (String(value).indexOf('{') > -1) { f = function(element, attribute) { attribute = element.getAttribute(attribute); if (!attribute) return null; attribute = attribute.toString(); attribute = attribute.split('{')[1]; attribute = attribute.split('}')[0]; return attribute.strip(); }; } else if (value === '') { f = function(element, attribute) { attribute = element.getAttribute(attribute); if (!attribute) return null; return attribute.strip(); }; } el = null; return f; })(), _flag: function(element, attribute) { return $(element).hasAttribute(attribute) ? attribute : null; }, style: function(element) { return element.style.cssText.toLowerCase(); }, title: function(element) { return element.title; } } } } })(); Element._attributeTranslations.write = { names: Object.extend({ cellpadding: 'cellPadding', cellspacing: 'cellSpacing' }, Element._attributeTranslations.read.names), values: { checked: function(element, value) { element.checked = !!value; }, style: function(element, value) { element.style.cssText = value ? value : ''; } } }; Element._attributeTranslations.has = {}; $w('colSpan rowSpan vAlign dateTime accessKey tabIndex ' + 'encType maxLength readOnly longDesc frameBorder').each(function(attr) { Element._attributeTranslations.write.names[attr.toLowerCase()] = attr; Element._attributeTranslations.has[attr.toLowerCase()] = attr; }); (function(v) { Object.extend(v, { href: v._getAttr2, src: v._getAttr2, type: v._getAttr, action: v._getAttrNode, disabled: v._flag, checked: v._flag, readonly: v._flag, multiple: v._flag, onload: v._getEv, onunload: v._getEv, onclick: v._getEv, ondblclick: v._getEv, onmousedown: v._getEv, onmouseup: v._getEv, onmouseover: v._getEv, onmousemove: v._getEv, onmouseout: v._getEv, onfocus: v._getEv, onblur: v._getEv, onkeypress: v._getEv, onkeydown: v._getEv, onkeyup: v._getEv, onsubmit: v._getEv, onreset: v._getEv, onselect: v._getEv, onchange: v._getEv }); })(Element._attributeTranslations.read.values); if (Prototype.BrowserFeatures.ElementExtensions) { (function() { function _descendants(element) { var nodes = element.getElementsByTagName('*'), results = []; for (var i = 0, node; node = nodes[i]; i++) if (node.tagName !== "!") // Filter out comment nodes. results.push(node); return results; } Element.Methods.down = function(element, expression, index) { element = $(element); if (arguments.length == 1) return element.firstDescendant(); return Object.isNumber(expression) ? _descendants(element)[expression] : Element.select(element, expression)[index || 0]; } })(); } } else if (Prototype.Browser.Gecko && /rv:1\.8\.0/.test(navigator.userAgent)) { Element.Methods.setOpacity = function(element, value) { element = $(element); element.style.opacity = (value == 1) ? 0.999999 : (value === '') ? '' : (value < 0.00001) ? 0 : value; return element; }; } else if (Prototype.Browser.WebKit) { Element.Methods.setOpacity = function(element, value) { element = $(element); element.style.opacity = (value == 1 || value === '') ? '' : (value < 0.00001) ? 0 : value; if (value == 1) if(element.tagName.toUpperCase() == 'IMG' && element.width) { element.width++; element.width--; } else try { var n = document.createTextNode(' '); element.appendChild(n); element.removeChild(n); } catch (e) { } return element; }; Element.Methods.cumulativeOffset = function(element) { var valueT = 0, valueL = 0; do { valueT += element.offsetTop || 0; valueL += element.offsetLeft || 0; if (element.offsetParent == document.body) if (Element.getStyle(element, 'position') == 'absolute') break; element = element.offsetParent; } while (element); return Element._returnOffset(valueL, valueT); }; } if ('outerHTML' in document.documentElement) { Element.Methods.replace = function(element, content) { element = $(element); if (content && content.toElement) content = content.toElement(); if (Object.isElement(content)) { element.parentNode.replaceChild(content, element); return element; } content = Object.toHTML(content); var parent = element.parentNode, tagName = parent.tagName.toUpperCase(); if (Element._insertionTranslations.tags[tagName]) { var nextSibling = element.next(); var fragments = Element._getContentFromAnonymousElement(tagName, content.stripScripts()); parent.removeChild(element); if (nextSibling) fragments.each(function(node) { parent.insertBefore(node, nextSibling) }); else fragments.each(function(node) { parent.appendChild(node) }); } else element.outerHTML = content.stripScripts(); content.evalScripts.bind(content).defer(); return element; }; } Element._returnOffset = function(l, t) { var result = [l, t]; result.left = l; result.top = t; return result; }; Element._getContentFromAnonymousElement = function(tagName, html) { var div = new Element('div'), t = Element._insertionTranslations.tags[tagName]; if (t) { div.innerHTML = t[0] + html + t[1]; t[2].times(function() { div = div.firstChild }); } else div.innerHTML = html; return $A(div.childNodes); }; Element._insertionTranslations = { before: function(element, node) { element.parentNode.insertBefore(node, element); }, top: function(element, node) { element.insertBefore(node, element.firstChild); }, bottom: function(element, node) { element.appendChild(node); }, after: function(element, node) { element.parentNode.insertBefore(node, element.nextSibling); }, tags: { TABLE: ['', '
', 1], TBODY: ['', '
', 2], TR: ['', '
', 3], TD: ['
', '
', 4], SELECT: ['', 1] } }; (function() { var tags = Element._insertionTranslations.tags; Object.extend(tags, { THEAD: tags.TBODY, TFOOT: tags.TBODY, TH: tags.TD }); })(); Element.Methods.Simulated = { hasAttribute: function(element, attribute) { attribute = Element._attributeTranslations.has[attribute] || attribute; var node = $(element).getAttributeNode(attribute); return !!(node && node.specified); } }; Element.Methods.ByTag = { }; Object.extend(Element, Element.Methods); (function(div) { if (!Prototype.BrowserFeatures.ElementExtensions && div['__proto__']) { window.HTMLElement = { }; window.HTMLElement.prototype = div['__proto__']; Prototype.BrowserFeatures.ElementExtensions = true; } div = null; })(document.createElement('div')) Element.extend = (function() { function checkDeficiency(tagName) { if (typeof window.Element != 'undefined') { var proto = window.Element.prototype; if (proto) { var id = '_' + (Math.random()+'').slice(2); var el = document.createElement(tagName); proto[id] = 'x'; var isBuggy = (el[id] !== 'x'); delete proto[id]; el = null; return isBuggy; } } return false; } function extendElementWith(element, methods) { for (var property in methods) { var value = methods[property]; if (Object.isFunction(value) && !(property in element)) element[property] = value.methodize(); } } var HTMLOBJECTELEMENT_PROTOTYPE_BUGGY = checkDeficiency('object'); if (Prototype.BrowserFeatures.SpecificElementExtensions) { if (HTMLOBJECTELEMENT_PROTOTYPE_BUGGY) { return function(element) { if (element && typeof element._extendedByPrototype == 'undefined') { var t = element.tagName; if (t && (/^(?:object|applet|embed)$/i.test(t))) { extendElementWith(element, Element.Methods); extendElementWith(element, Element.Methods.Simulated); extendElementWith(element, Element.Methods.ByTag[t.toUpperCase()]); } } return element; } } return Prototype.K; } var Methods = { }, ByTag = Element.Methods.ByTag; var extend = Object.extend(function(element) { if (!element || typeof element._extendedByPrototype != 'undefined' || element.nodeType != 1 || element == window) return element; var methods = Object.clone(Methods), tagName = element.tagName.toUpperCase(); if (ByTag[tagName]) Object.extend(methods, ByTag[tagName]); extendElementWith(element, methods); element._extendedByPrototype = Prototype.emptyFunction; return element; }, { refresh: function() { if (!Prototype.BrowserFeatures.ElementExtensions) { Object.extend(Methods, Element.Methods); Object.extend(Methods, Element.Methods.Simulated); } } }); extend.refresh(); return extend; })(); Element.hasAttribute = function(element, attribute) { if (element.hasAttribute) return element.hasAttribute(attribute); return Element.Methods.Simulated.hasAttribute(element, attribute); }; Element.addMethods = function(methods) { var F = Prototype.BrowserFeatures, T = Element.Methods.ByTag; if (!methods) { Object.extend(Form, Form.Methods); Object.extend(Form.Element, Form.Element.Methods); Object.extend(Element.Methods.ByTag, { "FORM": Object.clone(Form.Methods), "INPUT": Object.clone(Form.Element.Methods), "SELECT": Object.clone(Form.Element.Methods), "TEXTAREA": Object.clone(Form.Element.Methods) }); } if (arguments.length == 2) { var tagName = methods; methods = arguments[1]; } if (!tagName) Object.extend(Element.Methods, methods || { }); else { if (Object.isArray(tagName)) tagName.each(extend); else extend(tagName); } function extend(tagName) { tagName = tagName.toUpperCase(); if (!Element.Methods.ByTag[tagName]) Element.Methods.ByTag[tagName] = { }; Object.extend(Element.Methods.ByTag[tagName], methods); } function copy(methods, destination, onlyIfAbsent) { onlyIfAbsent = onlyIfAbsent || false; for (var property in methods) { var value = methods[property]; if (!Object.isFunction(value)) continue; if (!onlyIfAbsent || !(property in destination)) destination[property] = value.methodize(); } } function findDOMClass(tagName) { var klass; var trans = { "OPTGROUP": "OptGroup", "TEXTAREA": "TextArea", "P": "Paragraph", "FIELDSET": "FieldSet", "UL": "UList", "OL": "OList", "DL": "DList", "DIR": "Directory", "H1": "Heading", "H2": "Heading", "H3": "Heading", "H4": "Heading", "H5": "Heading", "H6": "Heading", "Q": "Quote", "INS": "Mod", "DEL": "Mod", "A": "Anchor", "IMG": "Image", "CAPTION": "TableCaption", "COL": "TableCol", "COLGROUP": "TableCol", "THEAD": "TableSection", "TFOOT": "TableSection", "TBODY": "TableSection", "TR": "TableRow", "TH": "TableCell", "TD": "TableCell", "FRAMESET": "FrameSet", "IFRAME": "IFrame" }; if (trans[tagName]) klass = 'HTML' + trans[tagName] + 'Element'; if (window[klass]) return window[klass]; klass = 'HTML' + tagName + 'Element'; if (window[klass]) return window[klass]; klass = 'HTML' + tagName.capitalize() + 'Element'; if (window[klass]) return window[klass]; var element = document.createElement(tagName); var proto = element['__proto__'] || element.constructor.prototype; element = null; return proto; } var elementPrototype = window.HTMLElement ? HTMLElement.prototype : Element.prototype; if (F.ElementExtensions) { copy(Element.Methods, elementPrototype); copy(Element.Methods.Simulated, elementPrototype, true); } if (F.SpecificElementExtensions) { for (var tag in Element.Methods.ByTag) { var klass = findDOMClass(tag); if (Object.isUndefined(klass)) continue; copy(T[tag], klass.prototype); } } Object.extend(Element, Element.Methods); delete Element.ByTag; if (Element.extend.refresh) Element.extend.refresh(); Element.cache = { }; }; document.viewport = { getDimensions: function() { return { width: this.getWidth(), height: this.getHeight() }; }, getScrollOffsets: function() { return Element._returnOffset( window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft, window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop); } }; (function(viewport) { var B = Prototype.Browser, doc = document, element, property = {}; function getRootElement() { if (B.WebKit && !doc.evaluate) return document; if (B.Opera && window.parseFloat(window.opera.version()) < 9.5) return document.body; return document.documentElement; } function define(D) { if (!element) element = getRootElement(); property[D] = 'client' + D; viewport['get' + D] = function() { return element[property[D]] }; return viewport['get' + D](); } viewport.getWidth = define.curry('Width'); viewport.getHeight = define.curry('Height'); })(document.viewport); Element.Storage = { UID: 1 }; Element.addMethods({ getStorage: function(element) { if (!(element = $(element))) return; var uid; if (element === window) { uid = 0; } else { if (typeof element._prototypeUID === "undefined") element._prototypeUID = [Element.Storage.UID++]; uid = element._prototypeUID[0]; } if (!Element.Storage[uid]) Element.Storage[uid] = $H(); return Element.Storage[uid]; }, store: function(element, key, value) { if (!(element = $(element))) return; if (arguments.length === 2) { Element.getStorage(element).update(key); } else { Element.getStorage(element).set(key, value); } return element; }, retrieve: function(element, key, defaultValue) { if (!(element = $(element))) return; var hash = Element.getStorage(element), value = hash.get(key); if (Object.isUndefined(value)) { hash.set(key, defaultValue); value = defaultValue; } return value; }, clone: function(element, deep) { if (!(element = $(element))) return; var clone = element.cloneNode(deep); clone._prototypeUID = void 0; if (deep) { var descendants = Element.select(clone, '*'), i = descendants.length; while (i--) { descendants[i]._prototypeUID = void 0; } } return Element.extend(clone); } }); /* Portions of the Selector class are derived from Jack Slocum's DomQuery, * part of YUI-Ext version 0.40, distributed under the terms of an MIT-style * license. Please see http://www.yui-ext.com/ for more information. */ var Selector = Class.create({ initialize: function(expression) { this.expression = expression.strip(); if (this.shouldUseSelectorsAPI()) { this.mode = 'selectorsAPI'; } else if (this.shouldUseXPath()) { this.mode = 'xpath'; this.compileXPathMatcher(); } else { this.mode = "normal"; this.compileMatcher(); } }, shouldUseXPath: (function() { var IS_DESCENDANT_SELECTOR_BUGGY = (function(){ var isBuggy = false; if (document.evaluate && window.XPathResult) { var el = document.createElement('div'); el.innerHTML = '
'; var xpath = ".//*[local-name()='ul' or local-name()='UL']" + "//*[local-name()='li' or local-name()='LI']"; var result = document.evaluate(xpath, el, null, XPathResult.ORDERED_NODE_SNAPSHOT_TYPE, null); isBuggy = (result.snapshotLength !== 2); el = null; } return isBuggy; })(); return function() { if (!Prototype.BrowserFeatures.XPath) return false; var e = this.expression; if (Prototype.Browser.WebKit && (e.include("-of-type") || e.include(":empty"))) return false; if ((/(\[[\w-]*?:|:checked)/).test(e)) return false; if (IS_DESCENDANT_SELECTOR_BUGGY) return false; return true; } })(), shouldUseSelectorsAPI: function() { if (!Prototype.BrowserFeatures.SelectorsAPI) return false; if (Selector.CASE_INSENSITIVE_CLASS_NAMES) return false; if (!Selector._div) Selector._div = new Element('div'); try { Selector._div.querySelector(this.expression); } catch(e) { return false; } return true; }, compileMatcher: function() { var e = this.expression, ps = Selector.patterns, h = Selector.handlers, c = Selector.criteria, le, p, m, len = ps.length, name; if (Selector._cache[e]) { this.matcher = Selector._cache[e]; return; } this.matcher = ["this.matcher = function(root) {", "var r = root, h = Selector.handlers, c = false, n;"]; while (e && le != e && (/\S/).test(e)) { le = e; for (var i = 0; i"; } }); if (Prototype.BrowserFeatures.SelectorsAPI && document.compatMode === 'BackCompat') { Selector.CASE_INSENSITIVE_CLASS_NAMES = (function(){ var div = document.createElement('div'), span = document.createElement('span'); div.id = "prototype_test_id"; span.className = 'Test'; div.appendChild(span); var isIgnored = (div.querySelector('#prototype_test_id .test') !== null); div = span = null; return isIgnored; })(); } Object.extend(Selector, { _cache: { }, xpath: { descendant: "//*", child: "/*", adjacent: "/following-sibling::*[1]", laterSibling: '/following-sibling::*', tagName: function(m) { if (m[1] == '*') return ''; return "[local-name()='" + m[1].toLowerCase() + "' or local-name()='" + m[1].toUpperCase() + "']"; }, className: "[contains(concat(' ', @class, ' '), ' #{1} ')]", id: "[@id='#{1}']", attrPresence: function(m) { m[1] = m[1].toLowerCase(); return new Template("[@#{1}]").evaluate(m); }, attr: function(m) { m[1] = m[1].toLowerCase(); m[3] = m[5] || m[6]; return new Template(Selector.xpath.operators[m[2]]).evaluate(m); }, pseudo: function(m) { var h = Selector.xpath.pseudos[m[1]]; if (!h) return ''; if (Object.isFunction(h)) return h(m); return new Template(Selector.xpath.pseudos[m[1]]).evaluate(m); }, operators: { '=': "[@#{1}='#{3}']", '!=': "[@#{1}!='#{3}']", '^=': "[starts-with(@#{1}, '#{3}')]", '$=': "[substring(@#{1}, (string-length(@#{1}) - string-length('#{3}') + 1))='#{3}']", '*=': "[contains(@#{1}, '#{3}')]", '~=': "[contains(concat(' ', @#{1}, ' '), ' #{3} ')]", '|=': "[contains(concat('-', @#{1}, '-'), '-#{3}-')]" }, pseudos: { 'first-child': '[not(preceding-sibling::*)]', 'last-child': '[not(following-sibling::*)]', 'only-child': '[not(preceding-sibling::* or following-sibling::*)]', 'empty': "[count(*) = 0 and (count(text()) = 0)]", 'checked': "[@checked]", 'disabled': "[(@disabled) and (@type!='hidden')]", 'enabled': "[not(@disabled) and (@type!='hidden')]", 'not': function(m) { var e = m[6], p = Selector.patterns, x = Selector.xpath, le, v, len = p.length, name; var exclusion = []; while (e && le != e && (/\S/).test(e)) { le = e; for (var i = 0; i= 0)]"; return new Template(predicate).evaluate({ fragment: fragment, a: a, b: b }); } } } }, criteria: { tagName: 'n = h.tagName(n, r, "#{1}", c); c = false;', className: 'n = h.className(n, r, "#{1}", c); c = false;', id: 'n = h.id(n, r, "#{1}", c); c = false;', attrPresence: 'n = h.attrPresence(n, r, "#{1}", c); c = false;', attr: function(m) { m[3] = (m[5] || m[6]); return new Template('n = h.attr(n, r, "#{1}", "#{3}", "#{2}", c); c = false;').evaluate(m); }, pseudo: function(m) { if (m[6]) m[6] = m[6].replace(/"/g, '\\"'); return new Template('n = h.pseudo(n, "#{1}", "#{6}", r, c); c = false;').evaluate(m); }, descendant: 'c = "descendant";', child: 'c = "child";', adjacent: 'c = "adjacent";', laterSibling: 'c = "laterSibling";' }, patterns: [ { name: 'laterSibling', re: /^\s*~\s*/ }, { name: 'child', re: /^\s*>\s*/ }, { name: 'adjacent', re: /^\s*\+\s*/ }, { name: 'descendant', re: /^\s/ }, { name: 'tagName', re: /^\s*(\*|[\w\-]+)(\b|$)?/ }, { name: 'id', re: /^#([\w\-\*]+)(\b|$)/ }, { name: 'className', re: /^\.([\w\-\*]+)(\b|$)/ }, { name: 'pseudo', re: /^:((first|last|nth|nth-last|only)(-child|-of-type)|empty|checked|(en|dis)abled|not)(\((.*?)\))?(\b|$|(?=\s|[:+~>]))/ }, { name: 'attrPresence', re: /^\[((?:[\w-]+:)?[\w-]+)\]/ }, { name: 'attr', re: /\[((?:[\w-]*:)?[\w-]+)\s*(?:([!^$*~|]?=)\s*((['"])([^\4]*?)\4|([^'"][^\]]*?)))?\]/ } ], assertions: { tagName: function(element, matches) { return matches[1].toUpperCase() == element.tagName.toUpperCase(); }, className: function(element, matches) { return Element.hasClassName(element, matches[1]); }, id: function(element, matches) { return element.id === matches[1]; }, attrPresence: function(element, matches) { return Element.hasAttribute(element, matches[1]); }, attr: function(element, matches) { var nodeValue = Element.readAttribute(element, matches[1]); return nodeValue && Selector.operators[matches[2]](nodeValue, matches[5] || matches[6]); } }, handlers: { concat: function(a, b) { for (var i = 0, node; node = b[i]; i++) a.push(node); return a; }, mark: function(nodes) { var _true = Prototype.emptyFunction; for (var i = 0, node; node = nodes[i]; i++) node._countedByPrototype = _true; return nodes; }, unmark: (function(){ var PROPERTIES_ATTRIBUTES_MAP = (function(){ var el = document.createElement('div'), isBuggy = false, propName = '_countedByPrototype', value = 'x' el[propName] = value; isBuggy = (el.getAttribute(propName) === value); el = null; return isBuggy; })(); return PROPERTIES_ATTRIBUTES_MAP ? function(nodes) { for (var i = 0, node; node = nodes[i]; i++) node.removeAttribute('_countedByPrototype'); return nodes; } : function(nodes) { for (var i = 0, node; node = nodes[i]; i++) node._countedByPrototype = void 0; return nodes; } })(), index: function(parentNode, reverse, ofType) { parentNode._countedByPrototype = Prototype.emptyFunction; if (reverse) { for (var nodes = parentNode.childNodes, i = nodes.length - 1, j = 1; i >= 0; i--) { var node = nodes[i]; if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++; } } else { for (var i = 0, j = 1, nodes = parentNode.childNodes; node = nodes[i]; i++) if (node.nodeType == 1 && (!ofType || node._countedByPrototype)) node.nodeIndex = j++; } }, unique: function(nodes) { if (nodes.length == 0) return nodes; var results = [], n; for (var i = 0, l = nodes.length; i < l; i++) if (typeof (n = nodes[i])._countedByPrototype == 'undefined') { n._countedByPrototype = Prototype.emptyFunction; results.push(Element.extend(n)); } return Selector.handlers.unmark(results); }, descendant: function(nodes) { var h = Selector.handlers; for (var i = 0, results = [], node; node = nodes[i]; i++) h.concat(results, node.getElementsByTagName('*')); return results; }, child: function(nodes) { var h = Selector.handlers; for (var i = 0, results = [], node; node = nodes[i]; i++) { for (var j = 0, child; child = node.childNodes[j]; j++) if (child.nodeType == 1 && child.tagName != '!') results.push(child); } return results; }, adjacent: function(nodes) { for (var i = 0, results = [], node; node = nodes[i]; i++) { var next = this.nextElementSibling(node); if (next) results.push(next); } return results; }, laterSibling: function(nodes) { var h = Selector.handlers; for (var i = 0, results = [], node; node = nodes[i]; i++) h.concat(results, Element.nextSiblings(node)); return results; }, nextElementSibling: function(node) { while (node = node.nextSibling) if (node.nodeType == 1) return node; return null; }, previousElementSibling: function(node) { while (node = node.previousSibling) if (node.nodeType == 1) return node; return null; }, tagName: function(nodes, root, tagName, combinator) { var uTagName = tagName.toUpperCase(); var results = [], h = Selector.handlers; if (nodes) { if (combinator) { if (combinator == "descendant") { for (var i = 0, node; node = nodes[i]; i++) h.concat(results, node.getElementsByTagName(tagName)); return results; } else nodes = this[combinator](nodes); if (tagName == "*") return nodes; } for (var i = 0, node; node = nodes[i]; i++) if (node.tagName.toUpperCase() === uTagName) results.push(node); return results; } else return root.getElementsByTagName(tagName); }, id: function(nodes, root, id, combinator) { var targetNode = $(id), h = Selector.handlers; if (root == document) { if (!targetNode) return []; if (!nodes) return [targetNode]; } else { if (!root.sourceIndex || root.sourceIndex < 1) { var nodes = root.getElementsByTagName('*'); for (var j = 0, node; node = nodes[j]; j++) { if (node.id === id) return [node]; } } } if (nodes) { if (combinator) { if (combinator == 'child') { for (var i = 0, node; node = nodes[i]; i++) if (targetNode.parentNode == node) return [targetNode]; } else if (combinator == 'descendant') { for (var i = 0, node; node = nodes[i]; i++) if (Element.descendantOf(targetNode, node)) return [targetNode]; } else if (combinator == 'adjacent') { for (var i = 0, node; node = nodes[i]; i++) if (Selector.handlers.previousElementSibling(targetNode) == node) return [targetNode]; } else nodes = h[combinator](nodes); } for (var i = 0, node; node = nodes[i]; i++) if (node == targetNode) return [targetNode]; return []; } return (targetNode && Element.descendantOf(targetNode, root)) ? [targetNode] : []; }, className: function(nodes, root, className, combinator) { if (nodes && combinator) nodes = this[combinator](nodes); return Selector.handlers.byClassName(nodes, root, className); }, byClassName: function(nodes, root, className) { if (!nodes) nodes = Selector.handlers.descendant([root]); var needle = ' ' + className + ' '; for (var i = 0, results = [], node, nodeClassName; node = nodes[i]; i++) { nodeClassName = node.className; if (nodeClassName.length == 0) continue; if (nodeClassName == className || (' ' + nodeClassName + ' ').include(needle)) results.push(node); } return results; }, attrPresence: function(nodes, root, attr, combinator) { if (!nodes) nodes = root.getElementsByTagName("*"); if (nodes && combinator) nodes = this[combinator](nodes); var results = []; for (var i = 0, node; node = nodes[i]; i++) if (Element.hasAttribute(node, attr)) results.push(node); return results; }, attr: function(nodes, root, attr, value, operator, combinator) { if (!nodes) nodes = root.getElementsByTagName("*"); if (nodes && combinator) nodes = this[combinator](nodes); var handler = Selector.operators[operator], results = []; for (var i = 0, node; node = nodes[i]; i++) { var nodeValue = Element.readAttribute(node, attr); if (nodeValue === null) continue; if (handler(nodeValue, value)) results.push(node); } return results; }, pseudo: function(nodes, name, value, root, combinator) { if (nodes && combinator) nodes = this[combinator](nodes); if (!nodes) nodes = root.getElementsByTagName("*"); return Selector.pseudos[name](nodes, value, root); } }, pseudos: { 'first-child': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) { if (Selector.handlers.previousElementSibling(node)) continue; results.push(node); } return results; }, 'last-child': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) { if (Selector.handlers.nextElementSibling(node)) continue; results.push(node); } return results; }, 'only-child': function(nodes, value, root) { var h = Selector.handlers; for (var i = 0, results = [], node; node = nodes[i]; i++) if (!h.previousElementSibling(node) && !h.nextElementSibling(node)) results.push(node); return results; }, 'nth-child': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, formula, root); }, 'nth-last-child': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, formula, root, true); }, 'nth-of-type': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, formula, root, false, true); }, 'nth-last-of-type': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, formula, root, true, true); }, 'first-of-type': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, "1", root, false, true); }, 'last-of-type': function(nodes, formula, root) { return Selector.pseudos.nth(nodes, "1", root, true, true); }, 'only-of-type': function(nodes, formula, root) { var p = Selector.pseudos; return p['last-of-type'](p['first-of-type'](nodes, formula, root), formula, root); }, getIndices: function(a, b, total) { if (a == 0) return b > 0 ? [b] : []; return $R(1, total).inject([], function(memo, i) { if (0 == (i - b) % a && (i - b) / a >= 0) memo.push(i); return memo; }); }, nth: function(nodes, formula, root, reverse, ofType) { if (nodes.length == 0) return []; if (formula == 'even') formula = '2n+0'; if (formula == 'odd') formula = '2n+1'; var h = Selector.handlers, results = [], indexed = [], m; h.mark(nodes); for (var i = 0, node; node = nodes[i]; i++) { if (!node.parentNode._countedByPrototype) { h.index(node.parentNode, reverse, ofType); indexed.push(node.parentNode); } } if (formula.match(/^\d+$/)) { // just a number formula = Number(formula); for (var i = 0, node; node = nodes[i]; i++) if (node.nodeIndex == formula) results.push(node); } else if (m = formula.match(/^(-?\d*)?n(([+-])(\d+))?/)) { // an+b if (m[1] == "-") m[1] = -1; var a = m[1] ? Number(m[1]) : 1; var b = m[2] ? Number(m[2]) : 0; var indices = Selector.pseudos.getIndices(a, b, nodes.length); for (var i = 0, node, l = indices.length; node = nodes[i]; i++) { for (var j = 0; j < l; j++) if (node.nodeIndex == indices[j]) results.push(node); } } h.unmark(nodes); h.unmark(indexed); return results; }, 'empty': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) { if (node.tagName == '!' || node.firstChild) continue; results.push(node); } return results; }, 'not': function(nodes, selector, root) { var h = Selector.handlers, selectorType, m; var exclusions = new Selector(selector).findElements(root); h.mark(exclusions); for (var i = 0, results = [], node; node = nodes[i]; i++) if (!node._countedByPrototype) results.push(node); h.unmark(exclusions); return results; }, 'enabled': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) if (!node.disabled && (!node.type || node.type !== 'hidden')) results.push(node); return results; }, 'disabled': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) if (node.disabled) results.push(node); return results; }, 'checked': function(nodes, value, root) { for (var i = 0, results = [], node; node = nodes[i]; i++) if (node.checked) results.push(node); return results; } }, operators: { '=': function(nv, v) { return nv == v; }, '!=': function(nv, v) { return nv != v; }, '^=': function(nv, v) { return nv == v || nv && nv.startsWith(v); }, '$=': function(nv, v) { return nv == v || nv && nv.endsWith(v); }, '*=': function(nv, v) { return nv == v || nv && nv.include(v); }, '~=': function(nv, v) { return (' ' + nv + ' ').include(' ' + v + ' '); }, '|=': function(nv, v) { return ('-' + (nv || "").toUpperCase() + '-').include('-' + (v || "").toUpperCase() + '-'); } }, split: function(expression) { var expressions = []; expression.scan(/(([\w#:.~>+()\s-]+|\*|\[.*?\])+)\s*(,|$)/, function(m) { expressions.push(m[1].strip()); }); return expressions; }, matchElements: function(elements, expression) { var matches = $$(expression), h = Selector.handlers; h.mark(matches); for (var i = 0, results = [], element; element = elements[i]; i++) if (element._countedByPrototype) results.push(element); h.unmark(matches); return results; }, findElement: function(elements, expression, index) { if (Object.isNumber(expression)) { index = expression; expression = false; } return Selector.matchElements(elements, expression || '*')[index || 0]; }, findChildElements: function(element, expressions) { expressions = Selector.split(expressions.join(',')); var results = [], h = Selector.handlers; for (var i = 0, l = expressions.length, selector; i < l; i++) { selector = new Selector(expressions[i].strip()); h.concat(results, selector.findElements(element)); } return (l > 1) ? h.unique(results) : results; } }); if (Prototype.Browser.IE) { Object.extend(Selector.handlers, { concat: function(a, b) { for (var i = 0, node; node = b[i]; i++) if (node.tagName !== "!") a.push(node); return a; } }); } function $$() { return Selector.findChildElements(document, $A(arguments)); } var Form = { reset: function(form) { form = $(form); form.reset(); return form; }, serializeElements: function(elements, options) { if (typeof options != 'object') options = { hash: !!options }; else if (Object.isUndefined(options.hash)) options.hash = true; var key, value, submitted = false, submit = options.submit; var data = elements.inject({ }, function(result, element) { if (!element.disabled && element.name) { key = element.name; value = $(element).getValue(); if (value != null && element.type != 'file' && (element.type != 'submit' || (!submitted && submit !== false && (!submit || key == submit) && (submitted = true)))) { if (key in result) { if (!Object.isArray(result[key])) result[key] = [result[key]]; result[key].push(value); } else result[key] = value; } } return result; }); return options.hash ? data : Object.toQueryString(data); } }; Form.Methods = { serialize: function(form, options) { return Form.serializeElements(Form.getElements(form), options); }, getElements: function(form) { var elements = $(form).getElementsByTagName('*'), element, arr = [ ], serializers = Form.Element.Serializers; for (var i = 0; element = elements[i]; i++) { arr.push(element); } return arr.inject([], function(elements, child) { if (serializers[child.tagName.toLowerCase()]) elements.push(Element.extend(child)); return elements; }) }, getInputs: function(form, typeName, name) { form = $(form); var inputs = form.getElementsByTagName('input'); if (!typeName && !name) return $A(inputs).map(Element.extend); for (var i = 0, matchingInputs = [], length = inputs.length; i < length; i++) { var input = inputs[i]; if ((typeName && input.type != typeName) || (name && input.name != name)) continue; matchingInputs.push(Element.extend(input)); } return matchingInputs; }, disable: function(form) { form = $(form); Form.getElements(form).invoke('disable'); return form; }, enable: function(form) { form = $(form); Form.getElements(form).invoke('enable'); return form; }, findFirstElement: function(form) { var elements = $(form).getElements().findAll(function(element) { return 'hidden' != element.type && !element.disabled; }); var firstByIndex = elements.findAll(function(element) { return element.hasAttribute('tabIndex') && element.tabIndex >= 0; }).sortBy(function(element) { return element.tabIndex }).first(); return firstByIndex ? firstByIndex : elements.find(function(element) { return /^(?:input|select|textarea)$/i.test(element.tagName); }); }, focusFirstElement: function(form) { form = $(form); form.findFirstElement().activate(); return form; }, request: function(form, options) { form = $(form), options = Object.clone(options || { }); var params = options.parameters, action = form.readAttribute('action') || ''; if (action.blank()) action = window.location.href; options.parameters = form.serialize(true); if (params) { if (Object.isString(params)) params = params.toQueryParams(); Object.extend(options.parameters, params); } if (form.hasAttribute('method') && !options.method) options.method = form.method; return new Ajax.Request(action, options); } }; /*--------------------------------------------------------------------------*/ Form.Element = { focus: function(element) { $(element).focus(); return element; }, select: function(element) { $(element).select(); return element; } }; Form.Element.Methods = { serialize: function(element) { element = $(element); if (!element.disabled && element.name) { var value = element.getValue(); if (value != undefined) { var pair = { }; pair[element.name] = value; return Object.toQueryString(pair); } } return ''; }, getValue: function(element) { element = $(element); var method = element.tagName.toLowerCase(); return Form.Element.Serializers[method](element); }, setValue: function(element, value) { element = $(element); var method = element.tagName.toLowerCase(); Form.Element.Serializers[method](element, value); return element; }, clear: function(element) { $(element).value = ''; return element; }, present: function(element) { return $(element).value != ''; }, activate: function(element) { element = $(element); try { element.focus(); if (element.select && (element.tagName.toLowerCase() != 'input' || !(/^(?:button|reset|submit)$/i.test(element.type)))) element.select(); } catch (e) { } return element; }, disable: function(element) { element = $(element); element.disabled = true; return element; }, enable: function(element) { element = $(element); element.disabled = false; return element; } }; /*--------------------------------------------------------------------------*/ var Field = Form.Element; var $F = Form.Element.Methods.getValue; /*--------------------------------------------------------------------------*/ Form.Element.Serializers = { input: function(element, value) { switch (element.type.toLowerCase()) { case 'checkbox': case 'radio': return Form.Element.Serializers.inputSelector(element, value); default: return Form.Element.Serializers.textarea(element, value); } }, inputSelector: function(element, value) { if (Object.isUndefined(value)) return element.checked ? element.value : null; else element.checked = !!value; }, textarea: function(element, value) { if (Object.isUndefined(value)) return element.value; else element.value = value; }, select: function(element, value) { if (Object.isUndefined(value)) return this[element.type == 'select-one' ? 'selectOne' : 'selectMany'](element); else { var opt, currentValue, single = !Object.isArray(value); for (var i = 0, length = element.length; i < length; i++) { opt = element.options[i]; currentValue = this.optionValue(opt); if (single) { if (currentValue == value) { opt.selected = true; return; } } else opt.selected = value.include(currentValue); } } }, selectOne: function(element) { var index = element.selectedIndex; return index >= 0 ? this.optionValue(element.options[index]) : null; }, selectMany: function(element) { var values, length = element.length; if (!length) return null; for (var i = 0, values = []; i < length; i++) { var opt = element.options[i]; if (opt.selected) values.push(this.optionValue(opt)); } return values; }, optionValue: function(opt) { return Element.extend(opt).hasAttribute('value') ? opt.value : opt.text; } }; /*--------------------------------------------------------------------------*/ Abstract.TimedObserver = Class.create(PeriodicalExecuter, { initialize: function($super, element, frequency, callback) { $super(callback, frequency); this.element = $(element); this.lastValue = this.getValue(); }, execute: function() { var value = this.getValue(); if (Object.isString(this.lastValue) && Object.isString(value) ? this.lastValue != value : String(this.lastValue) != String(value)) { this.callback(this.element, value); this.lastValue = value; } } }); Form.Element.Observer = Class.create(Abstract.TimedObserver, { getValue: function() { return Form.Element.getValue(this.element); } }); Form.Observer = Class.create(Abstract.TimedObserver, { getValue: function() { return Form.serialize(this.element); } }); /*--------------------------------------------------------------------------*/ Abstract.EventObserver = Class.create({ initialize: function(element, callback) { this.element = $(element); this.callback = callback; this.lastValue = this.getValue(); if (this.element.tagName.toLowerCase() == 'form') this.registerFormCallbacks(); else this.registerCallback(this.element); }, onElementEvent: function() { var value = this.getValue(); if (this.lastValue != value) { this.callback(this.element, value); this.lastValue = value; } }, registerFormCallbacks: function() { Form.getElements(this.element).each(this.registerCallback, this); }, registerCallback: function(element) { if (element.type) { switch (element.type.toLowerCase()) { case 'checkbox': case 'radio': Event.observe(element, 'click', this.onElementEvent.bind(this)); break; default: Event.observe(element, 'change', this.onElementEvent.bind(this)); break; } } } }); Form.Element.EventObserver = Class.create(Abstract.EventObserver, { getValue: function() { return Form.Element.getValue(this.element); } }); Form.EventObserver = Class.create(Abstract.EventObserver, { getValue: function() { return Form.serialize(this.element); } }); (function() { var Event = { KEY_BACKSPACE: 8, KEY_TAB: 9, KEY_RETURN: 13, KEY_ESC: 27, KEY_LEFT: 37, KEY_UP: 38, KEY_RIGHT: 39, KEY_DOWN: 40, KEY_DELETE: 46, KEY_HOME: 36, KEY_END: 35, KEY_PAGEUP: 33, KEY_PAGEDOWN: 34, KEY_INSERT: 45, cache: {} }; var docEl = document.documentElement; var MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED = 'onmouseenter' in docEl && 'onmouseleave' in docEl; var _isButton; if (Prototype.Browser.IE) { var buttonMap = { 0: 1, 1: 4, 2: 2 }; _isButton = function(event, code) { return event.button === buttonMap[code]; }; } else if (Prototype.Browser.WebKit) { _isButton = function(event, code) { switch (code) { case 0: return event.which == 1 && !event.metaKey; case 1: return event.which == 1 && event.metaKey; default: return false; } }; } else { _isButton = function(event, code) { return event.which ? (event.which === code + 1) : (event.button === code); }; } function isLeftClick(event) { return _isButton(event, 0) } function isMiddleClick(event) { return _isButton(event, 1) } function isRightClick(event) { return _isButton(event, 2) } function element(event) { event = Event.extend(event); var node = event.target, type = event.type, currentTarget = event.currentTarget; if (currentTarget && currentTarget.tagName) { if (type === 'load' || type === 'error' || (type === 'click' && currentTarget.tagName.toLowerCase() === 'input' && currentTarget.type === 'radio')) node = currentTarget; } if (node.nodeType == Node.TEXT_NODE) node = node.parentNode; return Element.extend(node); } function findElement(event, expression) { var element = Event.element(event); if (!expression) return element; var elements = [element].concat(element.ancestors()); return Selector.findElement(elements, expression, 0); } function pointer(event) { return { x: pointerX(event), y: pointerY(event) }; } function pointerX(event) { var docElement = document.documentElement, body = document.body || { scrollLeft: 0 }; return event.pageX || (event.clientX + (docElement.scrollLeft || body.scrollLeft) - (docElement.clientLeft || 0)); } function pointerY(event) { var docElement = document.documentElement, body = document.body || { scrollTop: 0 }; return event.pageY || (event.clientY + (docElement.scrollTop || body.scrollTop) - (docElement.clientTop || 0)); } function stop(event) { Event.extend(event); event.preventDefault(); event.stopPropagation(); event.stopped = true; } Event.Methods = { isLeftClick: isLeftClick, isMiddleClick: isMiddleClick, isRightClick: isRightClick, element: element, findElement: findElement, pointer: pointer, pointerX: pointerX, pointerY: pointerY, stop: stop }; var methods = Object.keys(Event.Methods).inject({ }, function(m, name) { m[name] = Event.Methods[name].methodize(); return m; }); if (Prototype.Browser.IE) { function _relatedTarget(event) { var element; switch (event.type) { case 'mouseover': element = event.fromElement; break; case 'mouseout': element = event.toElement; break; default: return null; } return Element.extend(element); } Object.extend(methods, { stopPropagation: function() { this.cancelBubble = true }, preventDefault: function() { this.returnValue = false }, inspect: function() { return '[object Event]' } }); Event.extend = function(event, element) { if (!event) return false; if (event._extendedByPrototype) return event; event._extendedByPrototype = Prototype.emptyFunction; var pointer = Event.pointer(event); Object.extend(event, { target: event.srcElement || element, relatedTarget: _relatedTarget(event), pageX: pointer.x, pageY: pointer.y }); return Object.extend(event, methods); }; } else { Event.prototype = window.Event.prototype || document.createEvent('HTMLEvents').__proto__; Object.extend(Event.prototype, methods); Event.extend = Prototype.K; } function _createResponder(element, eventName, handler) { var registry = Element.retrieve(element, 'prototype_event_registry'); if (Object.isUndefined(registry)) { CACHE.push(element); registry = Element.retrieve(element, 'prototype_event_registry', $H()); } var respondersForEvent = registry.get(eventName); if (Object.isUndefined(respondersForEvent)) { respondersForEvent = []; registry.set(eventName, respondersForEvent); } if (respondersForEvent.pluck('handler').include(handler)) return false; var responder; if (eventName.include(":")) { responder = function(event) { if (Object.isUndefined(event.eventName)) return false; if (event.eventName !== eventName) return false; Event.extend(event, element); handler.call(element, event); }; } else { if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED && (eventName === "mouseenter" || eventName === "mouseleave")) { if (eventName === "mouseenter" || eventName === "mouseleave") { responder = function(event) { Event.extend(event, element); var parent = event.relatedTarget; while (parent && parent !== element) { try { parent = parent.parentNode; } catch(e) { parent = element; } } if (parent === element) return; handler.call(element, event); }; } } else { responder = function(event) { Event.extend(event, element); handler.call(element, event); }; } } responder.handler = handler; respondersForEvent.push(responder); return responder; } function _destroyCache() { for (var i = 0, length = CACHE.length; i < length; i++) { Event.stopObserving(CACHE[i]); CACHE[i] = null; } } var CACHE = []; if (Prototype.Browser.IE) window.attachEvent('onunload', _destroyCache); if (Prototype.Browser.WebKit) window.addEventListener('unload', Prototype.emptyFunction, false); var _getDOMEventName = Prototype.K; if (!MOUSEENTER_MOUSELEAVE_EVENTS_SUPPORTED) { _getDOMEventName = function(eventName) { var translations = { mouseenter: "mouseover", mouseleave: "mouseout" }; return eventName in translations ? translations[eventName] : eventName; }; } function observe(element, eventName, handler) { element = $(element); var responder = _createResponder(element, eventName, handler); if (!responder) return element; if (eventName.include(':')) { if (element.addEventListener) element.addEventListener("dataavailable", responder, false); else { element.attachEvent("ondataavailable", responder); element.attachEvent("onfilterchange", responder); } } else { var actualEventName = _getDOMEventName(eventName); if (element.addEventListener) element.addEventListener(actualEventName, responder, false); else element.attachEvent("on" + actualEventName, responder); } return element; } function stopObserving(element, eventName, handler) { element = $(element); var registry = Element.retrieve(element, 'prototype_event_registry'); if (Object.isUndefined(registry)) return element; if (eventName && !handler) { var responders = registry.get(eventName); if (Object.isUndefined(responders)) return element; responders.each( function(r) { Element.stopObserving(element, eventName, r.handler); }); return element; } else if (!eventName) { registry.each( function(pair) { var eventName = pair.key, responders = pair.value; responders.each( function(r) { Element.stopObserving(element, eventName, r.handler); }); }); return element; } var responders = registry.get(eventName); if (!responders) return; var responder = responders.find( function(r) { return r.handler === handler; }); if (!responder) return element; var actualEventName = _getDOMEventName(eventName); if (eventName.include(':')) { if (element.removeEventListener) element.removeEventListener("dataavailable", responder, false); else { element.detachEvent("ondataavailable", responder); element.detachEvent("onfilterchange", responder); } } else { if (element.removeEventListener) element.removeEventListener(actualEventName, responder, false); else element.detachEvent('on' + actualEventName, responder); } registry.set(eventName, responders.without(responder)); return element; } function fire(element, eventName, memo, bubble) { element = $(element); if (Object.isUndefined(bubble)) bubble = true; if (element == document && document.createEvent && !element.dispatchEvent) element = document.documentElement; var event; if (document.createEvent) { event = document.createEvent('HTMLEvents'); event.initEvent('dataavailable', true, true); } else { event = document.createEventObject(); event.eventType = bubble ? 'ondataavailable' : 'onfilterchange'; } event.eventName = eventName; event.memo = memo || { }; if (document.createEvent) element.dispatchEvent(event); else element.fireEvent(event.eventType, event); return Event.extend(event); } Object.extend(Event, Event.Methods); Object.extend(Event, { fire: fire, observe: observe, stopObserving: stopObserving }); Element.addMethods({ fire: fire, observe: observe, stopObserving: stopObserving }); Object.extend(document, { fire: fire.methodize(), observe: observe.methodize(), stopObserving: stopObserving.methodize(), loaded: false }); if (window.Event) Object.extend(window.Event, Event); else window.Event = Event; })(); (function() { /* Support for the DOMContentLoaded event is based on work by Dan Webb, Matthias Miller, Dean Edwards, John Resig, and Diego Perini. */ var timer; function fireContentLoadedEvent() { if (document.loaded) return; if (timer) window.clearTimeout(timer); document.loaded = true; document.fire('dom:loaded'); } function checkReadyState() { if (document.readyState === 'complete') { document.stopObserving('readystatechange', checkReadyState); fireContentLoadedEvent(); } } function pollDoScroll() { try { document.documentElement.doScroll('left'); } catch(e) { timer = pollDoScroll.defer(); return; } fireContentLoadedEvent(); } if (document.addEventListener) { document.addEventListener('DOMContentLoaded', fireContentLoadedEvent, false); } else { document.observe('readystatechange', checkReadyState); if (window == top) timer = pollDoScroll.defer(); } Event.observe(window, 'load', fireContentLoadedEvent); })(); Element.addMethods(); /*------------------------------- DEPRECATED -------------------------------*/ Hash.toQueryString = Object.toQueryString; var Toggle = { display: Element.toggle }; Element.Methods.childOf = Element.Methods.descendantOf; var Insertion = { Before: function(element, content) { return Element.insert(element, {before:content}); }, Top: function(element, content) { return Element.insert(element, {top:content}); }, Bottom: function(element, content) { return Element.insert(element, {bottom:content}); }, After: function(element, content) { return Element.insert(element, {after:content}); } }; var $continue = new Error('"throw $continue" is deprecated, use "return" instead'); var Position = { includeScrollOffsets: false, prepare: function() { this.deltaX = window.pageXOffset || document.documentElement.scrollLeft || document.body.scrollLeft || 0; this.deltaY = window.pageYOffset || document.documentElement.scrollTop || document.body.scrollTop || 0; }, within: function(element, x, y) { if (this.includeScrollOffsets) return this.withinIncludingScrolloffsets(element, x, y); this.xcomp = x; this.ycomp = y; this.offset = Element.cumulativeOffset(element); return (y >= this.offset[1] && y < this.offset[1] + element.offsetHeight && x >= this.offset[0] && x < this.offset[0] + element.offsetWidth); }, withinIncludingScrolloffsets: function(element, x, y) { var offsetcache = Element.cumulativeScrollOffset(element); this.xcomp = x + offsetcache[0] - this.deltaX; this.ycomp = y + offsetcache[1] - this.deltaY; this.offset = Element.cumulativeOffset(element); return (this.ycomp >= this.offset[1] && this.ycomp < this.offset[1] + element.offsetHeight && this.xcomp >= this.offset[0] && this.xcomp < this.offset[0] + element.offsetWidth); }, overlap: function(mode, element) { if (!mode) return 0; if (mode == 'vertical') return ((this.offset[1] + element.offsetHeight) - this.ycomp) / element.offsetHeight; if (mode == 'horizontal') return ((this.offset[0] + element.offsetWidth) - this.xcomp) / element.offsetWidth; }, cumulativeOffset: Element.Methods.cumulativeOffset, positionedOffset: Element.Methods.positionedOffset, absolutize: function(element) { Position.prepare(); return Element.absolutize(element); }, relativize: function(element) { Position.prepare(); return Element.relativize(element); }, realOffset: Element.Methods.cumulativeScrollOffset, offsetParent: Element.Methods.getOffsetParent, page: Element.Methods.viewportOffset, clone: function(source, target, options) { options = options || { }; return Element.clonePosition(target, source, options); } }; /*--------------------------------------------------------------------------*/ if (!document.getElementsByClassName) document.getElementsByClassName = function(instanceMethods){ function iter(name) { return name.blank() ? null : "[contains(concat(' ', @class, ' '), ' " + name + " ')]"; } instanceMethods.getElementsByClassName = Prototype.BrowserFeatures.XPath ? function(element, className) { className = className.toString().strip(); var cond = /\s/.test(className) ? $w(className).map(iter).join('') : iter(className); return cond ? document._getElementsByXPath('.//*' + cond, element) : []; } : function(element, className) { className = className.toString().strip(); var elements = [], classNames = (/\s/.test(className) ? $w(className) : null); if (!classNames && !className) return elements; var nodes = $(element).getElementsByTagName('*'); className = ' ' + className + ' '; for (var i = 0, child, cn; child = nodes[i]; i++) { if (child.className && (cn = ' ' + child.className + ' ') && (cn.include(className) || (classNames && classNames.all(function(name) { return !name.toString().blank() && cn.include(' ' + name + ' '); })))) elements.push(Element.extend(child)); } return elements; }; return function(className, parentElement) { return $(parentElement || document.body).getElementsByClassName(className); }; }(Element.Methods); /*--------------------------------------------------------------------------*/ Element.ClassNames = Class.create(); Element.ClassNames.prototype = { initialize: function(element) { this.element = $(element); }, _each: function(iterator) { this.element.className.split(/\s+/).select(function(name) { return name.length > 0; })._each(iterator); }, set: function(className) { this.element.className = className; }, add: function(classNameToAdd) { if (this.include(classNameToAdd)) return; this.set($A(this).concat(classNameToAdd).join(' ')); }, remove: function(classNameToRemove) { if (!this.include(classNameToRemove)) return; this.set($A(this).without(classNameToRemove).join(' ')); }, toString: function() { return $A(this).join(' '); } }; Object.extend(Element.ClassNames.prototype, Enumerable); /*--------------------------------------------------------------------------*/ email_spec-2.2.0/examples/rails4_root/public/javascripts/dragdrop.js0000644000004100000410000007452013317714226025710 0ustar www-datawww-data// script.aculo.us dragdrop.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009 // Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us) // // 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 = this.element.cumulativeOffset(); 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._originallyAbsolute = (this.element.getStyle('position') == 'absolute'); if (!this._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._originallyAbsolute) Position.relativize(this.element); delete this._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 = this.element.cumulativeOffset(); 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){ element = $(element); var s = Sortable.sortables[element.id]; 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.identify()] = 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 = dropon.cumulativeOffset(); 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')]; };email_spec-2.2.0/examples/rails4_root/public/javascripts/rails.js0000644000004100000410000000736213317714226025220 0ustar www-datawww-datadocument.observe("dom:loaded", function() { var authToken = $$('meta[name=csrf-token]').first().readAttribute('content'), authParam = $$('meta[name=csrf-param]').first().readAttribute('content'), formTemplate = '
\ #{realmethod}\
', realmethodTemplate = ''; function handleRemote(element) { var method, url, params; if (element.tagName.toLowerCase() == 'form') { method = element.readAttribute('method') || 'post'; url = element.readAttribute('action'); params = element.serialize(true); } else { method = element.readAttribute('data-method') || 'get'; url = element.readAttribute('href'); params = {}; } var event = element.fire("ajax:before"); if (event.stopped) return false; new Ajax.Request(url, { method: method, parameters: params, asynchronous: true, evalScripts: true, onLoading: function(request) { element.fire("ajax:loading", {request: request}); }, onLoaded: function(request) { element.fire("ajax:loaded", {request: request}); }, onInteractive: function(request) { element.fire("ajax:interactive", {request: request}); }, onComplete: function(request) { element.fire("ajax:complete", {request: request}); }, onSuccess: function(request) { element.fire("ajax:success", {request: request}); }, onFailure: function(request) { element.fire("ajax:failure", {request: request}); } }); element.fire("ajax:after"); } $(document.body).observe("click", function(event) { var message = event.element().readAttribute('data-confirm'); if (message && !confirm(message)) { event.stop(); return false; } var element = event.findElement("a[data-remote=true]"); if (element) { handleRemote(element); event.stop(); } var element = event.findElement("a[data-method]"); if (element && element.readAttribute('data-remote') != 'true') { var method = element.readAttribute('data-method'), piggyback = method.toLowerCase() != 'post', formHTML = formTemplate.interpolate({ method: 'POST', realmethod: piggyback ? realmethodTemplate.interpolate({ method: method }) : '', action: element.readAttribute('href'), token: authToken, param: authParam }); var form = new Element('div').update(formHTML).down().hide(); this.insert({ bottom: form }); form.submit(); event.stop(); } }); // TODO: I don't think submit bubbles in IE $(document.body).observe("submit", function(event) { var message = event.element().readAttribute('data-confirm'); if (message && !confirm(message)) { event.stop(); return false; } var inputs = event.element().select("input[type=submit][data-disable-with]"); inputs.each(function(input) { input.disabled = true; input.writeAttribute('data-original-value', input.value); input.value = input.readAttribute('data-disable-with'); }); var element = event.findElement("form[data-remote=true]"); if (element) { handleRemote(element); event.stop(); } }); $(document.body).observe("ajax:complete", function(event) { var element = event.element(); if (element.tagName.toLowerCase() == 'form') { var inputs = element.select("input[type=submit][disabled=true][data-disable-with]"); inputs.each(function(input) { input.value = input.readAttribute('data-original-value'); input.writeAttribute('data-original-value', null); input.disabled = false; }); } }); }); email_spec-2.2.0/examples/rails4_root/public/javascripts/application.js0000644000004100000410000000022413317714226026377 0ustar www-datawww-data// Place your application-specific JavaScript functions and classes here // This file is automatically included by javascript_include_tag :defaults email_spec-2.2.0/examples/rails4_root/public/javascripts/controls.js0000644000004100000410000010374313317714226025751 0ustar www-datawww-data// script.aculo.us controls.js v1.8.3, Thu Oct 08 11:23:33 +0200 2009 // Copyright (c) 2005-2009 Thomas Fuchs (http://script.aculo.us, http://mir.aculo.us) // (c) 2005-2009 Ivan Krstic (http://blogs.law.harvard.edu/ivan) // (c) 2005-2009 Jon Tirsen (http://www.tirsen.com) // Contributors: // Richard Livsey // Rahul Bhargava // Rob Wills // // 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/ // Autocompleter.Base handles all the autocompletion functionality // that's independent of the data source for autocompletion. This // includes drawing the autocompletion menu, observing keyboard // and mouse events, and similar. // // Specific autocompleters need to provide, at the very least, // a getUpdatedChoices function that will be invoked every time // the text inside the monitored textbox changes. This method // should get the text for which to provide autocompletion by // invoking this.getToken(), NOT by directly accessing // this.element.value. This is to allow incremental tokenized // autocompletion. Specific auto-completion logic (AJAX, etc) // belongs in getUpdatedChoices. // // Tokenized incremental autocompletion is enabled automatically // when an autocompleter is instantiated with the 'tokens' option // in the options parameter, e.g.: // new Ajax.Autocompleter('id','upd', '/url/', { tokens: ',' }); // will incrementally autocomplete with a comma as the token. // Additionally, ',' in the above example can be replaced with // a token array, e.g. { tokens: [',', '\n'] } which // enables autocompletion on multiple tokens. This is most // useful when one of the tokens is \n (a newline), as it // allows smart autocompletion after linebreaks. if(typeof Effect == 'undefined') throw("controls.js requires including script.aculo.us' effects.js library"); var Autocompleter = { }; Autocompleter.Base = Class.create({ baseInitialize: function(element, update, options) { element = $(element); this.element = element; this.update = $(update); this.hasFocus = false; this.changed = false; this.active = false; this.index = 0; this.entryCount = 0; this.oldElementValue = this.element.value; if(this.setOptions) this.setOptions(options); else this.options = options || { }; this.options.paramName = this.options.paramName || this.element.name; this.options.tokens = this.options.tokens || []; this.options.frequency = this.options.frequency || 0.4; this.options.minChars = this.options.minChars || 1; this.options.onShow = this.options.onShow || function(element, update){ if(!update.style.position || update.style.position=='absolute') { update.style.position = 'absolute'; Position.clone(element, update, { setHeight: false, offsetTop: element.offsetHeight }); } Effect.Appear(update,{duration:0.15}); }; this.options.onHide = this.options.onHide || function(element, update){ new Effect.Fade(update,{duration:0.15}) }; if(typeof(this.options.tokens) == 'string') this.options.tokens = new Array(this.options.tokens); // Force carriage returns as token delimiters anyway if (!this.options.tokens.include('\n')) this.options.tokens.push('\n'); this.observer = null; this.element.setAttribute('autocomplete','off'); Element.hide(this.update); Event.observe(this.element, 'blur', this.onBlur.bindAsEventListener(this)); Event.observe(this.element, 'keydown', this.onKeyPress.bindAsEventListener(this)); }, show: function() { if(Element.getStyle(this.update, 'display')=='none') this.options.onShow(this.element, this.update); if(!this.iefix && (Prototype.Browser.IE) && (Element.getStyle(this.update, 'position')=='absolute')) { new Insertion.After(this.update, ''); this.iefix = $(this.update.id+'_iefix'); } if(this.iefix) setTimeout(this.fixIEOverlapping.bind(this), 50); }, fixIEOverlapping: function() { Position.clone(this.update, this.iefix, {setTop:(!this.update.style.height)}); this.iefix.style.zIndex = 1; this.update.style.zIndex = 2; Element.show(this.iefix); }, hide: function() { this.stopIndicator(); if(Element.getStyle(this.update, 'display')!='none') this.options.onHide(this.element, this.update); if(this.iefix) Element.hide(this.iefix); }, startIndicator: function() { if(this.options.indicator) Element.show(this.options.indicator); }, stopIndicator: function() { if(this.options.indicator) Element.hide(this.options.indicator); }, onKeyPress: function(event) { if(this.active) switch(event.keyCode) { case Event.KEY_TAB: case Event.KEY_RETURN: this.selectEntry(); Event.stop(event); case Event.KEY_ESC: this.hide(); this.active = false; Event.stop(event); return; case Event.KEY_LEFT: case Event.KEY_RIGHT: return; case Event.KEY_UP: this.markPrevious(); this.render(); Event.stop(event); return; case Event.KEY_DOWN: this.markNext(); this.render(); Event.stop(event); return; } else if(event.keyCode==Event.KEY_TAB || event.keyCode==Event.KEY_RETURN || (Prototype.Browser.WebKit > 0 && event.keyCode == 0)) return; this.changed = true; this.hasFocus = true; if(this.observer) clearTimeout(this.observer); this.observer = setTimeout(this.onObserverEvent.bind(this), this.options.frequency*1000); }, activate: function() { this.changed = false; this.hasFocus = true; this.getUpdatedChoices(); }, onHover: function(event) { var element = Event.findElement(event, 'LI'); if(this.index != element.autocompleteIndex) { this.index = element.autocompleteIndex; this.render(); } Event.stop(event); }, onClick: function(event) { var element = Event.findElement(event, 'LI'); this.index = element.autocompleteIndex; this.selectEntry(); this.hide(); }, onBlur: function(event) { // needed to make click events working setTimeout(this.hide.bind(this), 250); this.hasFocus = false; this.active = false; }, render: function() { if(this.entryCount > 0) { for (var i = 0; i < this.entryCount; i++) this.index==i ? Element.addClassName(this.getEntry(i),"selected") : Element.removeClassName(this.getEntry(i),"selected"); if(this.hasFocus) { this.show(); this.active = true; } } else { this.active = false; this.hide(); } }, markPrevious: function() { if(this.index > 0) this.index--; else this.index = this.entryCount-1; this.getEntry(this.index).scrollIntoView(true); }, markNext: function() { if(this.index < this.entryCount-1) this.index++; else this.index = 0; this.getEntry(this.index).scrollIntoView(false); }, getEntry: function(index) { return this.update.firstChild.childNodes[index]; }, getCurrentEntry: function() { return this.getEntry(this.index); }, selectEntry: function() { this.active = false; this.updateElement(this.getCurrentEntry()); }, updateElement: function(selectedElement) { if (this.options.updateElement) { this.options.updateElement(selectedElement); return; } var value = ''; if (this.options.select) { var nodes = $(selectedElement).select('.' + this.options.select) || []; if(nodes.length>0) value = Element.collectTextNodes(nodes[0], this.options.select); } else value = Element.collectTextNodesIgnoreClass(selectedElement, 'informal'); var bounds = this.getTokenBounds(); if (bounds[0] != -1) { var newValue = this.element.value.substr(0, bounds[0]); var whitespace = this.element.value.substr(bounds[0]).match(/^\s+/); if (whitespace) newValue += whitespace[0]; this.element.value = newValue + value + this.element.value.substr(bounds[1]); } else { this.element.value = value; } this.oldElementValue = this.element.value; this.element.focus(); if (this.options.afterUpdateElement) this.options.afterUpdateElement(this.element, selectedElement); }, updateChoices: function(choices) { if(!this.changed && this.hasFocus) { this.update.innerHTML = choices; Element.cleanWhitespace(this.update); Element.cleanWhitespace(this.update.down()); if(this.update.firstChild && this.update.down().childNodes) { this.entryCount = this.update.down().childNodes.length; for (var i = 0; i < this.entryCount; i++) { var entry = this.getEntry(i); entry.autocompleteIndex = i; this.addObservers(entry); } } else { this.entryCount = 0; } this.stopIndicator(); this.index = 0; if(this.entryCount==1 && this.options.autoSelect) { this.selectEntry(); this.hide(); } else { this.render(); } } }, addObservers: function(element) { Event.observe(element, "mouseover", this.onHover.bindAsEventListener(this)); Event.observe(element, "click", this.onClick.bindAsEventListener(this)); }, onObserverEvent: function() { this.changed = false; this.tokenBounds = null; if(this.getToken().length>=this.options.minChars) { this.getUpdatedChoices(); } else { this.active = false; this.hide(); } this.oldElementValue = this.element.value; }, getToken: function() { var bounds = this.getTokenBounds(); return this.element.value.substring(bounds[0], bounds[1]).strip(); }, getTokenBounds: function() { if (null != this.tokenBounds) return this.tokenBounds; var value = this.element.value; if (value.strip().empty()) return [-1, 0]; var diff = arguments.callee.getFirstDifferencePos(value, this.oldElementValue); var offset = (diff == this.oldElementValue.length ? 1 : 0); var prevTokenPos = -1, nextTokenPos = value.length; var tp; for (var index = 0, l = this.options.tokens.length; index < l; ++index) { tp = value.lastIndexOf(this.options.tokens[index], diff + offset - 1); if (tp > prevTokenPos) prevTokenPos = tp; tp = value.indexOf(this.options.tokens[index], diff + offset); if (-1 != tp && tp < nextTokenPos) nextTokenPos = tp; } return (this.tokenBounds = [prevTokenPos + 1, nextTokenPos]); } }); Autocompleter.Base.prototype.getTokenBounds.getFirstDifferencePos = function(newS, oldS) { var boundary = Math.min(newS.length, oldS.length); for (var index = 0; index < boundary; ++index) if (newS[index] != oldS[index]) return index; return boundary; }; Ajax.Autocompleter = Class.create(Autocompleter.Base, { initialize: function(element, update, url, options) { this.baseInitialize(element, update, options); this.options.asynchronous = true; this.options.onComplete = this.onComplete.bind(this); this.options.defaultParams = this.options.parameters || null; this.url = url; }, getUpdatedChoices: function() { this.startIndicator(); var entry = encodeURIComponent(this.options.paramName) + '=' + encodeURIComponent(this.getToken()); this.options.parameters = this.options.callback ? this.options.callback(this.element, entry) : entry; if(this.options.defaultParams) this.options.parameters += '&' + this.options.defaultParams; new Ajax.Request(this.url, this.options); }, onComplete: function(request) { this.updateChoices(request.responseText); } }); // The local array autocompleter. Used when you'd prefer to // inject an array of autocompletion options into the page, rather // than sending out Ajax queries, which can be quite slow sometimes. // // The constructor takes four parameters. The first two are, as usual, // the id of the monitored textbox, and id of the autocompletion menu. // The third is the array you want to autocomplete from, and the fourth // is the options block. // // Extra local autocompletion options: // - choices - How many autocompletion choices to offer // // - partialSearch - If false, the autocompleter will match entered // text only at the beginning of strings in the // autocomplete array. Defaults to true, which will // match text at the beginning of any *word* in the // strings in the autocomplete array. If you want to // search anywhere in the string, additionally set // the option fullSearch to true (default: off). // // - fullSsearch - Search anywhere in autocomplete array strings. // // - partialChars - How many characters to enter before triggering // a partial match (unlike minChars, which defines // how many characters are required to do any match // at all). Defaults to 2. // // - ignoreCase - Whether to ignore case when autocompleting. // Defaults to true. // // It's possible to pass in a custom function as the 'selector' // option, if you prefer to write your own autocompletion logic. // In that case, the other options above will not apply unless // you support them. Autocompleter.Local = Class.create(Autocompleter.Base, { initialize: function(element, update, array, options) { this.baseInitialize(element, update, options); this.options.array = array; }, getUpdatedChoices: function() { this.updateChoices(this.options.selector(this)); }, setOptions: function(options) { this.options = Object.extend({ choices: 10, partialSearch: true, partialChars: 2, ignoreCase: true, fullSearch: false, selector: function(instance) { var ret = []; // Beginning matches var partial = []; // Inside matches var entry = instance.getToken(); var count = 0; for (var i = 0; i < instance.options.array.length && ret.length < instance.options.choices ; i++) { var elem = instance.options.array[i]; var foundPos = instance.options.ignoreCase ? elem.toLowerCase().indexOf(entry.toLowerCase()) : elem.indexOf(entry); while (foundPos != -1) { if (foundPos == 0 && elem.length != entry.length) { ret.push("
  • " + elem.substr(0, entry.length) + "" + elem.substr(entry.length) + "
  • "); break; } else if (entry.length >= instance.options.partialChars && instance.options.partialSearch && foundPos != -1) { if (instance.options.fullSearch || /\s/.test(elem.substr(foundPos-1,1))) { partial.push("
  • " + elem.substr(0, foundPos) + "" + elem.substr(foundPos, entry.length) + "" + elem.substr( foundPos + entry.length) + "
  • "); break; } } foundPos = instance.options.ignoreCase ? elem.toLowerCase().indexOf(entry.toLowerCase(), foundPos + 1) : elem.indexOf(entry, foundPos + 1); } } if (partial.length) ret = ret.concat(partial.slice(0, instance.options.choices - ret.length)); return "
      " + ret.join('') + "
    "; } }, options || { }); } }); // AJAX in-place editor and collection editor // Full rewrite by Christophe Porteneuve (April 2007). // Use this if you notice weird scrolling problems on some browsers, // the DOM might be a bit confused when this gets called so do this // waits 1 ms (with setTimeout) until it does the activation Field.scrollFreeActivate = function(field) { setTimeout(function() { Field.activate(field); }, 1); }; Ajax.InPlaceEditor = Class.create({ initialize: function(element, url, options) { this.url = url; this.element = element = $(element); this.prepareOptions(); this._controls = { }; arguments.callee.dealWithDeprecatedOptions(options); // DEPRECATION LAYER!!! Object.extend(this.options, options || { }); if (!this.options.formId && this.element.id) { this.options.formId = this.element.id + '-inplaceeditor'; if ($(this.options.formId)) this.options.formId = ''; } if (this.options.externalControl) this.options.externalControl = $(this.options.externalControl); if (!this.options.externalControl) this.options.externalControlOnly = false; this._originalBackground = this.element.getStyle('background-color') || 'transparent'; this.element.title = this.options.clickToEditText; this._boundCancelHandler = this.handleFormCancellation.bind(this); this._boundComplete = (this.options.onComplete || Prototype.emptyFunction).bind(this); this._boundFailureHandler = this.handleAJAXFailure.bind(this); this._boundSubmitHandler = this.handleFormSubmission.bind(this); this._boundWrapperHandler = this.wrapUp.bind(this); this.registerListeners(); }, checkForEscapeOrReturn: function(e) { if (!this._editing || e.ctrlKey || e.altKey || e.shiftKey) return; if (Event.KEY_ESC == e.keyCode) this.handleFormCancellation(e); else if (Event.KEY_RETURN == e.keyCode) this.handleFormSubmission(e); }, createControl: function(mode, handler, extraClasses) { var control = this.options[mode + 'Control']; var text = this.options[mode + 'Text']; if ('button' == control) { var btn = document.createElement('input'); btn.type = 'submit'; btn.value = text; btn.className = 'editor_' + mode + '_button'; if ('cancel' == mode) btn.onclick = this._boundCancelHandler; this._form.appendChild(btn); this._controls[mode] = btn; } else if ('link' == control) { var link = document.createElement('a'); link.href = '#'; link.appendChild(document.createTextNode(text)); link.onclick = 'cancel' == mode ? this._boundCancelHandler : this._boundSubmitHandler; link.className = 'editor_' + mode + '_link'; if (extraClasses) link.className += ' ' + extraClasses; this._form.appendChild(link); this._controls[mode] = link; } }, createEditField: function() { var text = (this.options.loadTextURL ? this.options.loadingText : this.getText()); var fld; if (1 >= this.options.rows && !/\r|\n/.test(this.getText())) { fld = document.createElement('input'); fld.type = 'text'; var size = this.options.size || this.options.cols || 0; if (0 < size) fld.size = size; } else { fld = document.createElement('textarea'); fld.rows = (1 >= this.options.rows ? this.options.autoRows : this.options.rows); fld.cols = this.options.cols || 40; } fld.name = this.options.paramName; fld.value = text; // No HTML breaks conversion anymore fld.className = 'editor_field'; if (this.options.submitOnBlur) fld.onblur = this._boundSubmitHandler; this._controls.editor = fld; if (this.options.loadTextURL) this.loadExternalText(); this._form.appendChild(this._controls.editor); }, createForm: function() { var ipe = this; function addText(mode, condition) { var text = ipe.options['text' + mode + 'Controls']; if (!text || condition === false) return; ipe._form.appendChild(document.createTextNode(text)); }; this._form = $(document.createElement('form')); this._form.id = this.options.formId; this._form.addClassName(this.options.formClassName); this._form.onsubmit = this._boundSubmitHandler; this.createEditField(); if ('textarea' == this._controls.editor.tagName.toLowerCase()) this._form.appendChild(document.createElement('br')); if (this.options.onFormCustomization) this.options.onFormCustomization(this, this._form); addText('Before', this.options.okControl || this.options.cancelControl); this.createControl('ok', this._boundSubmitHandler); addText('Between', this.options.okControl && this.options.cancelControl); this.createControl('cancel', this._boundCancelHandler, 'editor_cancel'); addText('After', this.options.okControl || this.options.cancelControl); }, destroy: function() { if (this._oldInnerHTML) this.element.innerHTML = this._oldInnerHTML; this.leaveEditMode(); this.unregisterListeners(); }, enterEditMode: function(e) { if (this._saving || this._editing) return; this._editing = true; this.triggerCallback('onEnterEditMode'); if (this.options.externalControl) this.options.externalControl.hide(); this.element.hide(); this.createForm(); this.element.parentNode.insertBefore(this._form, this.element); if (!this.options.loadTextURL) this.postProcessEditField(); if (e) Event.stop(e); }, enterHover: function(e) { if (this.options.hoverClassName) this.element.addClassName(this.options.hoverClassName); if (this._saving) return; this.triggerCallback('onEnterHover'); }, getText: function() { return this.element.innerHTML.unescapeHTML(); }, handleAJAXFailure: function(transport) { this.triggerCallback('onFailure', transport); if (this._oldInnerHTML) { this.element.innerHTML = this._oldInnerHTML; this._oldInnerHTML = null; } }, handleFormCancellation: function(e) { this.wrapUp(); if (e) Event.stop(e); }, handleFormSubmission: function(e) { var form = this._form; var value = $F(this._controls.editor); this.prepareSubmission(); var params = this.options.callback(form, value) || ''; if (Object.isString(params)) params = params.toQueryParams(); params.editorId = this.element.id; if (this.options.htmlResponse) { var options = Object.extend({ evalScripts: true }, this.options.ajaxOptions); Object.extend(options, { parameters: params, onComplete: this._boundWrapperHandler, onFailure: this._boundFailureHandler }); new Ajax.Updater({ success: this.element }, this.url, options); } else { var options = Object.extend({ method: 'get' }, this.options.ajaxOptions); Object.extend(options, { parameters: params, onComplete: this._boundWrapperHandler, onFailure: this._boundFailureHandler }); new Ajax.Request(this.url, options); } if (e) Event.stop(e); }, leaveEditMode: function() { this.element.removeClassName(this.options.savingClassName); this.removeForm(); this.leaveHover(); this.element.style.backgroundColor = this._originalBackground; this.element.show(); if (this.options.externalControl) this.options.externalControl.show(); this._saving = false; this._editing = false; this._oldInnerHTML = null; this.triggerCallback('onLeaveEditMode'); }, leaveHover: function(e) { if (this.options.hoverClassName) this.element.removeClassName(this.options.hoverClassName); if (this._saving) return; this.triggerCallback('onLeaveHover'); }, loadExternalText: function() { this._form.addClassName(this.options.loadingClassName); this._controls.editor.disabled = true; var options = Object.extend({ method: 'get' }, this.options.ajaxOptions); Object.extend(options, { parameters: 'editorId=' + encodeURIComponent(this.element.id), onComplete: Prototype.emptyFunction, onSuccess: function(transport) { this._form.removeClassName(this.options.loadingClassName); var text = transport.responseText; if (this.options.stripLoadedTextTags) text = text.stripTags(); this._controls.editor.value = text; this._controls.editor.disabled = false; this.postProcessEditField(); }.bind(this), onFailure: this._boundFailureHandler }); new Ajax.Request(this.options.loadTextURL, options); }, postProcessEditField: function() { var fpc = this.options.fieldPostCreation; if (fpc) $(this._controls.editor)['focus' == fpc ? 'focus' : 'activate'](); }, prepareOptions: function() { this.options = Object.clone(Ajax.InPlaceEditor.DefaultOptions); Object.extend(this.options, Ajax.InPlaceEditor.DefaultCallbacks); [this._extraDefaultOptions].flatten().compact().each(function(defs) { Object.extend(this.options, defs); }.bind(this)); }, prepareSubmission: function() { this._saving = true; this.removeForm(); this.leaveHover(); this.showSaving(); }, registerListeners: function() { this._listeners = { }; var listener; $H(Ajax.InPlaceEditor.Listeners).each(function(pair) { listener = this[pair.value].bind(this); this._listeners[pair.key] = listener; if (!this.options.externalControlOnly) this.element.observe(pair.key, listener); if (this.options.externalControl) this.options.externalControl.observe(pair.key, listener); }.bind(this)); }, removeForm: function() { if (!this._form) return; this._form.remove(); this._form = null; this._controls = { }; }, showSaving: function() { this._oldInnerHTML = this.element.innerHTML; this.element.innerHTML = this.options.savingText; this.element.addClassName(this.options.savingClassName); this.element.style.backgroundColor = this._originalBackground; this.element.show(); }, triggerCallback: function(cbName, arg) { if ('function' == typeof this.options[cbName]) { this.options[cbName](this, arg); } }, unregisterListeners: function() { $H(this._listeners).each(function(pair) { if (!this.options.externalControlOnly) this.element.stopObserving(pair.key, pair.value); if (this.options.externalControl) this.options.externalControl.stopObserving(pair.key, pair.value); }.bind(this)); }, wrapUp: function(transport) { this.leaveEditMode(); // Can't use triggerCallback due to backward compatibility: requires // binding + direct element this._boundComplete(transport, this.element); } }); Object.extend(Ajax.InPlaceEditor.prototype, { dispose: Ajax.InPlaceEditor.prototype.destroy }); Ajax.InPlaceCollectionEditor = Class.create(Ajax.InPlaceEditor, { initialize: function($super, element, url, options) { this._extraDefaultOptions = Ajax.InPlaceCollectionEditor.DefaultOptions; $super(element, url, options); }, createEditField: function() { var list = document.createElement('select'); list.name = this.options.paramName; list.size = 1; this._controls.editor = list; this._collection = this.options.collection || []; if (this.options.loadCollectionURL) this.loadCollection(); else this.checkForExternalText(); this._form.appendChild(this._controls.editor); }, loadCollection: function() { this._form.addClassName(this.options.loadingClassName); this.showLoadingText(this.options.loadingCollectionText); var options = Object.extend({ method: 'get' }, this.options.ajaxOptions); Object.extend(options, { parameters: 'editorId=' + encodeURIComponent(this.element.id), onComplete: Prototype.emptyFunction, onSuccess: function(transport) { var js = transport.responseText.strip(); if (!/^\[.*\]$/.test(js)) // TODO: improve sanity check throw('Server returned an invalid collection representation.'); this._collection = eval(js); this.checkForExternalText(); }.bind(this), onFailure: this.onFailure }); new Ajax.Request(this.options.loadCollectionURL, options); }, showLoadingText: function(text) { this._controls.editor.disabled = true; var tempOption = this._controls.editor.firstChild; if (!tempOption) { tempOption = document.createElement('option'); tempOption.value = ''; this._controls.editor.appendChild(tempOption); tempOption.selected = true; } tempOption.update((text || '').stripScripts().stripTags()); }, checkForExternalText: function() { this._text = this.getText(); if (this.options.loadTextURL) this.loadExternalText(); else this.buildOptionList(); }, loadExternalText: function() { this.showLoadingText(this.options.loadingText); var options = Object.extend({ method: 'get' }, this.options.ajaxOptions); Object.extend(options, { parameters: 'editorId=' + encodeURIComponent(this.element.id), onComplete: Prototype.emptyFunction, onSuccess: function(transport) { this._text = transport.responseText.strip(); this.buildOptionList(); }.bind(this), onFailure: this.onFailure }); new Ajax.Request(this.options.loadTextURL, options); }, buildOptionList: function() { this._form.removeClassName(this.options.loadingClassName); this._collection = this._collection.map(function(entry) { return 2 === entry.length ? entry : [entry, entry].flatten(); }); var marker = ('value' in this.options) ? this.options.value : this._text; var textFound = this._collection.any(function(entry) { return entry[0] == marker; }.bind(this)); this._controls.editor.update(''); var option; this._collection.each(function(entry, index) { option = document.createElement('option'); option.value = entry[0]; option.selected = textFound ? entry[0] == marker : 0 == index; option.appendChild(document.createTextNode(entry[1])); this._controls.editor.appendChild(option); }.bind(this)); this._controls.editor.disabled = false; Field.scrollFreeActivate(this._controls.editor); } }); //**** DEPRECATION LAYER FOR InPlace[Collection]Editor! **** //**** This only exists for a while, in order to let **** //**** users adapt to the new API. Read up on the new **** //**** API and convert your code to it ASAP! **** Ajax.InPlaceEditor.prototype.initialize.dealWithDeprecatedOptions = function(options) { if (!options) return; function fallback(name, expr) { if (name in options || expr === undefined) return; options[name] = expr; }; fallback('cancelControl', (options.cancelLink ? 'link' : (options.cancelButton ? 'button' : options.cancelLink == options.cancelButton == false ? false : undefined))); fallback('okControl', (options.okLink ? 'link' : (options.okButton ? 'button' : options.okLink == options.okButton == false ? false : undefined))); fallback('highlightColor', options.highlightcolor); fallback('highlightEndColor', options.highlightendcolor); }; Object.extend(Ajax.InPlaceEditor, { DefaultOptions: { ajaxOptions: { }, autoRows: 3, // Use when multi-line w/ rows == 1 cancelControl: 'link', // 'link'|'button'|false cancelText: 'cancel', clickToEditText: 'Click to edit', externalControl: null, // id|elt externalControlOnly: false, fieldPostCreation: 'activate', // 'activate'|'focus'|false formClassName: 'inplaceeditor-form', formId: null, // id|elt highlightColor: '#ffff99', highlightEndColor: '#ffffff', hoverClassName: '', htmlResponse: true, loadingClassName: 'inplaceeditor-loading', loadingText: 'Loading...', okControl: 'button', // 'link'|'button'|false okText: 'ok', paramName: 'value', rows: 1, // If 1 and multi-line, uses autoRows savingClassName: 'inplaceeditor-saving', savingText: 'Saving...', size: 0, stripLoadedTextTags: false, submitOnBlur: false, textAfterControls: '', textBeforeControls: '', textBetweenControls: '' }, DefaultCallbacks: { callback: function(form) { return Form.serialize(form); }, onComplete: function(transport, element) { // For backward compatibility, this one is bound to the IPE, and passes // the element directly. It was too often customized, so we don't break it. new Effect.Highlight(element, { startcolor: this.options.highlightColor, keepBackgroundImage: true }); }, onEnterEditMode: null, onEnterHover: function(ipe) { ipe.element.style.backgroundColor = ipe.options.highlightColor; if (ipe._effect) ipe._effect.cancel(); }, onFailure: function(transport, ipe) { alert('Error communication with the server: ' + transport.responseText.stripTags()); }, onFormCustomization: null, // Takes the IPE and its generated form, after editor, before controls. onLeaveEditMode: null, onLeaveHover: function(ipe) { ipe._effect = new Effect.Highlight(ipe.element, { startcolor: ipe.options.highlightColor, endcolor: ipe.options.highlightEndColor, restorecolor: ipe._originalBackground, keepBackgroundImage: true }); } }, Listeners: { click: 'enterEditMode', keydown: 'checkForEscapeOrReturn', mouseover: 'enterHover', mouseout: 'leaveHover' } }); Ajax.InPlaceCollectionEditor.DefaultOptions = { loadingCollectionText: 'Loading options...' }; // Delayed observer, like Form.Element.Observer, // but waits for delay after last key input // Ideal for live-search fields Form.Element.DelayedObserver = Class.create({ initialize: function(element, delay, callback) { this.delay = delay || 0.5; this.element = $(element); this.callback = callback; this.timer = null; this.lastValue = $F(this.element); Event.observe(this.element,'keyup',this.delayedListener.bindAsEventListener(this)); }, delayedListener: function(event) { if(this.lastValue == $F(this.element)) return; if(this.timer) clearTimeout(this.timer); this.timer = setTimeout(this.onTimerEvent.bind(this), this.delay * 1000); this.lastValue = $F(this.element); }, onTimerEvent: function() { this.timer = null; this.callback(this.element, $F(this.element)); } });email_spec-2.2.0/examples/rails4_root/public/422.html0000644000004100000410000000130713317714226022405 0ustar www-datawww-data The change you wanted was rejected (422)

    The change you wanted was rejected.

    Maybe you tried to change something you didn't have access to.

    email_spec-2.2.0/examples/rails4_root/public/500.html0000644000004100000410000000133013317714226022376 0ustar www-datawww-data We're sorry, but something went wrong (500)

    We're sorry, but something went wrong.

    We've been notified about this issue and we'll take a look at it shortly.

    email_spec-2.2.0/examples/rails4_root/public/404.html0000644000004100000410000000133013317714226022401 0ustar www-datawww-data The page you were looking for doesn't exist (404)

    The page you were looking for doesn't exist.

    You may have mistyped the address or the page may have moved.

    email_spec-2.2.0/examples/rails4_root/public/images/0000755000004100000410000000000013317714226022454 5ustar www-datawww-dataemail_spec-2.2.0/examples/rails4_root/public/images/rails.png0000644000004100000410000001476613317714226024312 0ustar www-datawww-dataPNG  IHDR2@X${tEXtSoftwareAdobe ImageReadyqe<IDATxڬ[ \eR֮^;Iwga@`gGgDgtqDFqFDqF@NRU]˫|_-Qy^Ǹ.݋0W_6kbf̻ܸ6<4 w5~*r?9m&"M7@vm' {_S)Vi\WG?իjMd@ lDLX鸺W-TU+@EPo\&*Rnn, fDWrX|3M=\EJB[d6A'=tx^$a86̈{, ϱPepN*_W_3o;ޥ(0E:i6eXnhGf"L|S+(+.gФg=Ych=m#V_#}Ǫ|tR D8VՄM~xg!ni%Dy( B,{(Np$3iر$h.@e[a'eJԂyϠ4>H*MHQ(Jgt-֢QI ^d„@s-'- 51{'0 |n4ۉh{V@ܩw"BT =rzqPpBHȃ?ň ]-qpgsPiSӪg`jn)m 御B2L.x!jJP! K/\ ʮRB[09Trӈu. uH$ DDQ+:ݘٻ 3/nލ%Sjm2!&D/[EHwW A-RR!PeuHim"t6lFgЫ-O.1?ƞksX~VtmZJR11Nu&<⽩,Tb,`w WPx-G5 `մ/5pbAtIVJ_]0/DiH=ô#*77-3 VuQ0.pݔ%yw hљW0),2$b6&I/@bj$I(fx' JnO"`<-/LѮ%^ȫͶn2wҗ2}}XսL'Q-,m/ꤋ4#0Q&00NKrsA,Aײ)aIEC(ERK{8Ȭ[y?iI5$%f{}u F 1~;v1l'@F 'IF'm!K7"&]w 15#4Vižn[v 8Ě)>C=LBo~/3% wF4֓ʿ8>bWX@bb@IzP9IvFfQL!2cEP(se4~5RhAŽ90_? cMEteVOaOr B]pȱؓ"Eyx: NJ)bl׋hYuTdԫw=آMgwVPOFΒ25-TD[Z2>]V,xӛIOƅ)Jͺ[)?cn28p#(mk+./phʮQ6?w7HIoSj)1<#-N9O1ͰސkIKr:(ŗ;rR&<93v@w(w:~:TFSޒ" ՊTdT9PIb3JzTQׄBP23ƵW*|@^)Qw?Iq =,<@ B8);50H-=T SA@@f5r[T%#c|Z&w(B)tDQ%vyC(,Ɵ|ʰi&<#u:3EHkzд)ndF>1V2kFGYL KMQlR&TB,igv8]C8Sf#ą4Q'?,= aV9WEXYrr*!cƯ~),=yџ]jlGeE̺5r_2Ԏ}d"a]0M9PZG17nE"Rr\YQ)!|5U(d=^ŗo8+2NU6jB[B5V.]ŲW/^䩬 ;Y"Vi$2ٲ_c(F^Egq{CP/ #K8Y+Q M1>ܞAߏ,gytޕn,zE$V.v.PyLapG9Tn:uiRZ! zI0?Џ1u#$6ɱGMhFdtd|~d\O9Ij**zD؍b)PBҽh-q ql%/{Gz*d7=QS]:RQbUMPᒯ$% du] XefQz$('ИZH#ARXDB ~`0.F|XXK)wFolzyhߚKz>.&n EjU,2' &iw[d[ V)*Qavl QDit[VIQhR@$)y~m|>?cJ+VH'6? 7 i.XH8Fި)dAYUBjE".4w-?l2Y.RjWD@Bج.߆s[H-gASF3Fj]آBP떬_>M%bt ?_rլ -h]r_ž[nȶQ+Gԭ_\Ê Z٦fet(|U('.g VFEN9}Ll4T&nto¨Ӓ X F "_fYzF~y& Gu]$O[v#].@$VA`ⱧTѰZ[2u+/mUC_ TnyѠ |l\ M"G[R$d|:ěFIire"ٵt,+ی1Z11udt*K2 sd; [)xW.z2jTh#DV\NO &e_vU2B^%0FH(/ԘI2>=L]dv UUpk"ijB$,O-0y<}~*T5LErE4B߳XXN:<9>Ed -V*uBLsN**JxRU辖,T( Gu @ůY{u|CJF(OLbnմiKhpFtx8#9FsFڋDTAn1veF^M ^kf.ĆݠVʓǰ3JaY@n&jLl:McӚ…vu?9w!/~#hM ڛ ̴nMA}m W,)(î.N y%$*={P9c DzH>Blu޾K78x->V,'JU \L]l>W*r-hXf~oI Z3f玱>vN3 uZTgg}Վ363:.g /-H+"PKۉSZ4Z_GlXMc7";ҿ (5fMUCOF6 CNft>$S1VaR&4) ٗay]%W A*|gX{Qc>iTX1F M`|![$P4ʊ$#,dɌ(?KTJR۸S%C7jHb浃j+N$,[.@˹_ ?.3ĵH"U$Z^ X02!Kc 8q.NMI6N&3n8exoWfPIJB<pREAdo$*m)e9D 5X[T$LΠ:]C$n#mC[P~Yt*d?\q^WXs!E-2#_mw8;2!vw:DUn$)GiGn3_o EZE3k-EHv.OûzE>"֛}l\/-nرQHԽab*#K׋eIƳd#G et\ ,:MێÜIC}m ٽO?eb%ːٰStB|Aznaz*FlQ/K uu*1wDvE֯SJTK;(4kƣ;v2P9`k{?~_[hʢ^9фǡ;m|]o9<#jz\wD,8V]]%K9er懇0n^FcI>`Ub+kօO1|NO]t+,Ȑl_ˮ6 ĒDbrz^pe7^[aþo確jN+xsNC߅wμ7|za2, omrbZ~,pN>;?Y,z[u◿jq 4aqڶNu6Zid@h!!F9#,#UrOa0=Då ,,,bE#ȮX3ªޏ=a< =&_~ ٵѽacj񫒆LsXuXB (wzEk_QIف*4'ѣSl{.,p۵2`jp^؇nZXPź^]wމ]aQ-oI5O3a] _wb ŭL]$"|sԩȬ= VсLIUbYY搮͢I$tf$2|r;~'GSXkᇦԭF4b4 xo[,04F~<}ۭR%myb׾\mlO.4}tE\7}M)tՉ13xF [-26t䢄&E"9;ٜrq e)K!:bwY }g;Jר)5D$!Kɤ9߫-K$$ hlDUFF J{s2R6rC&&0;@>]/Z3E,k;( 2^09 4.3.3) launchy (~> 2.1) mail (~> 2.7) GEM remote: http://rubygems.org/ specs: addressable (2.5.2) public_suffix (>= 2.0.2, < 4.0) backports (3.11.1) builder (3.2.3) capybara (2.18.0) addressable mini_mime (>= 0.1.3) nokogiri (>= 1.3.3) rack (>= 1.0.0) rack-test (>= 0.5.4) xpath (>= 2.0, < 4.0) cucumber (3.1.0) builder (>= 2.1.2) cucumber-core (~> 3.1.0) cucumber-expressions (~> 5.0.4) cucumber-wire (~> 0.0.1) diff-lcs (~> 1.3) gherkin (~> 5.0) multi_json (>= 1.7.5, < 2.0) multi_test (>= 0.1.2) cucumber-core (3.1.0) backports (>= 3.8.0) cucumber-tag_expressions (~> 1.1.0) gherkin (>= 5.0.0) cucumber-expressions (5.0.13) cucumber-sinatra (0.5.0) templater (>= 1.0.0) cucumber-tag_expressions (1.1.1) cucumber-wire (0.0.1) diff-lcs (1.3) extlib (0.9.16) gherkin (5.0.0) highline (1.7.10) htmlentities (4.3.4) launchy (2.4.3) addressable (~> 2.3) mail (2.7.0) mini_mime (>= 0.1.1) mini_mime (1.0.0) mini_portile2 (2.3.0) multi_json (1.13.1) multi_test (0.1.2) mustermann (1.0.2) nokogiri (1.8.2) mini_portile2 (~> 2.3.0) pony (1.12) mail (>= 2.0) public_suffix (3.0.2) rack (2.0.4) rack-protection (2.0.1) rack rack-test (1.0.0) rack (>= 1.0, < 3) rspec (3.7.0) rspec-core (~> 3.7.0) rspec-expectations (~> 3.7.0) rspec-mocks (~> 3.7.0) rspec-core (3.7.1) rspec-support (~> 3.7.0) rspec-expectations (3.7.0) diff-lcs (>= 1.2.0, < 2.0) rspec-support (~> 3.7.0) rspec-mocks (3.7.0) diff-lcs (>= 1.2.0, < 2.0) rspec-support (~> 3.7.0) rspec-support (3.7.1) sinatra (2.0.1) mustermann (~> 1.0) rack (~> 2.0) rack-protection (= 2.0.1) tilt (~> 2.0) templater (1.0.0) diff-lcs (>= 1.1.2) extlib (>= 0.9.5) highline (>= 1.4.0) tilt (2.0.8) xpath (3.0.0) nokogiri (~> 1.8) PLATFORMS ruby DEPENDENCIES capybara cucumber cucumber-sinatra email_spec! nokogiri pony rack-test rspec sinatra xpath BUNDLED WITH 1.16.1 email_spec-2.2.0/examples/sinatra_root/config.ru0000644000004100000410000000016113317714226022007 0ustar www-datawww-datarequire 'rubygems' require File.join(File.dirname(__FILE__), 'lib/example_sinatra_app.rb') run ExampleSinatraAppemail_spec-2.2.0/examples/sinatra_root/lib/0000755000004100000410000000000013317714226020742 5ustar www-datawww-dataemail_spec-2.2.0/examples/sinatra_root/lib/example_sinatra_app.rb0000644000004100000410000000215413317714226025305 0ustar www-datawww-datarequire 'sinatra/base' require 'pony' class ExampleSinatraApp < Sinatra::Base get '/' do <<-EOHTML
    EOHTML end post '/signup' do user = params[:user] body = <<-EOTEXT Hello #{user['name']}! Copy and paste this URL into your browser to confirm your account! http://www.example.com/confirm This is the text part. EOTEXT html_body = <<-EOHTML Hello #{user['name']}! Click here to confirm your account! This is the HTML part. EOHTML Pony.mail(:from => 'admin@example.com', :to => user['email'], :subject => 'Account confirmation', :body => body, :html_body => html_body ) 'Thanks! Go check your email!' end get '/confirm' do 'Confirm your new account!' end end email_spec-2.2.0/examples/sinatra_root/Gemfile0000644000004100000410000000043213317714226021466 0ustar www-datawww-datasource 'http://rubygems.org' gem 'sinatra' group :test do gem 'cucumber' gem "cucumber-sinatra", :require => false gem 'pony' gem 'capybara' gem 'nokogiri' gem 'xpath' gem 'rspec' gem 'rack-test' gem "email_spec", :path => "../../", :require => "email_spec" end email_spec-2.2.0/examples/sinatra_root/features/0000755000004100000410000000000013317714226022012 5ustar www-datawww-dataemail_spec-2.2.0/examples/sinatra_root/features/example.feature0000644000004100000410000000731313317714226025026 0ustar www-datawww-dataFeature: EmailSpec Example -- Prevent Bots from creating accounts In order to help alleviate email testing in apps As an email-spec contributor I want new users of the library to easily adopt email-spec in their app by following this example In order to prevent bots from setting up new accounts As a site manager I want new users to verify their email address with a confirmation link Background: Given no emails have been sent And I am a real person wanting to sign up for an account And I am on the homepage And I submit my registration information Scenario: First person signup (as myself) with three ways of opening email Then I should receive an email And I should have 1 email # Opening email #1 When I open the email Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body # Opening email #2 When I open the email with subject "Account confirmation" Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body # Opening email #3 When I open the email with subject /Account confirmation/ Then I should see "Account confirmation" in the email subject And I should see "Joe Someone" in the email body And I should see "confirm" in the email body When I follow "Click here to confirm your account!" in the email Then I should see "Confirm your new account" And "foo@bar.com" should have no emails Scenario: Third person signup (emails sent to others) with three ways of opening email Then "foo@bar.com" should have no emails And "example@example.com" should receive an email And "example@example.com" should have 1 email # Opening email #1 When they open the email Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body # Opening email #2 When "example@example.com" opens the email with subject "Account confirmation" Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body # Opening email #3 When "example@example.com" opens the email with subject /Account confirmation/ Then they should see "Account confirmation" in the email subject And they should see "Joe Someone" in the email body And they should see "confirm" in the email body When they follow "Click here to confirm your account!" in the email Then they should see "Confirm your new account" Scenario: Declarative First Person signup Then I should receive an email with a link to a confirmation page Scenario: Declarative First Person signup Then they should receive an email with a link to a confirmation page Scenario: Checking for text in different parts Then I should receive an email And I should have 1 email # Opening email #1 When I open the email Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body # Opening email #2 When I open the email with text "This is the HTML part" Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body # Opening email #3 When I open the email with text /This is the HTML part/ Then I should see "This is the HTML part" in the email html part body And I should see "This is the text part" in the email text part body email_spec-2.2.0/examples/sinatra_root/features/step_definitions/0000755000004100000410000000000013317714226025360 5ustar www-datawww-dataemail_spec-2.2.0/examples/sinatra_root/features/step_definitions/web_steps.rb0000644000004100000410000001410013317714226027674 0ustar www-datawww-data# Taken from the cucumber-rails project. # IMPORTANT: This file is generated by cucumber-sinatra - edit at your own peril. # It is recommended to regenerate this file in the future when you upgrade to a # newer version of cucumber-sinatra. Consider adding your own code to a new file # instead of editing this one. Cucumber will automatically load all features/**/*.rb # files. require 'uri' require 'cgi' require File.expand_path(File.join(File.dirname(__FILE__), "..", "support", "paths")) module WithinHelpers def with_scope(locator) locator ? within(locator) { yield } : yield end end World(WithinHelpers) Given /^(?:|I )am on (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )go to (.+)$/ do |page_name| visit path_to(page_name) end When /^(?:|I )press "([^\"]*)"(?: within "([^\"]*)")?$/ do |button, selector| with_scope(selector) do click_button(button) end end When /^(?:|I )follow "([^\"]*)"(?: within "([^\"]*)")?$/ do |link, selector| with_scope(selector) do click_link(link) end end When /^(?:|I )fill in "([^\"]*)" with "([^\"]*)"(?: within "([^\"]*)")?$/ do |field, value, selector| with_scope(selector) do fill_in(field, :with => value) end end When /^(?:|I )fill in "([^\"]*)" for "([^\"]*)"(?: within "([^\"]*)")?$/ do |value, field, selector| with_scope(selector) do fill_in(field, :with => value) end end # Use this to fill in an entire form with data from a table. Example: # # When I fill in the following: # | Account Number | 5002 | # | Expiry date | 2009-11-01 | # | Note | Nice guy | # | Wants Email? | | # # TODO: Add support for checkbox, select og option # based on naming conventions. # When /^(?:|I )fill in the following(?: within "([^\"]*)")?:$/ do |selector, fields| with_scope(selector) do fields.rows_hash.each do |name, value| When %{I fill in "#{name}" with "#{value}"} end end end When /^(?:|I )select "([^\"]*)" from "([^\"]*)"(?: within "([^\"]*)")?$/ do |value, field, selector| with_scope(selector) do select(value, :from => field) end end When /^(?:|I )check "([^\"]*)"(?: within "([^\"]*)")?$/ do |field, selector| with_scope(selector) do check(field) end end When /^(?:|I )uncheck "([^\"]*)"(?: within "([^\"]*)")?$/ do |field, selector| with_scope(selector) do uncheck(field) end end When /^(?:|I )choose "([^\"]*)"(?: within "([^\"]*)")?$/ do |field, selector| with_scope(selector) do choose(field) end end When /^(?:|I )attach the file "([^\"]*)" to "([^\"]*)"(?: within "([^\"]*)")?$/ do |path, field, selector| with_scope(selector) do attach_file(field, path) end end Then /^(?:|I )should see JSON:$/ do |expected_json| require 'json' expected = JSON.pretty_generate(JSON.parse(expected_json)) actual = JSON.pretty_generate(JSON.parse(response.body)) expect(expected).to eql actual end Then /^(?:|I )should see "([^\"]*)"(?: within "([^\"]*)")?$/ do |text, selector| with_scope(selector) do if self.respond_to? :expect expect(page).to have_content(text) else assert page.has_content?(text) end end end Then /^(?:|I )should see \/([^\/]*)\/(?: within "([^\"]*)")?$/ do |regexp, selector| regexp = Regexp.new(regexp) with_scope(selector) do if self.respond_to? :expect expect(page).to have_xpath('//*', :text => regexp) else assert page.has_xpath?('//*', :text => regexp) end end end Then /^(?:|I )should not see "([^\"]*)"(?: within "([^\"]*)")?$/ do |text, selector| with_scope(selector) do if self.respond_to? :expect expect(page).to have_no_content(text) else assert page.has_no_content?(text) end end end Then /^(?:|I )should not see \/([^\/]*)\/(?: within "([^\"]*)")?$/ do |regexp, selector| regexp = Regexp.new(regexp) with_scope(selector) do if self.respond_to? :expect expect(page).to have_no_xpath('//*', :text => regexp) else assert page.has_no_xpath?('//*', :text => regexp) end end end Then /^the "([^\"]*)" field(?: within "([^\"]*)")? should contain "([^\"]*)"$/ do |field, selector, value| with_scope(selector) do field = find_field(field) field_value = (field.tag_name == 'textarea') ? field.text : field.value if self.respond_to? :expect expect(field_value).to match /#{value}/ else assert_match(/#{value}/, field_value) end end end Then /^the "([^\"]*)" field(?: within "([^\"]*)")? should not contain "([^\"]*)"$/ do |field, selector, value| with_scope(selector) do field = find_field(field) field_value = (field.tag_name == 'textarea') ? field.text : field.value if self.respond_to? :expect expect(field_value).to_not match /#{value}/ else assert_no_match(/#{value}/, field_value) end end end Then /^the "([^\"]*)" checkbox(?: within "([^\"]*)")? should be checked$/ do |label, selector| with_scope(selector) do field_checked = find_field(label)['checked'] if self.respond_to? :expect epect(field_checked).to eql 'checked' else assert_equal 'checked', field_checked end end end Then /^the "([^\"]*)" checkbox(?: within "([^\"]*)")? should not be checked$/ do |label, selector| with_scope(selector) do field_checked = find_field(label)['checked'] if self.respond_to? :expect expect(field_checked).to_not eql 'checked' else assert_not_equal 'checked', field_checked end end end Then /^(?:|I )should be on (.+)$/ do |page_name| current_path = URI.parse(current_url).path if self.respond_to? :expect expect(current_path).to eql path_to(page_name) else assert_equal path_to(page_name), current_path end end Then /^(?:|I )should have the following query string:$/ do |expected_pairs| query = URI.parse(current_url).query actual_params = query ? CGI.parse(query) : {} expected_params = {} expected_pairs.rows_hash.each_pair{|k,v| expected_params[k] = v.split(',')} if self.respond_to? :expect expect(actual_params).to eql expected_params else assert_equal expected_params, actual_params end end Then /^show me the page$/ do save_and_open_page end email_spec-2.2.0/examples/sinatra_root/features/step_definitions/user_steps.rb0000644000004100000410000000163113317714226030102 0ustar www-datawww-dataGiven "I am a real person wanting to sign up for an account" do # no-op.. for documentation purposes only! end When /^I submit my registration information$/ do fill_in "Name", :with => 'Joe Someone' fill_in "Email", :with => 'example@example.com' click_button 'Sign up' end Then /^(?:I|they) should receive an email with a link to a confirmation page$/ do expect(unread_emails_for(current_email_address).size).to eq 1 # this call will store the email and you can access it with current_email open_last_email_for(last_email_address) expect(current_email).to have_subject(/Account confirmation/) expect(current_email).to have_body_text('Joe Someone') click_email_link_matching /confirm/ expect(page).to have_content("Confirm your new account") end # Basically aliases "I should see [text]", but for third person Then /^they should see "([^\"]*)"$/ do |text| step "I should see \"#{text}\"" end email_spec-2.2.0/examples/sinatra_root/features/step_definitions/email_steps.rb0000644000004100000410000001577213317714226030226 0ustar www-datawww-data# Commonly used email steps # # To add your own steps make a custom_email_steps.rb # The provided methods are: # # last_email_address # reset_mailer # open_last_email # visit_in_email # unread_emails_for # mailbox_for # current_email # open_email # read_emails_for # find_email # # General form for email scenarios are: # - clear the email queue (done automatically by email_spec) # - execute steps that sends an email # - check the user received an/no/[0-9] emails # - open the email # - inspect the email contents # - interact with the email (e.g. click links) # # The Cucumber steps below are setup in this order. module EmailHelpers def current_email_address # Replace with your a way to find your current email. e.g @current_user.email # last_email_address will return the last email address used by email spec to find an email. # Note that last_email_address will be reset after each Scenario. last_email_address || "example@example.com" end end World(EmailHelpers) # # Reset the e-mail queue within a scenario. # This is done automatically before each scenario. # Given /^(?:a clear email queue|no emails have been sent)$/ do reset_mailer end # # Check how many emails have been sent/received # Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails?$/ do |address, amount| expect(unread_emails_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should have (an|no|\d+) emails?$/ do |address, amount| expect(mailbox_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject "([^"]*?)"$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(Regexp.escape(subject)) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject \/([^"]*?)\/$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(subject) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive an email with the following body:$/ do |address, expected_body| open_email(address, :with_text => expected_body) end # # Accessing emails # # Opens the most recently received email When /^(?:I|they|"([^"]*?)") opens? the email$/ do |address| open_email(address) end When /^(?:I|they|"([^"]*?)") opens? the email with subject "([^"]*?)"$/ do |address, subject| open_email(address, :with_subject => subject) end When /^(?:I|they|"([^"]*?)") opens? the email with subject \/([^"]*?)\/$/ do |address, subject| open_email(address, :with_subject => Regexp.new(subject)) end When /^(?:I|they|"([^"]*?)") opens? the email with text "([^"]*?)"$/ do |address, text| open_email(address, :with_text => text) end When /^(?:I|they|"([^"]*?)") opens? the email with text \/([^"]*?)\/$/ do |address, text| open_email(address, :with_text => Regexp.new(text)) end # # Inspect the Email Contents # Then /^(?:I|they) should see "([^"]*?)" in the email subject$/ do |text| expect(current_email).to have_subject(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).to have_subject(Regexp.new(text)) end Then /^(?:I|they) should not see "([^"]*?)" in the email subject$/ do |text| expect(current_email).not_to have_subject(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).not_to have_subject(Regexp.new(text)) end Then /^(?:I|they) should see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).to include(text) end Then /^(?:I|they) should not see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to include(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).to match Regexp.new(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to match Regexp.new(text) end Then /^(?:I|they) should see the email delivered from "([^"]*?)"$/ do |text| expect(current_email).to be_delivered_from(text) end Then /^(?:I|they) should see the email reply to "([^"]*?)"$/ do |text| expect(current_email).to have_reply_to(text) end Then /^(?:I|they) should see "([^\"]*)" in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, text) end Then /^(?:I|they) should see \/([^\"]*)\/ in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, Regexp.new(text)) end Then /^I should see it is a multi\-part email$/ do expect(current_email).to be_multipart end Then /^(?:I|they) should see "([^"]*?)" in the email html part body$/ do |text| expect(current_email.html_part.body.to_s).to include(text) end Then /^(?:I|they) should see "([^"]*?)" in the email text part body$/ do |text| expect(current_email.text_part.body.to_s).to include(text) end # # Inspect the Email Attachments # Then /^(?:I|they) should see (an|no|\d+) attachments? with the email$/ do |amount| expect(current_email_attachments.size).to eql parse_email_count(amount) end Then /^there should be (an|no|\d+) attachments? named "([^"]*?)"$/ do |amount, filename| expect(current_email_attachments.select { |a| a.filename == filename }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be named "([^"]*?)"$/ do |index, filename| expect(current_email_attachments[(index.to_i - 1)].filename).to eql filename end Then /^there should be (an|no|\d+) attachments? of type "([^"]*?)"$/ do |amount, content_type| expect(current_email_attachments.select { |a| a.content_type.include?(content_type) }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be of type "([^"]*?)"$/ do |index, content_type| expect(current_email_attachments[(index.to_i - 1)].content_type).to include(content_type) end Then /^all attachments should not be blank$/ do current_email_attachments.each do |attachment| expect(attachment.read.size).to_not eql 0 end end Then /^show me a list of email attachments$/ do EmailSpec::EmailViewer::save_and_open_email_attachments_list(current_email) end # # Interact with Email Contents # When /^(?:I|they|"([^"]*?)") follows? "([^"]*?)" in the email$/ do |address, link| visit_in_email(link, address) end When /^(?:I|they) click the first link in the email$/ do click_first_link_in_email end # # Debugging # These only work with Rails and OSx ATM since EmailViewer uses RAILS_ROOT and OSx's 'open' command. # Patches accepted. ;) # Then /^save and open current email$/ do EmailSpec::EmailViewer::save_and_open_email(current_email) end Then /^save and open all text emails$/ do EmailSpec::EmailViewer::save_and_open_all_text_emails end Then /^save and open all html emails$/ do EmailSpec::EmailViewer::save_and_open_all_html_emails end Then /^save and open all raw emails$/ do EmailSpec::EmailViewer::save_and_open_all_raw_emails end email_spec-2.2.0/examples/sinatra_root/features/errors.feature0000644000004100000410000000301013317714226024675 0ustar www-datawww-dataFeature: Email-spec errors example In order to help alleviate email testing in apps As a email-spec contributor I a newcomer Should be able to easily determine where I have gone wrong These scenarios should fail with helpful messages Background: Given I am on the homepage And no emails have been sent When I fill in "Email" with "example@example.com" And I press "Sign up" Scenario: I fail to open an email with incorrect subject Then I should receive an email When "example@example.com" opens the email with subject "no email" Scenario: I fail to open an email with incorrect subject Then I should receive an email When "example@example.com" opens the email with subject /no email/ Scenario: I fail to open an email with incorrect text Then I should receive an email When "example@example.com" opens the email with text "no email" Scenario: I fail to open an email with incorrect text Then I should receive an email When "example@example.com" opens the email with text /no email/ Scenario: I fail to receive an email with the expected link Then I should receive an email When I open the email When I follow "link that doesn't exist" in the email Scenario: I attempt to operate on an email that is not opened Then I should receive an email When I follow "confirm" in the email Scenario: I attempt to check out an unopened email Then I should see "confirm" in the email body And I should see "Account confirmation" in the email subject email_spec-2.2.0/examples/sinatra_root/features/support/0000755000004100000410000000000013317714226023526 5ustar www-datawww-dataemail_spec-2.2.0/examples/sinatra_root/features/support/paths.rb0000644000004100000410000000120313317714226025166 0ustar www-datawww-data# Taken from the cucumber-rails project. module NavigationHelpers # Maps a name to a path. Used by the # # When /^I go to (.+)$/ do |page_name| # # step definition in web_steps.rb # def path_to(page_name) case page_name when /the home\s?page/ '/' # Add more mappings here. # Here is an example that pulls values out of the Regexp: # # when /^(.*)'s profile page$/i # user_profile_path(User.find_by_login($1)) else raise "Can't find mapping from \"#{page_name}\" to a path.\n" + "Now, go and add a mapping in #{__FILE__}" end end end World(NavigationHelpers) email_spec-2.2.0/examples/sinatra_root/features/support/env.rb0000644000004100000410000000103013317714226024635 0ustar www-datawww-data# Generated by cucumber-sinatra. (Tue Jul 27 10:45:23 -0400 2010) require File.join(File.dirname(__FILE__), '..', '..', 'lib/example_sinatra_app.rb') require 'capybara' require 'capybara/cucumber' require 'rspec/core' #require 'activesupport' require File.expand_path(File.dirname(__FILE__) + '../../../../../lib/email_spec') require 'email_spec/cucumber' ExampleSinatraApp.set(:environment, :test) Capybara.app = ExampleSinatraApp class ExampleSinatraAppWorld include Capybara::DSL end World do ExampleSinatraAppWorld.new end email_spec-2.2.0/email_spec.gemspec0000644000004100000410000002306713317714226017330 0ustar www-datawww-data######################################################### # This file has been automatically generated by gem2tgz # ######################################################### # -*- encoding: utf-8 -*- # stub: email_spec 2.2.0 ruby lib Gem::Specification.new do |s| s.name = "email_spec".freeze s.version = "2.2.0" s.required_rubygems_version = Gem::Requirement.new(">= 0".freeze) if s.respond_to? :required_rubygems_version= s.require_paths = ["lib".freeze] s.authors = ["Ben Mabey".freeze, "Aaron Gibralter".freeze, "Mischa Fierer".freeze] s.date = "2018-04-03" s.description = "Easily test email in RSpec, Cucumber, and MiniTest".freeze s.email = "ben@benmabey.com".freeze s.extra_rdoc_files = ["MIT-LICENSE.txt".freeze, "README.md".freeze] s.files = ["Changelog.md".freeze, "MIT-LICENSE.txt".freeze, "README.md".freeze, "Rakefile".freeze, "examples/rails4_root/Gemfile".freeze, "examples/rails4_root/Gemfile.lock".freeze, "examples/rails4_root/README".freeze, "examples/rails4_root/Rakefile".freeze, "examples/rails4_root/app/controllers/application_controller.rb".freeze, "examples/rails4_root/app/controllers/welcome_controller.rb".freeze, "examples/rails4_root/app/helpers/application_helper.rb".freeze, "examples/rails4_root/app/mailers/user_mailer.rb".freeze, "examples/rails4_root/app/models/user.rb".freeze, "examples/rails4_root/app/views/user_mailer/email_with_attachment.html.erb".freeze, "examples/rails4_root/app/views/user_mailer/newsletter.html.erb".freeze, "examples/rails4_root/app/views/user_mailer/signup.html.erb".freeze, "examples/rails4_root/app/views/user_mailer/signup.text.erb".freeze, "examples/rails4_root/app/views/welcome/attachments.html.erb".freeze, "examples/rails4_root/app/views/welcome/confirm.html.erb".freeze, "examples/rails4_root/app/views/welcome/index.html.erb".freeze, "examples/rails4_root/app/views/welcome/newsletter.html.erb".freeze, "examples/rails4_root/app/views/welcome/signup.html.erb".freeze, "examples/rails4_root/attachments/document.pdf".freeze, "examples/rails4_root/attachments/image.png".freeze, "examples/rails4_root/config.ru".freeze, "examples/rails4_root/config/application.rb".freeze, "examples/rails4_root/config/boot.rb".freeze, "examples/rails4_root/config/cucumber.yml".freeze, "examples/rails4_root/config/database.yml".freeze, "examples/rails4_root/config/environment.rb".freeze, "examples/rails4_root/config/environments/development.rb".freeze, "examples/rails4_root/config/environments/production.rb".freeze, "examples/rails4_root/config/environments/test.rb".freeze, "examples/rails4_root/config/initializers/backtrace_silencers.rb".freeze, "examples/rails4_root/config/initializers/cookie_verification_secret.rb".freeze, "examples/rails4_root/config/initializers/delayed_job.rb".freeze, "examples/rails4_root/config/initializers/inflections.rb".freeze, "examples/rails4_root/config/initializers/mime_types.rb".freeze, "examples/rails4_root/config/initializers/notifier_job.rb".freeze, "examples/rails4_root/config/initializers/rspec_generator.rb".freeze, "examples/rails4_root/config/initializers/session_store.rb".freeze, "examples/rails4_root/config/locales/en.yml".freeze, "examples/rails4_root/config/routes.rb".freeze, "examples/rails4_root/config/secrets.yml".freeze, "examples/rails4_root/db/migrate/20090125013728_create_users.rb".freeze, "examples/rails4_root/db/migrate/20090908054656_create_delayed_jobs.rb".freeze, "examples/rails4_root/db/migrate/20141119224309_add_queue_to_delayed_jobs.rb".freeze, "examples/rails4_root/db/schema.rb".freeze, "examples/rails4_root/db/seeds.rb".freeze, "examples/rails4_root/db/test.sqlite3".freeze, "examples/rails4_root/doc/README_FOR_APP".freeze, "examples/rails4_root/features/attachments.feature".freeze, "examples/rails4_root/features/delayed_job.feature".freeze, "examples/rails4_root/features/errors.feature".freeze, "examples/rails4_root/features/example.feature".freeze, "examples/rails4_root/features/step_definitions/email_steps.rb".freeze, "examples/rails4_root/features/step_definitions/user_steps.rb".freeze, "examples/rails4_root/features/step_definitions/web_steps.rb".freeze, "examples/rails4_root/features/support/env.rb".freeze, "examples/rails4_root/features/support/env_ext.rb".freeze, "examples/rails4_root/features/support/paths.rb".freeze, "examples/rails4_root/lib/notifier_job.rb".freeze, "examples/rails4_root/lib/tasks/cucumber.rake".freeze, "examples/rails4_root/lib/tasks/rspec.rake".freeze, "examples/rails4_root/log/development.log".freeze, "examples/rails4_root/log/test.log".freeze, "examples/rails4_root/public/404.html".freeze, "examples/rails4_root/public/422.html".freeze, "examples/rails4_root/public/500.html".freeze, "examples/rails4_root/public/favicon.ico".freeze, "examples/rails4_root/public/images/rails.png".freeze, "examples/rails4_root/public/javascripts/application.js".freeze, "examples/rails4_root/public/javascripts/controls.js".freeze, "examples/rails4_root/public/javascripts/dragdrop.js".freeze, "examples/rails4_root/public/javascripts/effects.js".freeze, "examples/rails4_root/public/javascripts/prototype.js".freeze, "examples/rails4_root/public/javascripts/rails.js".freeze, "examples/rails4_root/public/robots.txt".freeze, "examples/rails4_root/script/cucumber".freeze, "examples/rails4_root/script/delayed_job".freeze, "examples/rails4_root/script/rails".freeze, "examples/rails4_root/spec/controllers/welcome_controller_spec.rb".freeze, "examples/rails4_root/spec/models/user_mailer_spec.rb".freeze, "examples/rails4_root/spec/spec_helper.rb".freeze, "examples/rails4_root/test/mailers/user_mailer_classic_test.rb".freeze, "examples/rails4_root/test/mailers/user_mailer_spec_test.rb".freeze, "examples/rails4_root/test/test_helper.rb".freeze, "examples/sinatra_root/Gemfile".freeze, "examples/sinatra_root/Gemfile.lock".freeze, "examples/sinatra_root/config.ru".freeze, "examples/sinatra_root/features/errors.feature".freeze, "examples/sinatra_root/features/example.feature".freeze, "examples/sinatra_root/features/step_definitions/email_steps.rb".freeze, "examples/sinatra_root/features/step_definitions/user_steps.rb".freeze, "examples/sinatra_root/features/step_definitions/web_steps.rb".freeze, "examples/sinatra_root/features/support/env.rb".freeze, "examples/sinatra_root/features/support/paths.rb".freeze, "examples/sinatra_root/lib/example_sinatra_app.rb".freeze, "features/rails4_app.feature".freeze, "features/sinatra_app.feature".freeze, "features/step_definitions/app_steps.rb".freeze, "features/support/env.rb".freeze, "lib/email-spec.rb".freeze, "lib/email_spec.rb".freeze, "lib/email_spec/address_converter.rb".freeze, "lib/email_spec/cucumber.rb".freeze, "lib/email_spec/deliveries.rb".freeze, "lib/email_spec/email_viewer.rb".freeze, "lib/email_spec/errors.rb".freeze, "lib/email_spec/helpers.rb".freeze, "lib/email_spec/mail_ext.rb".freeze, "lib/email_spec/matchers.rb".freeze, "lib/email_spec/rspec.rb".freeze, "lib/email_spec/spinach.rb".freeze, "lib/email_spec/test_observer.rb".freeze, "lib/email_spec/version.rb".freeze, "lib/generators/email_spec/steps/steps_generator.rb".freeze, "lib/generators/email_spec/steps/templates/email_steps.rb".freeze, "spec/email_spec/email_viewer_spec.rb".freeze, "spec/email_spec/helpers_spec.rb".freeze, "spec/email_spec/mail_ext_spec.rb".freeze, "spec/email_spec/matchers_spec.rb".freeze, "spec/spec_helper.rb".freeze] s.homepage = "http://github.com/email-spec/email-spec/".freeze s.licenses = ["MIT".freeze] s.rubyforge_project = "email-spec".freeze s.rubygems_version = "2.5.2.1".freeze s.summary = "Easily test email in RSpec, Cucumber or Minitest".freeze if s.respond_to? :specification_version then s.specification_version = 4 if Gem::Version.new(Gem::VERSION) >= Gem::Version.new('1.2.0') then s.add_development_dependency(%q.freeze, ["~> 4.2"]) s.add_development_dependency(%q.freeze, [">= 0"]) s.add_development_dependency(%q.freeze, ["~> 1.3.17"]) s.add_development_dependency(%q.freeze, [">= 0"]) s.add_runtime_dependency(%q.freeze, ["~> 4.3.3"]) s.add_runtime_dependency(%q.freeze, ["~> 2.1"]) s.add_runtime_dependency(%q.freeze, ["~> 2.7"]) s.add_development_dependency(%q.freeze, [">= 0"]) s.add_development_dependency(%q.freeze, [">= 0.8.7"]) s.add_development_dependency(%q.freeze, ["~> 3.1"]) s.add_development_dependency(%q.freeze, [">= 0"]) else s.add_dependency(%q.freeze, ["~> 4.2"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, ["~> 1.3.17"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, ["~> 4.3.3"]) s.add_dependency(%q.freeze, ["~> 2.1"]) s.add_dependency(%q.freeze, ["~> 2.7"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, [">= 0.8.7"]) s.add_dependency(%q.freeze, ["~> 3.1"]) s.add_dependency(%q.freeze, [">= 0"]) end else s.add_dependency(%q.freeze, ["~> 4.2"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, ["~> 1.3.17"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, ["~> 4.3.3"]) s.add_dependency(%q.freeze, ["~> 2.1"]) s.add_dependency(%q.freeze, ["~> 2.7"]) s.add_dependency(%q.freeze, [">= 0"]) s.add_dependency(%q.freeze, [">= 0.8.7"]) s.add_dependency(%q.freeze, ["~> 3.1"]) s.add_dependency(%q.freeze, [">= 0"]) end end email_spec-2.2.0/Rakefile0000644000004100000410000000060513317714226015320 0ustar www-datawww-datarequire 'rubygems' require 'bundler' Bundler::GemHelper.install_tasks begin require 'cucumber/rake/task' Cucumber::Rake::Task.new(:features) rescue LoadError task :features do abort "Cucumber is not available. In order to run features, you must: sudo gem install cucumber" end end require 'rspec/core/rake_task' RSpec::Core::RakeTask.new task :default => [:features, :spec] email_spec-2.2.0/lib/0000755000004100000410000000000013317714226014420 5ustar www-datawww-dataemail_spec-2.2.0/lib/email_spec.rb0000644000004100000410000000114513317714226017047 0ustar www-datawww-dataunless defined?(Pony) or defined?(ActionMailer) Kernel.warn("Neither Pony nor ActionMailer appear to be loaded so email-spec is requiring ActionMailer.") require 'action_mailer' end $LOAD_PATH.unshift(File.expand_path(File.dirname(__FILE__))) unless $LOAD_PATH.include?(File.expand_path(File.dirname(__FILE__))) require 'htmlentities' require 'mail' require 'email_spec/deliveries' require 'email_spec/address_converter' require 'email_spec/email_viewer' require 'email_spec/helpers' require 'email_spec/matchers' require 'email_spec/mail_ext' require 'email_spec/test_observer' require 'email_spec/errors' email_spec-2.2.0/lib/email_spec/0000755000004100000410000000000013317714226016521 5ustar www-datawww-dataemail_spec-2.2.0/lib/email_spec/email_viewer.rb0000644000004100000410000000471013317714226021520 0ustar www-datawww-datamodule EmailSpec class EmailViewer extend Deliveries def self.save_and_open_all_raw_emails filename = tmp_email_filename File.open(filename, "w") do |f| all_emails.each do |m| f.write m.to_s f.write "\n" + '='*80 + "\n" end end open_in_text_editor(filename) end def self.save_and_open_all_html_emails all_emails.each_with_index do |m, index| if m.multipart? && m.parts.detect{ |p| p.content_type.include?('text/html') } filename = tmp_email_filename("-#{index}.html") File.open(filename, "w") do |f| f.write m.parts[1].body end open_in_browser(filename) end end end def self.save_and_open_all_text_emails filename = tmp_email_filename File.open(filename, "w") do |f| all_emails.each do |m| if m.multipart? && text_part = m.parts.detect{ |p| p.content_type.include?('text/plain') } if m.respond_to?(:ordered_each) # Rails 2 / TMail m.ordered_each{|k,v| f.write "#{k}: #{v}\n" } else # Rails 4 / Mail f.write(text_part.header.to_s + "\n") end f.write text_part.body else f.write m.to_s end f.write "\n" + '='*80 + "\n" end end open_in_text_editor(filename) end def self.save_and_open_email(mail) filename = tmp_email_filename File.open(filename, "w") do |f| f.write mail.to_s end open_in_text_editor(filename) end def self.save_and_open_email_attachments_list(mail) filename = tmp_email_filename File.open(filename, "w") do |f| mail.attachments.each_with_index do |attachment, index| info = "#{index + 1}:" info += "\n\tfilename: #{attachment.original_filename}" info += "\n\tcontent type: #{attachment.content_type}" info += "\n\tsize: #{attachment.size}" f.write info + "\n" end end open_in_text_editor(filename) end def self.open_in_text_editor(filename) Launchy.open(URI.parse("file://#{File.expand_path(filename)}"), :application => :editor) end def self.open_in_browser(filename) Launchy.open(URI.parse("file://#{File.expand_path(filename)}")) end def self.tmp_email_filename(extension = '.txt') "#{Rails.root}/tmp/email-#{Time.now.to_i}#{extension}" end end end email_spec-2.2.0/lib/email_spec/matchers.rb0000644000004100000410000003007513317714226020661 0ustar www-datawww-datamodule EmailSpec module Matchers class EmailMatcher def address_array if @email.perform_deliveries Array(yield) else [] end end end class ReplyTo def initialize(email) @expected_reply_to = Mail::ReplyToField.new(email).addrs.first end def description "have reply to as #{@expected_reply_to.address}" end def matches?(email) @email = email @actual_reply_to = (email.reply_to || []).first !@actual_reply_to.nil? && @actual_reply_to == @expected_reply_to.address end def failure_message "expected #{@email.inspect} to reply to #{@expected_reply_to.address.inspect}, but it replied to #{@actual_reply_to.inspect}" end def failure_message_when_negated "expected #{@email.inspect} not to deliver to #{@expected_reply_to.address.inspect}, but it did" end alias negative_failure_message failure_message_when_negated end def reply_to(email) ReplyTo.new(email) end alias :have_reply_to :reply_to class DeliverTo < EmailMatcher def initialize(expected_email_addresses_or_objects_that_respond_to_email) emails = expected_email_addresses_or_objects_that_respond_to_email.map do |email_or_object| email_or_object.kind_of?(String) ? email_or_object : email_or_object.email end @expected_recipients = Mail::ToField.new(emails).addrs.map(&:to_s).sort end def description "be delivered to #{@expected_recipients.inspect}" end def matches?(email) @email = email recipients = email.header[:to] || email.header[:bcc] @actual_recipients = address_array{ recipients && recipients.addrs }.map(&:to_s).sort @actual_recipients == @expected_recipients end def failure_message "expected #{@email.inspect} to deliver to #{@expected_recipients.inspect}, but it delivered to #{@actual_recipients.inspect}" end def failure_message_when_negated "expected #{@email.inspect} not to deliver to #{@expected_recipients.inspect}, but it did" end alias negative_failure_message failure_message_when_negated end def deliver_to(*expected_email_addresses_or_objects_that_respond_to_email) DeliverTo.new(expected_email_addresses_or_objects_that_respond_to_email.flatten) end alias :be_delivered_to :deliver_to class DeliverFrom < EmailMatcher def initialize(email) @expected_sender = Mail::FromField.new(email).addrs.first end def description "be delivered from #{@expected_sender}" end def matches?(email) @email = email @actual_sender = address_array{ email.header[:from].addrs }.first !@actual_sender.nil? && @actual_sender.to_s == @expected_sender.to_s end def failure_message %(expected #{@email.inspect} to deliver from "#{@expected_sender.to_s}", but it delivered from "#{@actual_sender.to_s}") end def failure_message_when_negated %(expected #{@email.inspect} not to deliver from "#{@expected_sender.to_s}", but it did) end alias negative_failure_message failure_message_when_negated end def deliver_from(email) DeliverFrom.new(email) end alias :be_delivered_from :deliver_from class BccTo < EmailMatcher def initialize(expected_email_addresses_or_objects_that_respond_to_email) emails = expected_email_addresses_or_objects_that_respond_to_email.map do |email_or_object| email_or_object.kind_of?(String) ? email_or_object : email_or_object.email end @expected_email_addresses = emails.sort end def description "be bcc'd to #{@expected_email_addresses.inspect}" end def matches?(email) @email = email @actual_recipients = address_array{ email.bcc }.sort @actual_recipients == @expected_email_addresses end def failure_message "expected #{@email.inspect} to bcc to #{@expected_email_addresses.inspect}, but it was bcc'd to #{@actual_recipients.inspect}" end def failure_message_when_negated "expected #{@email.inspect} not to bcc to #{@expected_email_addresses.inspect}, but it did" end alias negative_failure_message failure_message_when_negated end def bcc_to(*expected_email_addresses_or_objects_that_respond_to_email) BccTo.new(expected_email_addresses_or_objects_that_respond_to_email.flatten) end class CcTo < EmailMatcher def initialize(expected_email_addresses_or_objects_that_respond_to_email) emails = expected_email_addresses_or_objects_that_respond_to_email.map do |email_or_object| email_or_object.kind_of?(String) ? email_or_object : email_or_object.email end @expected_email_addresses = emails.sort end def description "be cc'd to #{@expected_email_addresses.inspect}" end def matches?(email) @email = email @actual_recipients = address_array { email.cc }.sort @actual_recipients == @expected_email_addresses end def failure_message "expected #{@email.inspect} to cc to #{@expected_email_addresses.inspect}, but it was cc'd to #{@actual_recipients.inspect}" end def failure_message_when_negated "expected #{@email.inspect} not to cc to #{@expected_email_addresses.inspect}, but it did" end alias negative_failure_message failure_message_when_negated end def cc_to(*expected_email_addresses_or_objects_that_respond_to_email) CcTo.new(expected_email_addresses_or_objects_that_respond_to_email.flatten) end class HaveSubject def initialize(subject) @expected_subject = subject end def description if @expected_subject.is_a?(String) "have subject of #{@expected_subject.inspect}" else "have subject matching #{@expected_subject.inspect}" end end def matches?(email) @given_subject = email.subject if @expected_subject.is_a?(String) @given_subject == @expected_subject else !!(@given_subject =~ @expected_subject) end end def failure_message if @expected_subject.is_a?(String) "expected the subject to be #{@expected_subject.inspect} but was #{@given_subject.inspect}" else "expected the subject to match #{@expected_subject.inspect}, but did not. Actual subject was: #{@given_subject.inspect}" end end def failure_message_when_negated if @expected_subject.is_a?(String) "expected the subject not to be #{@expected_subject.inspect} but was" else "expected the subject not to match #{@expected_subject.inspect} but #{@given_subject.inspect} does match it." end end alias negative_failure_message failure_message_when_negated end def have_subject(subject) HaveSubject.new(subject) end class IncludeEmailWithSubject def initialize(subject) @expected_subject = subject end def description if @expected_subject.is_a?(String) "include email with subject of #{@expected_subject.inspect}" else "include email with subject matching #{@expected_subject.inspect}" end end def matches?(emails) @given_emails = emails if @expected_subject.is_a?(String) @given_emails.map(&:subject).include?(@expected_subject) else !!(@given_emails.any?{ |mail| mail.subject =~ @expected_subject }) end end def failure_message if @expected_subject.is_a?(String) "expected at least one email to have the subject #{@expected_subject.inspect} but none did. Subjects were #{@given_emails.map(&:subject).inspect}" else "expected at least one email to have a subject matching #{@expected_subject.inspect}, but none did. Subjects were #{@given_emails.map(&:subject).inspect}" end end def failure_message_when_negated if @expected_subject.is_a?(String) "expected no email with the subject #{@expected_subject.inspect} but found at least one. Subjects were #{@given_emails.map(&:subject).inspect}" else "expected no email to have a subject matching #{@expected_subject.inspect} but found at least one. Subjects were #{@given_emails.map(&:subject).inspect}" end end alias negative_failure_message failure_message_when_negated end def include_email_with_subject(*emails) IncludeEmailWithSubject.new(emails.flatten.first) end class HaveBodyText def initialize(text) @expected_text = text end def description if @expected_text.is_a?(String) "have body including #{@expected_text.inspect}" else "have body matching #{@expected_text.inspect}" end end def matches?(email) if @expected_text.is_a?(String) @given_text = email.default_part_body.to_s.gsub(/\s+/, " ") @expected_text = @expected_text.gsub(/\s+/, " ") @given_text.include?(@expected_text) else @given_text = email.default_part_body.to_s !!(@given_text =~ @expected_text) end end def failure_message if @expected_text.is_a?(String) "expected the body to contain #{@expected_text.inspect} but was #{@given_text.inspect}" else "expected the body to match #{@expected_text.inspect}, but did not. Actual body was: #{@given_text.inspect}" end end def failure_message_when_negated if @expected_text.is_a?(String) "expected the body not to contain #{@expected_text.inspect} but was #{@given_text.inspect}" else "expected the body not to match #{@expected_text.inspect} but #{@given_text.inspect} does match it." end end alias negative_failure_message failure_message_when_negated end def have_body_text(text) HaveBodyText.new(text) end class HaveHeader def initialize(name, value) @expected_name, @expected_value = name, value end def description if @expected_value.is_a?(String) "have header #{@expected_name}: #{@expected_value}" else "have header #{@expected_name} with value matching #{@expected_value.inspect}" end end def matches?(email) @given_header = email.header if @expected_value.is_a?(String) @given_header[@expected_name].to_s == @expected_value else @given_header[@expected_name].to_s =~ @expected_value end end def failure_message if @expected_value.is_a?(String) "expected the headers to include '#{@expected_name}: #{@expected_value}' but they were #{mail_headers_hash(@given_header).inspect}" else "expected the headers to include '#{@expected_name}' with a value matching #{@expected_value.inspect} but they were #{mail_headers_hash(@given_header).inspect}" end end def failure_message_when_negated if @expected_value.is_a?(String) "expected the headers not to include '#{@expected_name}: #{@expected_value}' but they were #{mail_headers_hash(@given_header).inspect}" else "expected the headers not to include '#{@expected_name}' with a value matching #{@expected_value.inspect} but they were #{mail_headers_hash(@given_header).inspect}" end end alias negative_failure_message failure_message_when_negated def mail_headers_hash(email_headers) email_headers.fields.inject({}) { |hash, field| hash[field.field.class::FIELD_NAME] = field.to_s; hash } end end def have_header(name, value) HaveHeader.new(name, value) end def self.included base if base.respond_to? :register_matcher instance_methods.each do |name| base.register_matcher name, name end end end end end email_spec-2.2.0/lib/email_spec/version.rb0000644000004100000410000000005113317714226020527 0ustar www-datawww-datamodule EmailSpec VERSION = '2.2.0' end email_spec-2.2.0/lib/email_spec/helpers.rb0000644000004100000410000001436213317714226020516 0ustar www-datawww-datarequire 'uri' require 'email_spec/deliveries' module EmailSpec module Helpers include Deliveries A_TAG_BEGIN_REGEX = %r{]*href=['"]?([^'"]*)['"]?[^>]*>\s*(?:(?!).)*?\s*} A_TAG_END_REGEX = %r{\s*(?:(?!).)*?\s*} def visit_in_email(link_text, address = '') if address.nil? || address.empty? email = current_email else email = find_email!(address) end visit(parse_email_for_link(email, link_text)) end def click_email_link_matching(regex, email = current_email) url = links_in_email(email).detect { |link| link =~ regex } raise "No link found matching #{regex.inspect} in #{email.default_part_body}" unless url visit request_uri(url) end def click_first_link_in_email(email = current_email) link = links_in_email(email).first visit request_uri(link) end def open_email(address, opts={}) set_current_email(find_email!(address, opts)) end alias_method :open_email_for, :open_email def open_last_email set_current_email(last_email_sent) end def open_last_email_for(address) set_current_email(mailbox_for(address).last) end def current_email(address=nil) address = convert_address(address) email = address ? email_spec_hash[:current_emails][address] : email_spec_hash[:current_email] exception_class = if defined?(RSpec) RSpec::Expectations::ExpectationNotMetError else StandardError end raise exception_class, "Expected an open email but none was found. Did you forget to call open_email?" unless email email end def current_email_attachments(address=nil) current_email(address).attachments || Array.new end def unread_emails_for(address) read_message_ids = read_emails_for(address).map(&:message_id) mailbox_for(address).reject { |m| read_message_ids.include?(m.message_id) } end def read_emails_for(address) email_spec_hash[:read_emails][convert_address(address)] ||= [] end # Should be able to accept String or Regexp options. def find_email(address, opts={}) address = convert_address(address) if opts[:with_subject] expected_subject = (opts[:with_subject].is_a?(String) ? Regexp.escape(opts[:with_subject]) : opts[:with_subject]) mailbox_for(address).find { |m| m.subject =~ Regexp.new(expected_subject) } elsif opts[:with_text] expected_text = (opts[:with_text].is_a?(String) ? Regexp.escape(opts[:with_text]) : opts[:with_text]) mailbox_for(address).find { |m| m.default_part_body =~ Regexp.new(expected_text) } elsif opts[:from] mailbox_for(address).find { |m| m.from.include? opts[:from] } else mailbox_for(address).first end end def links_in_email(email) links = URI::Parser.new.extract(email.default_part_body.to_s, ['http', 'https']) links.map{|url| HTMLEntities.new.decode(url) }.uniq end private def email_spec_hash @email_spec_hash ||= {:read_emails => {}, :unread_emails => {}, :current_emails => {}, :current_email => nil} end def find_email!(address, opts={}) email = find_email(address, opts) if current_email_address.nil? raise EmailSpec::NoEmailAddressProvided, "No email address has been provided. Make sure current_email_address is returning something." elsif email.nil? error = "#{opts.keys.first.to_s.gsub("_", " ").downcase unless opts.empty?} #{('"' + opts.values.first.to_s + '"') unless opts.empty?}" raise EmailSpec::CouldNotFindEmailError, "Could not find email #{error} in the mailbox for #{current_email_address}. \n Found the following emails:\n\n #{all_emails.to_s}" end email end def set_current_email(email) return unless email [email.to, email.cc, email.bcc].compact.flatten.each do |to| read_emails_for(to) << email email_spec_hash[:current_emails][to] = email end email_spec_hash[:current_email] = email end def parse_email_for_link(email, text_or_regex) matcher = EmailSpec::Matchers::HaveBodyText.new(text_or_regex) if defined?(RSpec) RSpec::Expectations::PositiveExpectationHandler.handle_matcher(email, matcher) else assert_must matcher, email end url = parse_email_for_explicit_link(email, text_or_regex) url ||= parse_email_for_anchor_text_link(email, text_or_regex) raise "No link found matching #{text_or_regex.inspect} in #{email}" unless url url end def request_uri(link) return unless link url = URI::parse(link) url.fragment ? (url.request_uri + "#" + url.fragment) : url.request_uri end # e.g. confirm in http://confirm def parse_email_for_explicit_link(email, regex) regex = /#{Regexp.escape(regex)}/ unless regex.is_a?(Regexp) url = links_in_email(email).detect { |link| link =~ regex } request_uri(url) end # e.g. Click here in Click here def parse_email_for_anchor_text_link(email, link_text) if textify_images(email.default_part_body) =~ %r{#{A_TAG_BEGIN_REGEX}#{link_text}#{A_TAG_END_REGEX}} URI.split($1)[5..-1].compact!.join("?").gsub("&", "&") # sub correct ampersand after rails switches it (http://dev.rubyonrails.org/ticket/4002) else return nil end end def textify_images(email_body) email_body.to_s.gsub(%r{]*alt=['"]?([^'"]*)['"]?[^>]*?/>}) { $1 } end def parse_email_count(amount) case amount when "no" 0 when "an" 1 else amount.to_i end end attr_reader :last_email_address def convert_address(address) @last_email_address = (address || current_email_address) AddressConverter.instance.convert(@last_email_address) end # Overwrite this method to set default email address, for example: # last_email_address || @current_user.email def current_email_address last_email_address end def mailbox_for(address) super(convert_address(address)) # super resides in Deliveries end def email_spec_deprecate(text) puts "" puts "DEPRECATION: #{text.split.join(' ')}" puts "" end end end email_spec-2.2.0/lib/email_spec/errors.rb0000644000004100000410000000020013317714226020352 0ustar www-datawww-datamodule EmailSpec class CouldNotFindEmailError < StandardError end class NoEmailAddressProvided < StandardError end end email_spec-2.2.0/lib/email_spec/test_observer.rb0000644000004100000410000000021413317714226021731 0ustar www-datawww-datamodule EmailSpec class TestObserver def self.delivered_email(message) ActionMailer::Base.deliveries << message end end endemail_spec-2.2.0/lib/email_spec/spinach.rb0000644000004100000410000000167513317714226020504 0ustar www-datawww-data# Require this in your spinach features/support/env.rb file to get access # to the helpers and matchers in your steps. if defined?(ActionMailer) unless [:test, :activerecord, :cache, :file].include?(ActionMailer::Base.delivery_method) ActionMailer::Base.register_observer(EmailSpec::TestObserver) end ActionMailer::Base.perform_deliveries = true Spinach.hooks.before_scenario do # Scenario setup case ActionMailer::Base.delivery_method when :test then ActionMailer::Base.deliveries.clear when :cache then ActionMailer::Base.clear_cache end end end Spinach.hooks.after_scenario do EmailSpec::EmailViewer.save_and_open_all_raw_emails if ENV['SHOW_EMAILS'] EmailSpec::EmailViewer.save_and_open_all_html_emails if ENV['SHOW_HTML_EMAILS'] EmailSpec::EmailViewer.save_and_open_all_text_emails if ENV['SHOW_TEXT_EMAILS'] end class Spinach::FeatureSteps include EmailSpec::Helpers include EmailSpec::Matchers end email_spec-2.2.0/lib/email_spec/deliveries.rb0000644000004100000410000000330413317714226021201 0ustar www-datawww-datamodule EmailSpec module MailerDeliveries def all_emails deliveries end def last_email_sent deliveries.last || raise("No email has been sent!") end def reset_mailer if defined?(ActionMailer) && ActionMailer::Base.delivery_method == :cache mailer.clear_cache else deliveries.clear end end def mailbox_for(address) deliveries.select { |email| email.destinations.include?(address) } end protected def deliveries if ActionMailer::Base.delivery_method == :cache mailer.cached_deliveries else mailer.deliveries end end end module ARMailerDeliveries def all_emails Email.all.map{ |email| parse_to_mail(email) } end def last_email_sent if email = Email.last parse_to_mail(email) else raise("No email has been sent!") end end def reset_mailer Email.delete_all end def mailbox_for(address) Email.all.select { |email| email.destinations.include?(address) }.map{ |email| parse_to_mail(email) } end def parse_to_mail(email) Mail.read(email.mail) end end if defined?(Pony) module ::Pony def self.deliveries @deliveries ||= [] end def self.mail(options) deliveries << build_mail(options) end end end module Deliveries if defined?(Pony) def deliveries; Pony::deliveries ; end include EmailSpec::MailerDeliveries elsif ActionMailer::Base.delivery_method == :activerecord include EmailSpec::ARMailerDeliveries else def mailer; ActionMailer::Base; end include EmailSpec::MailerDeliveries end end end email_spec-2.2.0/lib/email_spec/mail_ext.rb0000644000004100000410000000056313317714226020654 0ustar www-datawww-datamodule EmailSpec::MailExt def default_part @default_part ||= html_part || text_part || self end def default_part_body # Calling to_str as we want the actual String object HTMLEntities.new.decode(default_part.body.to_s.to_str) end def html html_part ? html_part.body.raw_source : nil end end Mail::Message.send(:include, EmailSpec::MailExt) email_spec-2.2.0/lib/email_spec/address_converter.rb0000644000004100000410000000113713317714226022564 0ustar www-datawww-datarequire 'singleton' module EmailSpec class AddressConverter include Singleton attr_accessor :converter # The block provided to conversion should convert to an email # address string or return the input untouched. For example: # # EmailSpec::AddressConverter.instance.conversion do |input| # if input.is_a?(User) # input.email # else # input # end # end # def conversion(&block) self.converter = block end def convert(input) return input unless converter converter.call(input) end end endemail_spec-2.2.0/lib/email_spec/cucumber.rb0000644000004100000410000000151313317714226020653 0ustar www-datawww-data# require this in your env.rb file after you require cucumber/rails/world # Global Setup if defined?(ActionMailer) unless [:test, :activerecord, :cache, :file].include?(ActionMailer::Base.delivery_method) ActionMailer::Base.register_observer(EmailSpec::TestObserver) end ActionMailer::Base.perform_deliveries = true Before do # Scenario setup case ActionMailer::Base.delivery_method when :test then ActionMailer::Base.deliveries.clear when :cache then ActionMailer::Base.clear_cache end end end After do EmailSpec::EmailViewer.save_and_open_all_raw_emails if ENV['SHOW_EMAILS'] EmailSpec::EmailViewer.save_and_open_all_html_emails if ENV['SHOW_HTML_EMAILS'] EmailSpec::EmailViewer.save_and_open_all_text_emails if ENV['SHOW_TEXT_EMAILS'] end World(EmailSpec::Helpers) World(EmailSpec::Matchers) email_spec-2.2.0/lib/email_spec/rspec.rb0000644000004100000410000000024513317714226020163 0ustar www-datawww-dataRSpec.configure do |config| config.include(EmailSpec::Helpers) config.include(EmailSpec::Matchers) config.before(:each) do |group| reset_mailer end end email_spec-2.2.0/lib/generators/0000755000004100000410000000000013317714226016571 5ustar www-datawww-dataemail_spec-2.2.0/lib/generators/email_spec/0000755000004100000410000000000013317714226020672 5ustar www-datawww-dataemail_spec-2.2.0/lib/generators/email_spec/steps/0000755000004100000410000000000013317714226022030 5ustar www-datawww-dataemail_spec-2.2.0/lib/generators/email_spec/steps/templates/0000755000004100000410000000000013317714226024026 5ustar www-datawww-dataemail_spec-2.2.0/lib/generators/email_spec/steps/templates/email_steps.rb0000644000004100000410000001577213317714226026674 0ustar www-datawww-data# Commonly used email steps # # To add your own steps make a custom_email_steps.rb # The provided methods are: # # last_email_address # reset_mailer # open_last_email # visit_in_email # unread_emails_for # mailbox_for # current_email # open_email # read_emails_for # find_email # # General form for email scenarios are: # - clear the email queue (done automatically by email_spec) # - execute steps that sends an email # - check the user received an/no/[0-9] emails # - open the email # - inspect the email contents # - interact with the email (e.g. click links) # # The Cucumber steps below are setup in this order. module EmailHelpers def current_email_address # Replace with your a way to find your current email. e.g @current_user.email # last_email_address will return the last email address used by email spec to find an email. # Note that last_email_address will be reset after each Scenario. last_email_address || "example@example.com" end end World(EmailHelpers) # # Reset the e-mail queue within a scenario. # This is done automatically before each scenario. # Given /^(?:a clear email queue|no emails have been sent)$/ do reset_mailer end # # Check how many emails have been sent/received # Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails?$/ do |address, amount| expect(unread_emails_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should have (an|no|\d+) emails?$/ do |address, amount| expect(mailbox_for(address).size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject "([^"]*?)"$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(Regexp.escape(subject)) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive (an|no|\d+) emails? with subject \/([^"]*?)\/$/ do |address, amount, subject| expect(unread_emails_for(address).select { |m| m.subject =~ Regexp.new(subject) }.size).to eql parse_email_count(amount) end Then /^(?:I|they|"([^"]*?)") should receive an email with the following body:$/ do |address, expected_body| open_email(address, :with_text => expected_body) end # # Accessing emails # # Opens the most recently received email When /^(?:I|they|"([^"]*?)") opens? the email$/ do |address| open_email(address) end When /^(?:I|they|"([^"]*?)") opens? the email with subject "([^"]*?)"$/ do |address, subject| open_email(address, :with_subject => subject) end When /^(?:I|they|"([^"]*?)") opens? the email with subject \/([^"]*?)\/$/ do |address, subject| open_email(address, :with_subject => Regexp.new(subject)) end When /^(?:I|they|"([^"]*?)") opens? the email with text "([^"]*?)"$/ do |address, text| open_email(address, :with_text => text) end When /^(?:I|they|"([^"]*?)") opens? the email with text \/([^"]*?)\/$/ do |address, text| open_email(address, :with_text => Regexp.new(text)) end # # Inspect the Email Contents # Then /^(?:I|they) should see "([^"]*?)" in the email subject$/ do |text| expect(current_email).to have_subject(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).to have_subject(Regexp.new(text)) end Then /^(?:I|they) should not see "([^"]*?)" in the email subject$/ do |text| expect(current_email).not_to have_subject(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email subject$/ do |text| expect(current_email).not_to have_subject(Regexp.new(text)) end Then /^(?:I|they) should see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).to include(text) end Then /^(?:I|they) should not see "([^"]*?)" in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to include(text) end Then /^(?:I|they) should see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).to match Regexp.new(text) end Then /^(?:I|they) should not see \/([^"]*?)\/ in the email body$/ do |text| expect(current_email.default_part_body.to_s).not_to match Regexp.new(text) end Then /^(?:I|they) should see the email delivered from "([^"]*?)"$/ do |text| expect(current_email).to be_delivered_from(text) end Then /^(?:I|they) should see the email reply to "([^"]*?)"$/ do |text| expect(current_email).to have_reply_to(text) end Then /^(?:I|they) should see "([^\"]*)" in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, text) end Then /^(?:I|they) should see \/([^\"]*)\/ in the email "([^"]*?)" header$/ do |text, name| expect(current_email).to have_header(name, Regexp.new(text)) end Then /^I should see it is a multi\-part email$/ do expect(current_email).to be_multipart end Then /^(?:I|they) should see "([^"]*?)" in the email html part body$/ do |text| expect(current_email.html_part.body.to_s).to include(text) end Then /^(?:I|they) should see "([^"]*?)" in the email text part body$/ do |text| expect(current_email.text_part.body.to_s).to include(text) end # # Inspect the Email Attachments # Then /^(?:I|they) should see (an|no|\d+) attachments? with the email$/ do |amount| expect(current_email_attachments.size).to eql parse_email_count(amount) end Then /^there should be (an|no|\d+) attachments? named "([^"]*?)"$/ do |amount, filename| expect(current_email_attachments.select { |a| a.filename == filename }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be named "([^"]*?)"$/ do |index, filename| expect(current_email_attachments[(index.to_i - 1)].filename).to eql filename end Then /^there should be (an|no|\d+) attachments? of type "([^"]*?)"$/ do |amount, content_type| expect(current_email_attachments.select { |a| a.content_type.include?(content_type) }.size).to eql parse_email_count(amount) end Then /^attachment (\d+) should be of type "([^"]*?)"$/ do |index, content_type| expect(current_email_attachments[(index.to_i - 1)].content_type).to include(content_type) end Then /^all attachments should not be blank$/ do current_email_attachments.each do |attachment| expect(attachment.read.size).to_not eql 0 end end Then /^show me a list of email attachments$/ do EmailSpec::EmailViewer::save_and_open_email_attachments_list(current_email) end # # Interact with Email Contents # When /^(?:I|they|"([^"]*?)") follows? "([^"]*?)" in the email$/ do |address, link| visit_in_email(link, address) end When /^(?:I|they) click the first link in the email$/ do click_first_link_in_email end # # Debugging # These only work with Rails and OSx ATM since EmailViewer uses RAILS_ROOT and OSx's 'open' command. # Patches accepted. ;) # Then /^save and open current email$/ do EmailSpec::EmailViewer::save_and_open_email(current_email) end Then /^save and open all text emails$/ do EmailSpec::EmailViewer::save_and_open_all_text_emails end Then /^save and open all html emails$/ do EmailSpec::EmailViewer::save_and_open_all_html_emails end Then /^save and open all raw emails$/ do EmailSpec::EmailViewer::save_and_open_all_raw_emails end email_spec-2.2.0/lib/generators/email_spec/steps/steps_generator.rb0000644000004100000410000000055013317714226025561 0ustar www-datawww-data# This generator adds email steps to the step definitions directory require 'rails/generators' module EmailSpec class StepsGenerator < Rails::Generators::Base def generate copy_file 'email_steps.rb', 'features/step_definitions/email_steps.rb' end def self.source_root File.join(File.dirname(__FILE__), 'templates') end end endemail_spec-2.2.0/lib/email-spec.rb0000644000004100000410000000011213317714226016756 0ustar www-datawww-datarequire File.expand_path(File.join(File.dirname(__FILE__), 'email_spec')) email_spec-2.2.0/features/0000755000004100000410000000000013317714226015470 5ustar www-datawww-dataemail_spec-2.2.0/features/step_definitions/0000755000004100000410000000000013317714226021036 5ustar www-datawww-dataemail_spec-2.2.0/features/step_definitions/app_steps.rb0000644000004100000410000000432513317714226023365 0ustar www-datawww-datarequire 'fileutils' Given /^the (\w+) app is setup with the latest email steps$/ do |app_name| app_dir = File.join(root_dir, 'examples',"#{app_name}_root") email_specs_path = File.join(app_dir, 'features', 'step_definitions', 'email_steps.rb') latest_specs_path = File.join(root_dir, 'lib', 'generators', 'email_spec', 'steps', 'templates','email_steps.rb') FileUtils.rm(email_specs_path) if File.exists?(email_specs_path) FileUtils.cp_r(latest_specs_path, email_specs_path) end Then /^the (\w+) app should have the email steps in place$/ do |app_name| email_specs_path = "#{root_dir}/examples/#{app_name}_root/features/step_definitions/email_steps.rb" expect(File.exists?(email_specs_path)).to be true end Then /^I should see the following summary report:$/ do |expected_report| expect(@output).to include(expected_report) end Given /^the (\w+) app is setup with the latest generators$/ do |app_name| app_dir= File.join(root_dir,'examples',"#{app_name}_root") email_specs_path = File.join(app_dir,'features','step_definitions','email_steps.rb') FileUtils.rm(email_specs_path) if File.exists?(email_specs_path) if app_name == 'rails4' #Testing using the gem #make sure we are listed in the bundle Dir.chdir(app_dir) do output =`bundle list` expect(output).to include('email_spec') end else FileUtils.mkdir_p("#{app_dir}/vendor/plugins/email_spec") FileUtils.cp_r("#{root_dir}/rails_generators","#{app_dir}/vendor/plugins/email_spec/") Dir.chdir(app_dir) do system "ruby ./script/generate email_spec" end end end When /^I run "([^\"]*)" in the (\w+) app$/ do |cmd, app_name| #cmd.gsub!('cucumber', "#{Cucumber::RUBY_BINARY} #{Cucumber::BINARY}") app_path = File.join(root_dir, 'examples', "#{app_name}_root") app_specific_gemfile = File.join(app_path,'Gemfile') Dir.chdir(app_path) do #hack to fight competing bundles (email specs vs rails4_root's if File.exists? app_specific_gemfile orig_gemfile = ENV['BUNDLE_GEMFILE'] ENV['BUNDLE_GEMFILE'] = app_specific_gemfile @output = `#{cmd}` ENV['BUNDLE_GEMFILE'] = orig_gemfile else @output = `#{cmd}` end end end email_spec-2.2.0/features/rails4_app.feature0000644000004100000410000000224013317714226021101 0ustar www-datawww-dataFeature: Email Spec in Rails 4 App In order to prevent me from shipping a defective email_spec gem As a email_spec dev I want to verify that the example rails 4 app runs all of it's features as expected Scenario: generators test Given the rails4 app is setup with the latest generators When I run "bundle exec rails g email_spec:steps" in the rails4 app Then the rails4 app should have the email steps in place Scenario: regression test Given the rails4 app is setup with the latest email steps When I run "bundle exec rake db:migrate RAILS_ENV=test" in the rails4 app And I run "bundle exec cucumber features -q --no-color" in the rails4 app Then I should see the following summary report: """ 15 scenarios (7 failed, 8 passed) 136 steps (7 failed, 1 skipped, 128 passed) """ When I run "bundle exec rake spec RAILS_ENV=test" in the rails4 app Then I should see the following summary report: """ 11 examples, 0 failures """ When I run "bundle exec rake test RAILS_ENV=test" in the rails4 app Then I should see the following summary report: """ 8 runs, 8 assertions, 0 failures, 0 errors, 0 skips """ email_spec-2.2.0/features/support/0000755000004100000410000000000013317714226017204 5ustar www-datawww-dataemail_spec-2.2.0/features/support/env.rb0000644000004100000410000000041313317714226020317 0ustar www-datawww-datarequire 'rubygems' #require 'spec/expectations' class EmailSpecWorld def self.root_dir @root_dir ||= File.join(File.expand_path(File.dirname(__FILE__)), "..", "..") end def root_dir EmailSpecWorld.root_dir end end World do EmailSpecWorld.new end email_spec-2.2.0/features/sinatra_app.feature0000644000004100000410000000102213317714226021341 0ustar www-datawww-dataFeature: Email Spec in Sinatra App In order to prevent me from shipping a defective email_spec gem As a email_spec dev I want to verify that the example sinatra app runs all of it's features as expected Scenario: regression test Given the sinatra app is setup with the latest email steps When I run "bundle exec cucumber features -q --no-color" in the sinatra app Then I should see the following summary report: """ 12 scenarios (7 failed, 5 passed) 110 steps (7 failed, 1 skipped, 102 passed) """ email_spec-2.2.0/MIT-LICENSE.txt0000644000004100000410000000204213317714226016122 0ustar www-datawww-dataCopyright (c) 2008-2009 Ben Mabey 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. email_spec-2.2.0/Changelog.md0000644000004100000410000002604613317714226016073 0ustar www-datawww-data## 2.2.0 2018-04-03 * Support for `spinach` (@pedantic-git) ## 2.1.2 2018-04-03 * Fix compatibility issue with `mail` gem (@sikachu) * Fix cucumber example (@adamdawkins) ## 2.1.1 2017-05-16 * Suppress warnings for URI.extract (@koic) * Dead link removal (@kingdonb) * Better regexp for link searches (@tiagotex) * Relax version requirement for `mail` gem (@thorncp) * Add ruby 2.4 to the build matrix (@eitoball) ## 2.1.0 2016-05-04 * Add Ruby 2.2.3 and 2.3.0 to Travis (@etagwerker) * Fix decoding of email body before matching (@schmierkov) * Invoke Reset Mailer before each scenario (@etagwerker) * Add html method for email (@eliotsykes) * Added actionmailer and droped cucumber-rails and cucumber-sinatra dependency (@mauro-oto) * Fix regex to improve matching of link text in email (@Hirurg103) * Improved README (@etagwerker) ## 2.0.0 2016-01-15 * Upgraded to RSpec 3.1 (Daniel Doubrovkine) * Remove integration of Delayed Job (@jeroenvandijk) * Better coverage for `#reply_to` (@akshayrawat) * Decode encoded links (@deadlyicon) * Add Ruby 2.1.0 to Travis (@salimane) * Upgraded Rspec and Cucumber (@dblock) * Don't assume Rspec is always available (@myobie) * Fix unread emails for cached deliverys (@thickpaddy) * Better link matching with regexp (@ronaldsalas) * Signal non backwards compatible release with version 2.0.0 (@etagwerker) * Upgraded mail dependency (@etagwerker) * Better documentation for new Rspec syntax (@etagwerker) * Updated Travis configuration to solve broken builds (@schmierkov) * Fixed Rspec warnings and removed rails_generators from repo (@mauro-oto) * Added coverage for links surrounded by tags (@mauro-oto) ## 1.6.0 2014-05-27 * New RSpec matcher API support (Morton Jonuschat) * Turnip Doc updates (Joshua Muheim) * Use Mail#destination for picking recipients (Lukasz Strzalkowski) * Emails deliver to nobody when perform_deliveries is set to false (Sean Griffin) ## 1.5.0 2013-07-22 * Upgraded use of Capybara to avoid deprecation warnings. * Upgraded use of RSpec to deal with new expect API (Thomas Drake-Brockman) ### New features * visit_in_email now works for emails that are not the current_email (Ernesto Tagwerker) * emails can be found based on :from field (John Cant) ### Bug fixes * Emails that are not found when clicking a link now raise an exception. (Adam Berlin & Alex Kramer) ## 1.4.0 2012-10-30 * removed jeweler in favor of just using bundler for all gem management. ### New features * MiniTest support! See README for documentation. (Mike Moore) ## 1.3.0 2011-04-07 Many thanks to Derek Hammer and George Ardeleanu who both spontaneously cleaned up various parts of the project and helped get the release out the door. Yay opensoruce! ### New features * Launchy is now used to open up the email viewer. (Matt Burke) * Email Viewer now works in Rails 3 (Woody Peterson) * Email Viewer now works with UTF-8. (Woody Peterson) * Documentation for email matchers. (Derek Hammer) ## 1.2.1 2011-06-20 ### Bugfixes * find_email now allows with_subject and with_text to be either a String or Regexp. Issue #67. (Curtis Miller) ## 1.2.0 2011-06-16 ### New features * [he|she] options for steps. (Dan Croak) * cc_to RSpec matcher. (Arie on github) * Check both html and text parts of multipart emails in Cucumber step. (Charles Barbier) * delivery_method is not set by email_spec to allow for SMTP (e.g. MockSMTP) delivery (Donald Piret) ### Bugfixes * delivered_to matcher now compares both sender name and email address. (Jason Garber) * find_email now matches with_subject and with_text containing text with regex sensitive characters. Issue #31. (Curtis Miller) * current_email_address to work outside of the context of Cucumber. (Szymon Przybył) ## 1.1.1 2010-12-29 * Require "action_mailer" to avoid deprecation warnings. (Ryan Bigg) * Relaxes pessimistic version dependency on RSpec. (GH-41 Dan Pickett) ## 1.1.0 2010-12-20 ### Bugfixes * anchor tags are no longer ignored when clicking links in emails (Anders Törnqist and Nicklas Ramhöj) * set_current_email handles cases where To to blank but may contain CC, or BCC recipients (Murray Steele) * Doc fixes (Florent Guilleux) ## 1.0 2010-07-31 Rails 3.0 Release This release makes email-spec compatible with Rails 3 and RSpec 2.x. IMPORTANT: As of 1.0 email-spec is no longer backwards compatible wit Rails 2.x. I will be maintaining a 0.6 branch on github/rubygems for Rails 2 compatibility if anyone wants new features backported. This was truly a community effort and I appreciate all of time and effort donated by those involved. Specifically: * Maxim Chernyak for the initial conversion work to Rails 3 * Tim Harper for ironing out various matcher bugs * Patrick Muldoon for doing a fantastic job of cleaning up the features and specs. He also fixed the DelayedJob support and worked through all the bundler issues the project was facing. ### Bugfixes * Further checks for older DelayedJob versions to fix compatibility issues. (Andrea Longhi and others) ## 0.6.2 2010-03-21 ### New features * New reply_to matcher. (David Balatero) ## 0.6.1 2010-03-17 ### New features * Ability to click image links via the image alt tag. (Tim Harper) ## 0.6.0 2010-03-05 ### New features * Ability to open a list of attachments on an email (with corresponding steps and EmailViewer support). (Kieran Pilkington) ### Bugfixes * Spelling mistake in steps. (Ben Mabey) * Delayed Job background processor fixes * Narrow delayed job collisions with other apps defining Delayed constant. (Kieran Pilkington) * Fix compatibility with earlier versions of DelayedJob (Michael Baumgarten and Kieran Pilkington) ### Changes * Deprecated steps were removed. (Kieran Pilkington) ## 0.5.0 2010-02-22 ### New features * "should receive emails with subject " step definition (Balint Erdi) * "should receive an email with the following body:" step definition (Ben Mabey) * Debugging steps that tie into EmailViewer: (Ben Mabey) * "save and open current email" * "save and open all text emails" * "save and open all html emails" * "save and open all raw emails" ### Bugfixes * Gracefully handle cases where emails do not have a 'to' value. (Kieran Pilkington) ## 0.4.0 2010-01-07 ### New features * Added support for action_mailer_cache_delivery plugin. (Dan Dofter) You must use the fork at: http://github.com/liangzan/action_mailer_cache_delivery ### Bugfixes * `be_delivered_from` matcher now compares both sender name and email address. (Dan Dofter) ## 0.3.8 2009-12-23 ### Bugfixes * Guard against cc and bcc fields being nil for ActionMailer. (Piotr Sarnacki) ## 0.3.7 2009-12-17 ### New features * Matchers are now Ruby 1.9 compatible. (John Dewey) ## 0.3.6 2009-12-16 ### New features * Cucumber steps for `be_delivered_from`, `have_header` matchers. (Joseph Holsten) * Explicit regular expression steps. i.e. `I should see /foo/ in the email body` (Joseph Holsten) ## 0.3.5 2009-09-30 The Pony Release! ### New features * Support for Pony mailer library. (Rob Holland) ## 0.3.4 2009-09-26 ### Bugfixes * Typo and logic fixes for DeliverFrom matcher. (Yury Kotlyarov) ## 0.3.3 2009-09-18 ### New features * DeliverFrom matcher. i.e. `email.should deliver_from(blah)` or `email.should be_delivered_from(blah)` (Diego Carrion) ## 0.3.2 2009-09-10 ### New features * Support for delayed_job. (Kieran Pilkington) ## 0.3.1 2009-08-19 This release is a general refactoring of the steps and helpers. The example rails app was also updated to use the lateset rails and webrat. It also takes advantages and the newer step definistions including the new third-person forms. Some of the generated steps are being removed in favor of some of the newer ones. The older ones will remain until 0.4.0 but will issue deprecation warnings. Big shoutout to Kieran Pilkington who did the majority of work for this release. ## 0.3.0 2009-08-13 ### New features * New helper #last_email_address which returns the last address used by email-spec for that scenario. (Ben Mabey) * Steps now support third person language. (Kieran P) * "[email] should receive ... " now supports "an" instead of just an integer. (Kieran Pilkington) With these changes, the following is now possible: ``` Then "jack@example.com" should receive an email When they open the email Then they should see "Account has been created" in the subject ``` * Additional default steps (Balint Erdi) * `Then /^I should not receive any emails?$/` * `When %r{^"([^"]*?)" opens? the email$} do |address|` ## 0.2.1 2009-5-29 ### New Features * BCC RSpec matcher. (Josh Nichols) ### Bugfixes * Include BCCed and CCed messsages in the mailbox. (Jakub Kosiński) ## 0.2.0 2009-6-08 No changes. Bumping version for RubyForge release. ## 0.1.4 2009-5-29 ### Bugfixes * Require deliveries in the helpers so it doesn't blow up with RSpec. (Craig Webster) ## 0.1.3 2009-4-15 ### Bugfixes * Fixed regular expressions in genertaed steps. (Ben Mabey) * World semantics changed in cucumber (0.2.3.2), email_spec now uses the new API. (Hector Morales) ## 0.1.2 2009-4-05 ### Bugfixes * Actually added the renamed generators to the gem so people could use it! D'oh! (Ben Mabey) * You can either use `./script generate email_spec` or `rubigen rails email_spec` * Removed Rake tasks from example application to prevent conflicts when used as a plugin. (Ben Mabey) ## 0.1.1 2009-3-26 ### New features * Switched dir structure over to support rubigen. (Dr. Nic) ## 0.1.0 2009-3-25 ### New features * Change Rakefile to run all specs and features, as well as prepare the db (Mischa Fierer) * Allow for array to be passed into deliver_to matcher. (Diego Carrion) * Added matcher for checking if a collision of emails includes an email with a particular subject (Luke Melia, Noah Davis) * Introduced hook to convert objects to email addresses (Luke Melia and Lee Bankewitz) This allows you, in your step matcher, to say something like: maillbox_for(some_user) Use it in your cucumber env.rb like so: ``` EmailSpec::AddressConverter.instance.conversion do |input| if input.is_a?(User) input.email else input end end ``` ### Bugfixes * Revert parse_email_for_link helper method to allow for text links as well as explicit link finding. (Mischa Fierer) * Isolated variances between using email-spec with an ARMailer project. (Luke Melia) ## 0.0.9 2009-2-15 ### New features * have_body_text, have_header matchers (Luke Melia) * EmailViewer - opens all sent emails in a given scenario when the environment variables are set. (Luke Melia) * Added compatibility with using ARMailer in test mode. (Luke Melia) ### Bugfixes * set_current_email now works with multiple addresses in To field. (Brian McManus, Ben Mabey) ## 0.0.7 2009-1-20 ### New features * have_subject matcher (Ben Mabey) ## 0.0.6 2008-12-23 ### New features * Improved RSpec documentation and refactorings. (Ben Mabey) ### Bugfixes * Removed sample app Rake Tasks to have it play nice with environments that use Cucumber as plugin- not gem. (Ben Mabey, Ivor- on github) ## 0.0.5 2008-12-18 * Initial release - see this post for full history and contributors: http://www.benmabey.com/2008/12/18/github-rocks/