specify MIT license + textile filter + fix bug with new tree plugin + fix tests

This commit is contained in:
dinedine 2010-07-06 17:00:02 +02:00
parent 172fe4c752
commit 96e4753ed6
17 changed files with 105 additions and 35 deletions

View File

@ -8,7 +8,7 @@ gem 'liquid'
gem 'bson_ext', '>= 1.0.1' gem 'bson_ext', '>= 1.0.1'
gem 'mongo_ext' gem 'mongo_ext'
gem 'mongoid', '2.0.0.beta6' gem 'mongoid', '2.0.0.beta6'
gem 'mongoid_acts_as_tree', '>= 0.1.2' gem 'mongoid_acts_as_tree', '0.1.5'
gem 'mongo_session_store', '2.0.0.pre' gem 'mongo_session_store', '2.0.0.pre'
gem 'warden' gem 'warden'
gem 'devise', '1.1.rc1' gem 'devise', '1.1.rc1'
@ -22,6 +22,7 @@ gem 'carrierwave-rails3', :require => 'carrierwave'
gem 'actionmailer-with-request', :require => 'actionmailer_with_request' gem 'actionmailer-with-request', :require => 'actionmailer_with_request'
gem 'heroku' gem 'heroku'
gem 'httparty', '0.6.0' gem 'httparty', '0.6.0'
gem 'RedCloth'
# Development environment # Development environment
group :development do group :development do

21
LICENSE Normal file
View File

@ -0,0 +1,21 @@
== MIT License
Copyright (c) 2010, Didier Lafforgue.
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.

View File

@ -1,20 +1,40 @@
h1. Locomotive CMS h1. Locomotive CMS
Locomotive is a simple but powerful CMS based on liquid templates and mongodb database. If we have to give only 4 main features to describe our application, there will be: Locomotive is a simple but powerful CMS based on liquid templates and mongodb database. At my company ("NoCoffee":http://www.nocoffee.fr), we use it for our clients when they request a simple website.
If we have to give only 5 main features to describe our application, there will be:
* managing as many websites as you want with one application instance * managing as many websites as you want with one application instance
* nice looking UI (see http://www.locomotiveapp.org for some screenshots) * nice looking UI (see http://www.locomotiveapp.org for some screenshots)
* flexible content types * flexible content types
* inline editing * playing smoothly with Heroku and MongoHQ
* inline editing (coming soon)
h2. Strategy / Development status h2. Strategy / Development status
We already developed a fully functional prototype in Rails 2.3.2 with both active record / mongomapper and it worked quite well. We are even using it for some client websites. We already developed a fully functional prototype in Rails 2.3.2 with both active record / mongomapper and it worked quite well. We are even using it for some client websites.
Now, our goal is to port our prototype to Rails 3 and migrate from mongomapper to mongoid. Besides, we put a lot of efforts to make it as robust as we can by writing better specs than we wrote for the prototype at first. Now, our goal is to port our prototype to Rails 3 and migrate from mongomapper to mongoid. Besides, we put a lot of efforts to make it as robust as we can by writing better specs than we wrote for the prototype at first.
h2. Gems
Here is a short list of main gems used in the application.
* Rails 3 (beta 3)
* Mongoid
* Liquid
* Devise
* Carrierwave
* Haml
h2. Installation h2. Installation
TODO See the "official website":http://www.locomotiveapp.org
h2. Credits
Many thanks to "Sacha Greif":http://www.sachagreif.com for his great work on the user interface and the LocomotiveApp website front page.
"Rodrigo Alvarez":http://blog.codecaster.es/ for his plugin named Congo which gave us a good starting point and for his availability for (very late) tech discussions.
h2. Contact h2. Contact

View File

@ -48,7 +48,7 @@ module Admin::BaseHelper
end end
def nocoffee_tag def nocoffee_tag
link_to content_tag(:em, 'no') + 'Coffee', 'http://www.nocoffee.fr', :id => 'nocoffee' link_to 'noCoffee', 'http://www.nocoffee.fr', :id => 'nocoffee'
end end
end end

View File

@ -16,7 +16,7 @@ module Models
## callbacks ## ## callbacks ##
before_validate :reset_parent before_validate :reset_parent
before_save { |p| p.parent_id = nil if p.parent_id.blank? } before_save { |p| p.send(:write_attribute, :parent_id, nil) if p.parent_id.blank? }
before_save :change_parent before_save :change_parent
before_create { |p| p.send(:fix_position, false) } before_create { |p| p.send(:fix_position, false) }
before_create :add_to_list_bottom before_create :add_to_list_bottom
@ -60,11 +60,11 @@ module Models
def hacked_fix_position(perform_save = true) def hacked_fix_position(perform_save = true)
if parent.nil? if parent.nil?
self[parent_id_field] = nil self.write_attribute parent_id_field, nil
self[path_field] = [] self[path_field] = []
self[depth_field] = 0 self[depth_field] = 0
else else
self[parent_id_field] = parent._id self.write_attribute parent_id_field, parent._id
self[path_field] = parent[path_field] + [parent._id] self[path_field] = parent[path_field] + [parent._id]
self[depth_field] = parent[depth_field] + 1 self[depth_field] = parent[depth_field] + 1
self.save if perform_save self.save if perform_save

View File

@ -21,9 +21,8 @@ class Layout < LiquidTemplate
self.value.scan(Locomotive::Regexps::CONTENT_FOR).each do |attributes| self.value.scan(Locomotive::Regexps::CONTENT_FOR).each do |attributes|
slug = attributes[0].strip.downcase slug = attributes[0].strip.downcase
name = attributes[1].strip.gsub("\"", '') name = slug.humanize
name = nil if name.empty? name = I18n.t('attributes.defaults.page_parts.name') if slug == 'layout'
name ||= I18n.t('attributes.defaults.page_parts.name') if slug == 'layout'
if part = self.parts.detect { |p| p.slug == slug } if part = self.parts.detect { |p| p.slug == slug }
part.name = name if name.present? part.name = name if name.present?

View File

@ -1,5 +1,2 @@
%p.tright %p.tcenter
= t('.developed_by') = t('.who_is_behind', :development => nocoffee_tag)
= nocoffee_tag
= t('.powered_by')
= link_to 'Ruby on Rails', 'http://www.rubyonrails.com', :id => 'powered-by'

View File

@ -47,8 +47,7 @@ fr:
site: Site site: Site
theme_assets: Fichiers Thème theme_assets: Fichiers Thème
footer: footer:
developed_by: Service développé par who_is_behind: "Service développé par {{development}} et désigné par <a href=\"http://www.sachagreif.com\">Sacha Greif</a>"
powered_by: et propulsé par
form_actions: form_actions:
back: Retour sans sauvegarder back: Retour sans sauvegarder
create: Créer create: Créer
@ -301,7 +300,7 @@ fr:
contents: contents:
index: index:
title: 'Liste "{{type}}"' title: 'Liste des "{{type}}"'
edit: éditer modèle edit: éditer modèle
destroy: supprimer modèle destroy: supprimer modèle
download: télécharger éléments download: télécharger éléments

View File

@ -3,7 +3,6 @@ BOARD:
- refactoring admin crud (pages + layouts + snippets) - refactoring admin crud (pages + layouts + snippets)
- refactor slugify method (use parameterize + create a module) - refactor slugify method (use parameterize + create a module)
- change credits in the admin footer
- change action icons according to the right action [Sacha] - change action icons according to the right action [Sacha]
BACKLOG: BACKLOG:
@ -52,3 +51,6 @@ x localize application in French
x carrierwave x carrierwave
x localize devise emails x localize devise emails
x admin x admin
x change credits in the admin footer
x license
x textile filter

View File

@ -0,0 +1,16 @@
module Locomotive
module Liquid
module Filters
module Text
def textile(input)
RedCloth.new(input).to_html
end
end
::Liquid::Template.register_filter(Text)
end
end
end

View File

@ -7,7 +7,7 @@ module Locomotive
CONTENT_FOR = /\{\{\s*content_for_([a-zA-Z]{1}[a-zA-Z_0-9]*)(\s+.*)?\s*\}\}/ CONTENT_FOR = /\{\{\s*content_for_([a-zA-Z]{1}[a-zA-Z_0-9]*)(\s+.*)?\s*\}\}/
CONTENT_FOR_LAYOUT = /\{\{\s*content_for_layout\s*\}\}/ CONTENT_FOR_LAYOUT = /\{\{\s*content_for_layout\s*/
end end
end end

Binary file not shown.

After

Width:  |  Height:  |  Size: 496 B

View File

@ -134,13 +134,23 @@ body {
/* ___ footer ___ */ /* ___ footer ___ */
#footer { #footer {
padding-top: 10px;
background: transparent url(/images/admin/background/footer.png) no-repeat 0 0; background: transparent url(/images/admin/background/footer.png) no-repeat 0 0;
} }
#footer p { #footer p {
padding: 15px 8px 0 0; padding: 15px 8px 0 0;
font-size: 0.6em; font-size: 0.8em;
color: #444; color: #E6E6E6;
}
#footer p a {
color: #1F82BC;
text-decoration: none;
}
#footer p a:hover {
text-decoration: underline;
} }
/* ___ Alignements ___ */ /* ___ Alignements ___ */
@ -166,10 +176,4 @@ body {
/* ___ NoCoffee / Rails Tags ___ */ /* ___ NoCoffee / Rails Tags ___ */
a#nocoffee { text-decoration: none; font-size: 1.1em; line-height: 20px; color: #505b64; padding-right: 20px; margin: 0 4px; background: transparent url(../../images/admin/nocoffee.gif) no-repeat right 0px; } a#nocoffee { color: #b0b4c0 !important; text-decoration: none; line-height: 20px; padding-right: 20px; margin: 0 4px; background: transparent url(../../images/admin/nocoffee.png) no-repeat right 0px; }
a#nocoffee em { color: #ef3f44; font-weight: normal; font-style: normal; }
a#powered-by { text-decoration: none; font-size: 1.1em; color: #666; }
a#powered-by:hover{ text-decoration: underline; color: #aaa; }

View File

@ -44,7 +44,7 @@ Factory.define :layout do |l|
<title>My website</title> <title>My website</title>
</head> </head>
<body> <body>
<div id="sidebar">\{\{ content_for_sidebar "Left Sidebar"\}\}</div> <div id="sidebar">\{\{ content_for_left_sidebar \}\}</div>
<div id="main">\{\{ content_for_layout \}\}</div> <div id="main">\{\{ content_for_layout \}\}</div>
</body> </body>
</html>} </html>}

View File

@ -0,0 +1,11 @@
require 'spec_helper'
describe Locomotive::Liquid::Filters::Text do
include Locomotive::Liquid::Filters::Text
it 'transforms a textile input into HTML' do
textile('This is *my* text.').should == "<p>This is <strong>my</strong> text.</p>"
end
end

View File

@ -27,8 +27,8 @@ describe Layout do
@layout.parts.first.name.should == 'Body' @layout.parts.first.name.should == 'Body'
@layout.parts.first.slug.should == 'layout' @layout.parts.first.slug.should == 'layout'
@layout.parts.last.name.should == 'Left Sidebar' @layout.parts.last.name.should == 'Left sidebar'
@layout.parts.last.slug.should == 'sidebar' @layout.parts.last.slug.should == 'left_sidebar'
end end
it 'should not add parts to pages if layout does not change' do it 'should not add parts to pages if layout does not change' do

View File

@ -293,7 +293,7 @@ describe Page do
before(:each) do before(:each) do
@page = Factory.build(:page, :site => nil) @page = Factory.build(:page, :site => nil)
@page.parts.build :slug => 'layout', :value => 'Hello world !' @page.parts.build :slug => 'layout', :value => 'Hello world !'
@page.parts.build :slug => 'sidebar', :value => 'A sidebar...' @page.parts.build :slug => 'left_sidebar', :value => 'A sidebar...'
@page.send(:store_template) @page.send(:store_template)
@layout = Factory.build(:layout, :site => nil) @layout = Factory.build(:layout, :site => nil)
@layout.send(:store_template) @layout.send(:store_template)