mercury_engine 0.0.1
Sign up to get free protection for your applications and to get access to all the features.
- data/MIT-LICENSE +20 -0
- data/README.md +115 -0
- data/Rakefile +29 -0
- data/app/assets/javascripts/mercury_engine/mercury.js.coffee +533 -0
- data/app/assets/stylesheets/mercury_engine/mercury.css +27 -0
- data/app/controllers/pages_controller.rb +18 -0
- data/app/helpers/pages_helper.rb +78 -0
- data/app/models/page.rb +42 -0
- data/app/views/layouts/mercury.html.haml +19 -0
- data/app/views/mercury_engine/shared/_edit_link.html.haml +3 -0
- data/config/routes.rb +6 -0
- data/db/migrate/20130318080000_create_versions.rb +13 -0
- data/db/migrate/20130318125100_create_pages.rb +11 -0
- data/lib/generators/mercury_engine/install_generator.rb +23 -0
- data/lib/mercury/authentication.rb +7 -0
- data/lib/mercury_engine.rb +7 -0
- data/lib/mercury_engine/engine.rb +5 -0
- data/lib/mercury_engine/version.rb +3 -0
- data/lib/tasks/mercury_engine_tasks.rake +4 -0
- metadata +211 -0
data/MIT-LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright 2013 Josh Nussbaum
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,115 @@
|
|
1
|
+
# Mercury Engine
|
2
|
+
|
3
|
+
A basic content editing system based on [Mercury Editor](http://jejacks0n.github.com/mercury/).
|
4
|
+
|
5
|
+
[![Travis CI Status](https://travis-ci.org/DynamoMTL/mercury-engine.png)](https://travis-ci.org/DynamoMTL/mercury-engine)
|
6
|
+
|
7
|
+
[![Code Climate](https://codeclimate.com/github/DynamoMTL/mercury-engine.png)](https://codeclimate.com/github/DynamoMTL/mercury-engine)
|
8
|
+
|
9
|
+
Whats that?
|
10
|
+
--------
|
11
|
+
|
12
|
+
mercury-engine is a lightweight content editing system that makes it easy to add live in-place editing to your site's static pages.
|
13
|
+
|
14
|
+
It is designed to fit a workflow where static pages are hand crafted using regular Rails templates/asset pipeline.
|
15
|
+
|
16
|
+
It does not impose any structure on the host application and has minimal requirements.
|
17
|
+
|
18
|
+
### How it works
|
19
|
+
|
20
|
+
Say you have a HAML template like this:
|
21
|
+
```haml
|
22
|
+
-# in app/views/pages/index.html.haml
|
23
|
+
%section#main
|
24
|
+
%h1 My Cyber Web Page
|
25
|
+
#details.highlight
|
26
|
+
:markdown
|
27
|
+
All your web belong to us
|
28
|
+
```
|
29
|
+
|
30
|
+
To make it editable, simply annotate it with the helper:
|
31
|
+
```haml
|
32
|
+
%section#main
|
33
|
+
= editable(:title, :h1) do
|
34
|
+
My Editable Cyber Web Page
|
35
|
+
|
36
|
+
= editable(:details, class: 'highlight') do
|
37
|
+
:markdown
|
38
|
+
All your web belong to us
|
39
|
+
```
|
40
|
+
|
41
|
+
## `editable` helper
|
42
|
+
|
43
|
+
`#editable(id, tag_or_options, options)`
|
44
|
+
|
45
|
+
- The `id` is required and should be unique per page
|
46
|
+
- The tag is optional and defaults to `:div`
|
47
|
+
- The last parameter is an optional hash of html attributes
|
48
|
+
|
49
|
+
## How we decide which content to use
|
50
|
+
|
51
|
+
If the user has edited a given section, content will be served from the database.
|
52
|
+
When there are no edits, then the content defined in the template will used.
|
53
|
+
|
54
|
+
## Content Types
|
55
|
+
|
56
|
+
The helper also supports Mercury's other content types:
|
57
|
+
|
58
|
+
```haml
|
59
|
+
= editable(:title, :h1, type: :simple) # only allow text in h1
|
60
|
+
= editable(:details, :div, type: :full) # allow any markup inside div
|
61
|
+
```
|
62
|
+
|
63
|
+
Most of the time you wont need to explicitly declare the `:type` because the helper can determine `:type` for common tags.
|
64
|
+
The following tags are considered simple: h1, h2, h3, h4, h5, h6, a, span, label
|
65
|
+
|
66
|
+
Installation
|
67
|
+
------------
|
68
|
+
Add `mercury_engine` to your Gemfile:
|
69
|
+
|
70
|
+
```ruby
|
71
|
+
gem 'mercury_engine'
|
72
|
+
```
|
73
|
+
|
74
|
+
Alternatively you can use the git repo directly:
|
75
|
+
|
76
|
+
```ruby
|
77
|
+
gem 'mercury_engine', github: 'DynamoMTL/mercury-engine'
|
78
|
+
```
|
79
|
+
|
80
|
+
Now, update your bundle:
|
81
|
+
|
82
|
+
```bash
|
83
|
+
bundle
|
84
|
+
```
|
85
|
+
|
86
|
+
Then, run the `mercury_engine:install` generator.
|
87
|
+
It will copy over database migrations and will mount the engine in your `config/routes.rb` file.
|
88
|
+
|
89
|
+
```
|
90
|
+
rails generate mercury_engine:install
|
91
|
+
```
|
92
|
+
|
93
|
+
Also, add the "Edit" link somewhere in your layout.
|
94
|
+
The engine provides a partial that will only show the link when an admin is logged in.
|
95
|
+
|
96
|
+
```haml
|
97
|
+
= render partial: 'mercury_engine/shared/edit_link'
|
98
|
+
```
|
99
|
+
# And - you're done.
|
100
|
+
coffee break?
|
101
|
+
|
102
|
+
Testing
|
103
|
+
-------
|
104
|
+
|
105
|
+
Clone the repo:
|
106
|
+
|
107
|
+
$ hub clone DynamoMTL/mercury-engine && cd mercury-engine
|
108
|
+
|
109
|
+
To run tests:
|
110
|
+
|
111
|
+
$ bundle exec rake spec
|
112
|
+
|
113
|
+
To run tests with guard (preferred):
|
114
|
+
|
115
|
+
$ bundle exec guard
|
data/Rakefile
ADDED
@@ -0,0 +1,29 @@
|
|
1
|
+
#!/usr/bin/env rake
|
2
|
+
begin
|
3
|
+
require 'bundler/setup'
|
4
|
+
rescue LoadError
|
5
|
+
puts 'You must `gem install bundler` and `bundle install` to run rake tasks'
|
6
|
+
end
|
7
|
+
begin
|
8
|
+
require 'rdoc/task'
|
9
|
+
rescue LoadError
|
10
|
+
require 'rdoc/rdoc'
|
11
|
+
require 'rake/rdoctask'
|
12
|
+
RDoc::Task = Rake::RDocTask
|
13
|
+
end
|
14
|
+
|
15
|
+
RDoc::Task.new(:rdoc) do |rdoc|
|
16
|
+
rdoc.rdoc_dir = 'rdoc'
|
17
|
+
rdoc.title = 'Mercury Engine'
|
18
|
+
rdoc.options << '--line-numbers'
|
19
|
+
rdoc.rdoc_files.include('README.rdoc')
|
20
|
+
rdoc.rdoc_files.include('lib/**/*.rb')
|
21
|
+
end
|
22
|
+
|
23
|
+
Bundler::GemHelper.install_tasks
|
24
|
+
|
25
|
+
task default: :spec
|
26
|
+
|
27
|
+
task :spec do
|
28
|
+
sh 'cd spec/dummy && bundle exec rake spec'
|
29
|
+
end
|
@@ -0,0 +1,533 @@
|
|
1
|
+
#!
|
2
|
+
# * Mercury Editor is a CoffeeScript and jQuery based WYSIWYG editor. Documentation and other useful information can be
|
3
|
+
# * found at https://github.com/jejacks0n/mercury
|
4
|
+
# *
|
5
|
+
# * Minimum jQuery requirements are 1.7
|
6
|
+
# *= require_self
|
7
|
+
# *
|
8
|
+
# * You can include the Rails jQuery ujs script here to get some nicer behaviors in modals, panels and lightviews when
|
9
|
+
# * using :remote => true within the contents rendered in them.
|
10
|
+
# * require jquery_ujs
|
11
|
+
# *
|
12
|
+
# * Add any requires for the support libraries that integrate nicely with Mercury Editor.
|
13
|
+
# * require mercury/support/history
|
14
|
+
# *
|
15
|
+
# * Require Mercury Editor itself.
|
16
|
+
# *= require mercury/mercury
|
17
|
+
# *
|
18
|
+
# * Require any localizations you wish to support
|
19
|
+
# * Example: es.locale, or fr.locale -- regional dialects are in each language file so never en_US for instance.
|
20
|
+
# * Make sure you enable the localization feature in the configuration.
|
21
|
+
# * require mercury/locales/swedish_chef.locale
|
22
|
+
# *
|
23
|
+
# * Add all requires for plugins that extend or change the behavior of Mercury Editor.
|
24
|
+
# * require mercury/plugins/save_as_xml/plugin.js
|
25
|
+
# *
|
26
|
+
# * Require any files you want to use that either extend, or change the default Mercury behavior.
|
27
|
+
# * require mercury_overrides
|
28
|
+
#
|
29
|
+
window.Mercury =
|
30
|
+
|
31
|
+
# # Mercury Configuration
|
32
|
+
config:
|
33
|
+
|
34
|
+
# ## Toolbars
|
35
|
+
#
|
36
|
+
# This is where you can customize the toolbars by adding or removing buttons, or changing them and their
|
37
|
+
# behaviors. Any top level object put here will create a new toolbar. Buttons are simply nested inside the
|
38
|
+
# toolbars, along with button groups.
|
39
|
+
#
|
40
|
+
# Some toolbars are custom (the snippets toolbar for instance), and to denote that use _custom: true. You can then
|
41
|
+
# build the toolbar yourself with it's own behavior.
|
42
|
+
#
|
43
|
+
# Buttons can be grouped, and a button group is simply a way to wrap buttons for styling -- they can also handle
|
44
|
+
# enabling or disabling all the buttons within it by using a context. The table button group is a good example of
|
45
|
+
# this.
|
46
|
+
#
|
47
|
+
# It's important to note that each of the button names (keys), in each toolbar object must be unique, regardless of
|
48
|
+
# if it's in a button group, or nested, etc. This is because styling is applied to them by name, and because their
|
49
|
+
# name is used in the event that's fired when you click on them.
|
50
|
+
#
|
51
|
+
# Button format: `[label, description, {type: action, type: action, etc}]`
|
52
|
+
#
|
53
|
+
# ### The available button types are:
|
54
|
+
#
|
55
|
+
# - toggle: toggles on or off when clicked, otherwise behaves like a button
|
56
|
+
# - modal: opens a modal window, expects the action to be one of:
|
57
|
+
# 1. a string url
|
58
|
+
# 2. a function that returns a string url
|
59
|
+
# - lightview: opens a lightview window (like modal, but different UI), expects the action to be one of:
|
60
|
+
# 1. a string url
|
61
|
+
# 2. a function that returns a string url
|
62
|
+
# - panel: opens a panel dialog, expects the action to be one of:
|
63
|
+
# 1. a string url
|
64
|
+
# 2. a function that returns a string url
|
65
|
+
# - palette: opens a palette window, expects the action to be one of:
|
66
|
+
# 1. a string url
|
67
|
+
# 2. a function that returns a string url
|
68
|
+
# - select: opens a pulldown style window, expects the action to be one of:
|
69
|
+
# 1. a string url
|
70
|
+
# 2. a function that returns a string url
|
71
|
+
# - context: calls a callback function, expects the action to be:
|
72
|
+
# 1. a function that returns a boolean to highlight the button
|
73
|
+
# note: if a function isn't provided, the key will be passed to the contextHandler, in which case a default
|
74
|
+
# context will be used (for more info read the Contexts section below)
|
75
|
+
# - mode: toggle a given mode in the editor, expects the action to be:
|
76
|
+
# 1. a string, denoting the name of the mode
|
77
|
+
# note: it's assumed that when a specific mode is turned on, all other modes will be turned off, which happens
|
78
|
+
# automatically, thus putting the editor into a specific "state"
|
79
|
+
# - regions: allows buttons to be enabled/disabled based on what region type has focus, expects:
|
80
|
+
# 1. an array of region types (eg. ['full', 'markdown'])
|
81
|
+
# - preload: allows some dialog views to be loaded when the button is created instead of on first open, expects:
|
82
|
+
# 1. a boolean true / false
|
83
|
+
# note: this is only used by panels, selects, and palettes
|
84
|
+
#
|
85
|
+
# Separators are any "button" that's not an array, and are expected to be a string. You can use two different
|
86
|
+
# separator styles: line ('-'), and spacer (' ').
|
87
|
+
#
|
88
|
+
# ### Adding Contexts
|
89
|
+
#
|
90
|
+
# Contexts are used callback functions used for highlighting and disabling/enabling buttons and buttongroups. When
|
91
|
+
# the cursor enters an element within an html region for instance we want to disable or highlight buttons based on
|
92
|
+
# the properties of the given node. You can see examples of contexts in, and add your own to:
|
93
|
+
# `Mercury.Toolbar.Button.contexts` and `Mercury.Toolbar.ButtonGroup.contexts`
|
94
|
+
toolbars:
|
95
|
+
primary:
|
96
|
+
save: ["Save", "Save this page"]
|
97
|
+
preview: ["Preview", "Preview this page",
|
98
|
+
toggle: true
|
99
|
+
mode: true
|
100
|
+
]
|
101
|
+
sep1: " "
|
102
|
+
undoredo:
|
103
|
+
undo: ["Undo", "Undo your last action"]
|
104
|
+
redo: ["Redo", "Redo your last action"]
|
105
|
+
sep: " "
|
106
|
+
|
107
|
+
insertLink: ["Link", "Insert Link",
|
108
|
+
modal: "/mercury/modals/link.html"
|
109
|
+
regions: ["full", "markdown"]
|
110
|
+
]
|
111
|
+
insertMedia: ["Media", "Insert Media (images and videos)",
|
112
|
+
modal: "/mercury/modals/media.html"
|
113
|
+
regions: ["full", "markdown"]
|
114
|
+
]
|
115
|
+
insertTable: ["Table", "Insert Table",
|
116
|
+
modal: "/mercury/modals/table.html"
|
117
|
+
regions: ["full", "markdown"]
|
118
|
+
]
|
119
|
+
insertCharacter: ["Character", "Special Characters",
|
120
|
+
modal: "/mercury/modals/character.html"
|
121
|
+
regions: ["full", "markdown"]
|
122
|
+
]
|
123
|
+
snippetPanel: ["Snippet", "Snippet Panel",
|
124
|
+
panel: "/mercury/panels/snippets.html"
|
125
|
+
]
|
126
|
+
sep2: " "
|
127
|
+
historyPanel: ["History", "Page Version History",
|
128
|
+
panel: "/mercury/panels/history.html"
|
129
|
+
]
|
130
|
+
sep3: " "
|
131
|
+
notesPanel: ["Notes", "Page Notes",
|
132
|
+
panel: "/mercury/panels/notes.html"
|
133
|
+
]
|
134
|
+
|
135
|
+
editable:
|
136
|
+
_regions: ["full", "markdown"]
|
137
|
+
predefined:
|
138
|
+
style: ["Style", null,
|
139
|
+
select: "/mercury/selects/style.html"
|
140
|
+
preload: true
|
141
|
+
]
|
142
|
+
sep1: " "
|
143
|
+
formatblock: ["Block Format", null,
|
144
|
+
select: "/mercury/selects/formatblock.html"
|
145
|
+
preload: true
|
146
|
+
]
|
147
|
+
sep2: "-"
|
148
|
+
|
149
|
+
colors:
|
150
|
+
backColor: ["Background Color", null,
|
151
|
+
palette: "/mercury/palettes/backcolor.html"
|
152
|
+
context: true
|
153
|
+
preload: true
|
154
|
+
regions: ["full"]
|
155
|
+
]
|
156
|
+
sep1: " "
|
157
|
+
foreColor: ["Text Color", null,
|
158
|
+
palette: "/mercury/palettes/forecolor.html"
|
159
|
+
context: true
|
160
|
+
preload: true
|
161
|
+
regions: ["full"]
|
162
|
+
]
|
163
|
+
sep2: "-"
|
164
|
+
|
165
|
+
decoration:
|
166
|
+
bold: ["Bold", null,
|
167
|
+
context: true
|
168
|
+
]
|
169
|
+
italic: ["Italicize", null,
|
170
|
+
context: true
|
171
|
+
]
|
172
|
+
overline: ["Overline", null,
|
173
|
+
context: true
|
174
|
+
regions: ["full"]
|
175
|
+
]
|
176
|
+
strikethrough: ["Strikethrough", null,
|
177
|
+
context: true
|
178
|
+
regions: ["full"]
|
179
|
+
]
|
180
|
+
underline: ["Underline", null,
|
181
|
+
context: true
|
182
|
+
regions: ["full"]
|
183
|
+
]
|
184
|
+
sep: "-"
|
185
|
+
|
186
|
+
script:
|
187
|
+
subscript: ["Subscript", null,
|
188
|
+
context: true
|
189
|
+
]
|
190
|
+
superscript: ["Superscript", null,
|
191
|
+
context: true
|
192
|
+
]
|
193
|
+
sep: "-"
|
194
|
+
|
195
|
+
justify:
|
196
|
+
justifyLeft: ["Align Left", null,
|
197
|
+
context: true
|
198
|
+
regions: ["full"]
|
199
|
+
]
|
200
|
+
justifyCenter: ["Center", null,
|
201
|
+
context: true
|
202
|
+
regions: ["full"]
|
203
|
+
]
|
204
|
+
justifyRight: ["Align Right", null,
|
205
|
+
context: true
|
206
|
+
regions: ["full"]
|
207
|
+
]
|
208
|
+
justifyFull: ["Justify Full", null,
|
209
|
+
context: true
|
210
|
+
regions: ["full"]
|
211
|
+
]
|
212
|
+
sep: "-"
|
213
|
+
|
214
|
+
list:
|
215
|
+
insertUnorderedList: ["Unordered List", null,
|
216
|
+
context: true
|
217
|
+
]
|
218
|
+
insertOrderedList: ["Numbered List", null,
|
219
|
+
context: true
|
220
|
+
]
|
221
|
+
sep: "-"
|
222
|
+
|
223
|
+
indent:
|
224
|
+
outdent: ["Decrease Indentation"]
|
225
|
+
indent: ["Increase Indentation"]
|
226
|
+
sep: "-"
|
227
|
+
|
228
|
+
table:
|
229
|
+
_context: true
|
230
|
+
insertRowBefore: ["Insert Table Row", "Insert a table row before the cursor",
|
231
|
+
regions: ["full"]
|
232
|
+
]
|
233
|
+
insertRowAfter: ["Insert Table Row", "Insert a table row after the cursor",
|
234
|
+
regions: ["full"]
|
235
|
+
]
|
236
|
+
deleteRow: ["Delete Table Row", "Delete this table row",
|
237
|
+
regions: ["full"]
|
238
|
+
]
|
239
|
+
insertColumnBefore: ["Insert Table Column", "Insert a table column before the cursor",
|
240
|
+
regions: ["full"]
|
241
|
+
]
|
242
|
+
insertColumnAfter: ["Insert Table Column", "Insert a table column after the cursor",
|
243
|
+
regions: ["full"]
|
244
|
+
]
|
245
|
+
deleteColumn: ["Delete Table Column", "Delete this table column",
|
246
|
+
regions: ["full"]
|
247
|
+
]
|
248
|
+
sep1: " "
|
249
|
+
increaseColspan: ["Increase Cell Columns", "Increase the cells colspan"]
|
250
|
+
decreaseColspan: ["Decrease Cell Columns", "Decrease the cells colspan and add a new cell"]
|
251
|
+
increaseRowspan: ["Increase Cell Rows", "Increase the cells rowspan"]
|
252
|
+
decreaseRowspan: ["Decrease Cell Rows", "Decrease the cells rowspan and add a new cell"]
|
253
|
+
sep2: "-"
|
254
|
+
|
255
|
+
rules:
|
256
|
+
horizontalRule: ["Horizontal Rule", "Insert a horizontal rule"]
|
257
|
+
sep1: "-"
|
258
|
+
|
259
|
+
formatting:
|
260
|
+
removeFormatting: ["Remove Formatting", "Remove formatting for the selection",
|
261
|
+
regions: ["full"]
|
262
|
+
]
|
263
|
+
sep2: " "
|
264
|
+
|
265
|
+
editors:
|
266
|
+
htmlEditor: ["Edit HTML", "Edit the HTML content",
|
267
|
+
regions: ["full"]
|
268
|
+
]
|
269
|
+
|
270
|
+
snippets:
|
271
|
+
_custom: true
|
272
|
+
actions:
|
273
|
+
editSnippet: ["Edit Snippet Settings"]
|
274
|
+
sep1: " "
|
275
|
+
removeSnippet: ["Remove Snippet"]
|
276
|
+
|
277
|
+
|
278
|
+
# ## Region Options
|
279
|
+
#
|
280
|
+
# You can customize some aspects of how regions are found, identified, and saved.
|
281
|
+
#
|
282
|
+
# attribute: Mercury identifies editable regions by a data-mercury attribute. This attribute has to be added in
|
283
|
+
# your HTML in advance, and is the only real code/naming exposed in the implementation of Mercury. To allow this
|
284
|
+
# to be as configurable as possible, you can set the name of this attribute. If you change this, you should adjust
|
285
|
+
# the injected styles as well.
|
286
|
+
#
|
287
|
+
# identifier: This is used as a unique identifier for any given region (and thus should be unique to the page).
|
288
|
+
# By default this is the id attribute but can be changed to a data attribute should you want to use something
|
289
|
+
# custom instead.
|
290
|
+
#
|
291
|
+
# dataAttributes: The dataAttributes is an array of data attributes that will be serialized and returned to the
|
292
|
+
# server upon saving. These attributes, when applied to a Mercury region element, will be automatically serialized
|
293
|
+
# and submitted with the AJAX request sent when a page is saved. These are expected to be HTML5 data attributes,
|
294
|
+
# and 'data-' will automatically be prepended to each item in this directive. (ex. ['scope', 'version'])
|
295
|
+
#
|
296
|
+
# determineType: This function is called after checking the data-type attribute for the correct field type. Use
|
297
|
+
# it if you want to dynamically set the type based on inspection of the region.
|
298
|
+
regions:
|
299
|
+
attribute: "data-mercury"
|
300
|
+
identifier: "id"
|
301
|
+
dataAttributes: []
|
302
|
+
|
303
|
+
|
304
|
+
# determineType: function(region){},
|
305
|
+
|
306
|
+
# ## Snippet Options / Preview
|
307
|
+
#
|
308
|
+
# When a user drags a snippet onto the page they'll be prompted to enter options for the given snippet. The server
|
309
|
+
# is expected to respond with a form. Once the user submits this form, an Ajax request is sent to the server with
|
310
|
+
# the options provided; this preview request is expected to respond with the rendered markup for the snippet.
|
311
|
+
#
|
312
|
+
# method: The HTTP method used when submitting both the options and the preview. We use POST by default because a
|
313
|
+
# snippet options form may contain large text inputs and we don't want that to be truncated when sent to the
|
314
|
+
# server.
|
315
|
+
#
|
316
|
+
# optionsUrl: The url that the options form will be loaded from.
|
317
|
+
#
|
318
|
+
# previewUrl: The url that the options will be submitted to, and will return the rendered snippet markup.
|
319
|
+
#
|
320
|
+
# **Note:** `:name` will be replaced with the snippet name in the urls (eg. /mercury/snippets/example/options.html)
|
321
|
+
snippets:
|
322
|
+
method: "POST"
|
323
|
+
optionsUrl: "/mercury/snippets/:name/options.html"
|
324
|
+
previewUrl: "/mercury/snippets/:name/preview.html"
|
325
|
+
|
326
|
+
|
327
|
+
# ## Image Uploading
|
328
|
+
#
|
329
|
+
# If you drag images from your desktop into regions that support it, it will be uploaded to the server and inserted
|
330
|
+
# into the region. You can disable or enable this feature, the accepted mime-types, file size restrictions, and
|
331
|
+
# other things related to uploading.
|
332
|
+
#
|
333
|
+
# **Note:** Image uploading is only supported in some region types, and some browsers.
|
334
|
+
#
|
335
|
+
# enabled: You can set this to true, or false if you want to disable the feature entirely.
|
336
|
+
#
|
337
|
+
# allowedMimeTypes: You can restrict the types of files that can be uploaded by providing a list of allowed mime
|
338
|
+
# types.
|
339
|
+
#
|
340
|
+
# maxFileSize: You can restrict large files by setting the maxFileSize (in bytes).
|
341
|
+
#
|
342
|
+
# inputName: When uploading, a form is generated and submitted to the server via Ajax. If your server would prefer
|
343
|
+
# a different name for how the image comes through, you can change the inputName.
|
344
|
+
#
|
345
|
+
# url: The url that the image upload will be submitted to.
|
346
|
+
#
|
347
|
+
# handler: You can use false to let Mercury handle it for you, or you can provide a handler function that can
|
348
|
+
# modify the response from the server. This can be useful if your server doesn't respond the way Mercury expects.
|
349
|
+
# The handler function should take the response from the server and return an object that matches:
|
350
|
+
# `{image: {url: '[your provided url]'}`
|
351
|
+
uploading:
|
352
|
+
enabled: true
|
353
|
+
allowedMimeTypes: ["image/jpeg", "image/gif", "image/png"]
|
354
|
+
maxFileSize: 1235242880
|
355
|
+
inputName: "image[image]"
|
356
|
+
url: "/mercury/images"
|
357
|
+
handler: false
|
358
|
+
|
359
|
+
|
360
|
+
# ## Localization / I18n
|
361
|
+
#
|
362
|
+
# Include the .locale files you want to support when loading Mercury. The files are always named by the language,
|
363
|
+
# and not the regional dialect (eg. en.locale.js) because the regional dialects are nested within the primary
|
364
|
+
# locale files.
|
365
|
+
#
|
366
|
+
# The client locale will be used first, and if no proper locale file is found for their language then the fallback
|
367
|
+
# preferredLocale configuration will be used. If one isn't provided, and the client locale isn't included, the
|
368
|
+
# strings will remain untranslated.
|
369
|
+
#
|
370
|
+
# enabled: Set to false to disable, true to enable.
|
371
|
+
#
|
372
|
+
# preferredLocale: If a client doesn't support the locales you've included, this is used as a fallback.
|
373
|
+
localization:
|
374
|
+
enabled: false
|
375
|
+
preferredLocale: "swedish_chef-BORK"
|
376
|
+
|
377
|
+
|
378
|
+
# ## Behaviors
|
379
|
+
#
|
380
|
+
# Behaviors are used to change the default behaviors of a given region type when a given button is clicked. For
|
381
|
+
# example, you may prefer to add HR tags using an HR wrapped within a div with a classname (for styling). You
|
382
|
+
# can add your own complex behaviors here and they'll be shared across all regions.
|
383
|
+
#
|
384
|
+
# If you want to add behaviors to specific region types, you can mix them into the actions property of any region
|
385
|
+
# type.
|
386
|
+
#
|
387
|
+
# Mercury.Regions.Full.actions.htmlEditor = function() {}
|
388
|
+
#
|
389
|
+
# You can see how the behavior matches up directly with the button names. It's also important to note that the
|
390
|
+
# callback functions are executed within the scope of the given region, so you have access to all it's methods.
|
391
|
+
behaviors:
|
392
|
+
|
393
|
+
#foreColor: function(selection, options) { selection.wrap('<span style="color:' + options.value.toHex() + '">', true) },
|
394
|
+
htmlEditor: ->
|
395
|
+
Mercury.modal "/mercury/modals/htmleditor.html",
|
396
|
+
title: "HTML Editor"
|
397
|
+
fullHeight: true
|
398
|
+
handler: "htmlEditor"
|
399
|
+
|
400
|
+
|
401
|
+
|
402
|
+
# ## Global Behaviors
|
403
|
+
#
|
404
|
+
# Global behaviors are much like behaviors, but are more "global". Things like save, exit, etc. can be included
|
405
|
+
# here. They'll only be called once, and execute within the scope of whatever editor is instantiated (eg.
|
406
|
+
# PageEditor).
|
407
|
+
#
|
408
|
+
# An example of changing how saving works:
|
409
|
+
#
|
410
|
+
# save: function() {
|
411
|
+
# var data = top.JSON.stringify(this.serialize(), null, ' ');
|
412
|
+
# var content = '<textarea style="width:500px;height:200px" wrap="off">' + data + '</textarea>';
|
413
|
+
# Mercury.modal(null, {title: 'Saving', closeButton: true, content: content})
|
414
|
+
# }
|
415
|
+
#
|
416
|
+
# This is a nice way to add functionality, when the behaviors aren't region specific. These can be triggered by a
|
417
|
+
# button, or manually with `Mercury.trigger('action', {action: 'barrelRoll'})`
|
418
|
+
globalBehaviors:
|
419
|
+
exit: ->
|
420
|
+
window.location.href = @iframeSrc()
|
421
|
+
|
422
|
+
barrelRoll: ->
|
423
|
+
$("body").css webkitTransform: "rotate(360deg)"
|
424
|
+
|
425
|
+
|
426
|
+
# ## Ajax and CSRF Headers
|
427
|
+
#
|
428
|
+
# Some server frameworks require that you provide a specific header for Ajax requests. The values for these CSRF
|
429
|
+
# tokens are typically stored in the rendered DOM. By default, Mercury will look for the Rails specific meta tag,
|
430
|
+
# and provide the X-CSRF-Token header on Ajax requests, but you can modify this configuration if the system you're
|
431
|
+
# using doesn't follow the same standard.
|
432
|
+
csrfSelector: "meta[name=\"csrf-token\"]"
|
433
|
+
csrfHeader: "X-CSRF-Token"
|
434
|
+
|
435
|
+
# ## Editor URLs
|
436
|
+
#
|
437
|
+
# When loading a given page, you may want to tweak this regex. It's to allow the url to differ from the page
|
438
|
+
# you're editing, and the url at which you access it.
|
439
|
+
editorUrlRegEx: /([http|https]:\/\/.[^\/]*)\/editor\/?(.*)/i
|
440
|
+
|
441
|
+
# ## Hijacking Links & Forms
|
442
|
+
#
|
443
|
+
# Mercury will hijack links and forms that don't have a target set, or the target is set to _self and will set it
|
444
|
+
# to _parent. This is because the target must be set properly for Mercury to not get in the way of some
|
445
|
+
# functionality, like proper page loads on form submissions etc. Mercury doesn't do this to links or forms that
|
446
|
+
# are within editable regions because it doesn't want to impact the html that's saved. With that being explained,
|
447
|
+
# you can add classes to links or forms that you don't want this behavior added to. Let's say you have links that
|
448
|
+
# open a lightbox style window, and you don't want the targets of these to be set to _parent. You can add classes
|
449
|
+
# to this array, and they will be ignored when the hijacking is applied.
|
450
|
+
nonHijackableClasses: []
|
451
|
+
|
452
|
+
# ## Pasting & Sanitizing
|
453
|
+
#
|
454
|
+
# When pasting content into Mercury it may sometimes contain HTML tags and attributes. This markup is used to
|
455
|
+
# style the content and makes the pasted content look (and behave) the same as the original content. This can be a
|
456
|
+
# desired feature or an annoyance, so you can enable various sanitizing methods to clean the content when it's
|
457
|
+
# pasted.
|
458
|
+
#
|
459
|
+
# sanitize: Can be any of the following:
|
460
|
+
# - false: no sanitizing is done, the content is pasted the exact same as it was copied by the user
|
461
|
+
# - 'whitelist': content is cleaned using the settings specified in the tag white list (described below)
|
462
|
+
# - 'text': all html is stripped before pasting, leaving only the raw text
|
463
|
+
#
|
464
|
+
# whitelist: The white list allows you to specify tags and attributes that are allowed when pasting content. Each
|
465
|
+
# item in this object should contain the allowed tag, and an array of attributes that are allowed on that tag. If
|
466
|
+
# the allowed attributes array is empty, all attributes will be removed. If a tag is not present in this list, it
|
467
|
+
# will be removed, but without removing any of the text or tags inside it.
|
468
|
+
#
|
469
|
+
# **Note:** Content is *always* sanitized if looks like it's from MS Word or similar editors regardless of this
|
470
|
+
# configuration.
|
471
|
+
pasting:
|
472
|
+
sanitize: "whitelist"
|
473
|
+
whitelist:
|
474
|
+
h1: []
|
475
|
+
h2: []
|
476
|
+
h3: []
|
477
|
+
h4: []
|
478
|
+
h5: []
|
479
|
+
h6: []
|
480
|
+
table: []
|
481
|
+
thead: []
|
482
|
+
tbody: []
|
483
|
+
tfoot: []
|
484
|
+
tr: []
|
485
|
+
th: ["colspan", "rowspan"]
|
486
|
+
td: ["colspan", "rowspan"]
|
487
|
+
div: ["class"]
|
488
|
+
span: ["class"]
|
489
|
+
ul: []
|
490
|
+
ol: []
|
491
|
+
li: []
|
492
|
+
b: []
|
493
|
+
strong: []
|
494
|
+
i: []
|
495
|
+
em: []
|
496
|
+
u: []
|
497
|
+
strike: []
|
498
|
+
br: []
|
499
|
+
p: []
|
500
|
+
hr: []
|
501
|
+
a: ["href", "target", "title", "name"]
|
502
|
+
img: ["src", "title", "alt"]
|
503
|
+
|
504
|
+
|
505
|
+
# ## Injected Styles
|
506
|
+
#
|
507
|
+
# Mercury tries to stay as much out of your code as possible, but because regions appear within your document we
|
508
|
+
# need to include a few styles to indicate regions, as well as the different states of them (eg. focused). These
|
509
|
+
# styles are injected into your document, and as simple as they might be, you may want to change them.
|
510
|
+
injectedStyles: "" + "[data-mercury] { min-height: 10px; outline: 1px dotted #09F } " + "[data-mercury]:focus { outline: none; -webkit-box-shadow: 0 0 10px #09F, 0 0 1px #045; box-shadow: 0 0 10px #09F, 0 0 1px #045 }" + "[data-mercury].focus { outline: none; -webkit-box-shadow: 0 0 10px #09F, 0 0 1px #045; box-shadow: 0 0 10px #09F, 0 0 1px #045 }" + "[data-mercury]:after { content: \".\"; display: block; visibility: hidden; clear: both; height: 0; overflow: hidden; }" + "[data-mercury] table { border: 1px dotted red; min-width: 6px; }" + "[data-mercury] th { border: 1px dotted red; min-width: 6px; }" + "[data-mercury] td { border: 1px dotted red; min-width: 6px; }" + "[data-mercury] .mercury-textarea { border: 0; box-sizing: border-box; -moz-box-sizing: border-box; -webkit-box-sizing: border-box; resize: none; }" + "[data-mercury] .mercury-textarea:focus { outline: none; -webkit-box-shadow: none; -moz-box-shadow: none; box-shadow: none; }"
|
511
|
+
|
512
|
+
|
513
|
+
# ## Silent Mode
|
514
|
+
#
|
515
|
+
# Turning silent mode on will disable asking about unsaved changes before leaving the page.
|
516
|
+
silent: false
|
517
|
+
|
518
|
+
# ## Debug Mode
|
519
|
+
#
|
520
|
+
# Turning debug mode on will log events and other various things (using console.debug if available).
|
521
|
+
debug: false
|
522
|
+
|
523
|
+
$(window).bind 'mercury:saved', ->
|
524
|
+
window.location = window.location.href.replace(/\/editor\//i, '/')
|
525
|
+
|
526
|
+
$(window).bind 'mercury:ready', ->
|
527
|
+
iframe = $('#mercury_iframe').contents()
|
528
|
+
body = iframe.find('body')
|
529
|
+
element = iframe.find('#edit-page')
|
530
|
+
|
531
|
+
Mercury.saveURL = element.data('save-url')
|
532
|
+
element.hide()
|
533
|
+
body.addClass('editing')
|
@@ -0,0 +1,27 @@
|
|
1
|
+
/*!
|
2
|
+
* Mercury Editor is a Coffeescript and jQuery based WYSIWYG editor. Documentation and other useful information can be
|
3
|
+
* found at https://github.com/jejacks0n/mercury
|
4
|
+
*
|
5
|
+
* Copyright (c) 2011 Jeremy Jackson
|
6
|
+
*
|
7
|
+
* Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated
|
8
|
+
* documentation files (the "Software"), to deal in the Software without restriction, including without limitation the
|
9
|
+
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit
|
10
|
+
* persons to whom the Software is furnished to do so, subject to the following conditions:
|
11
|
+
*
|
12
|
+
* The above copyright notice and this permission notice shall be included in all copies or substantial portions of the
|
13
|
+
* Software.
|
14
|
+
*
|
15
|
+
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE
|
16
|
+
* WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
|
17
|
+
* COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
|
18
|
+
* OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
19
|
+
*
|
20
|
+
*= require_self
|
21
|
+
*= require mercury/bootstrap-ish
|
22
|
+
*= require mercury/mercury
|
23
|
+
*/
|
24
|
+
|
25
|
+
.mercury-iframe {
|
26
|
+
height: 100%;
|
27
|
+
}
|
@@ -0,0 +1,18 @@
|
|
1
|
+
class PagesController < ApplicationController
|
2
|
+
include PagesHelper
|
3
|
+
protect_from_forgery
|
4
|
+
before_filter :authenticate_admin_user!, except: :show
|
5
|
+
|
6
|
+
def show
|
7
|
+
path = params[:path]
|
8
|
+
@page = Page.find_by_path(path)
|
9
|
+
|
10
|
+
render action: template_path(path)
|
11
|
+
end
|
12
|
+
|
13
|
+
def update
|
14
|
+
Page.update_content(params[:path], format_content(params[:content]))
|
15
|
+
|
16
|
+
render text: ''
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,78 @@
|
|
1
|
+
module PagesHelper
|
2
|
+
SIMPLE_TAGS = %w(h1 h2 h3 h4 h5 h6 a span label)
|
3
|
+
|
4
|
+
def format_content(content)
|
5
|
+
content.reduce({}) do |hash, (key, data)|
|
6
|
+
hash[key.to_s] = data[:value] || data[:attributes][:src]
|
7
|
+
hash
|
8
|
+
end
|
9
|
+
end
|
10
|
+
|
11
|
+
def editable(id, tag=:div, options={}, &block)
|
12
|
+
content = content_for(id)
|
13
|
+
type = options[:type] || find_type_for_tag(tag)
|
14
|
+
|
15
|
+
if tag.is_a?(Hash)
|
16
|
+
options = tag if tag.is_a?(Hash)
|
17
|
+
tag = :div
|
18
|
+
end
|
19
|
+
|
20
|
+
options[:id] = id
|
21
|
+
|
22
|
+
set_mercury_options(options, type)
|
23
|
+
|
24
|
+
if type == :simple
|
25
|
+
if content
|
26
|
+
content_tag(tag, content, options.except(:type))
|
27
|
+
else
|
28
|
+
content_tag(tag, options.except(:type), &block)
|
29
|
+
end
|
30
|
+
else
|
31
|
+
content_tag(tag, options.except(:type), false) do
|
32
|
+
raw(content || (block.call if block))
|
33
|
+
end
|
34
|
+
end
|
35
|
+
end
|
36
|
+
|
37
|
+
def editable_image(id, default=nil)
|
38
|
+
source = content_for(id) || default
|
39
|
+
options = {id: id}
|
40
|
+
|
41
|
+
set_mercury_options(options, :image)
|
42
|
+
|
43
|
+
image_tag(source, options)
|
44
|
+
end
|
45
|
+
|
46
|
+
def title
|
47
|
+
editable(:title, :h1) { @page.content[:title] }
|
48
|
+
end
|
49
|
+
|
50
|
+
def content_for(id)
|
51
|
+
@page.content[id.to_s] if @page
|
52
|
+
end
|
53
|
+
|
54
|
+
def set_mercury_options(options, type)
|
55
|
+
if params[:mercury_frame]
|
56
|
+
options[:data] ||= {}
|
57
|
+
options[:data][:mercury] = type
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def find_type_for_tag(tag)
|
62
|
+
if SIMPLE_TAGS.include?(tag.to_s)
|
63
|
+
:simple
|
64
|
+
else
|
65
|
+
:full
|
66
|
+
end
|
67
|
+
end
|
68
|
+
|
69
|
+
def template_path(path)
|
70
|
+
return 'index' if path.blank?
|
71
|
+
|
72
|
+
if template_exists?(path + '/index', %w(pages))
|
73
|
+
path + '/index'
|
74
|
+
else
|
75
|
+
path
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
data/app/models/page.rb
ADDED
@@ -0,0 +1,42 @@
|
|
1
|
+
class Page < ActiveRecord::Base
|
2
|
+
acts_as_tree
|
3
|
+
has_paper_trail
|
4
|
+
attr_accessible :content, :parent, :parent_id, :permalink
|
5
|
+
serialize :content, Hash
|
6
|
+
|
7
|
+
class << self
|
8
|
+
def find_by_path(path)
|
9
|
+
return unless page = root
|
10
|
+
|
11
|
+
page.walk(path) do |last, part|
|
12
|
+
last.children.find_by_permalink(part) if last
|
13
|
+
end
|
14
|
+
end
|
15
|
+
|
16
|
+
def update_content(path, content = {})
|
17
|
+
page = root || Page.create(permalink: '')
|
18
|
+
|
19
|
+
page = page.walk(path) do |last, part|
|
20
|
+
last.children.find_or_create_by_permalink(part)
|
21
|
+
end
|
22
|
+
|
23
|
+
page.update_attributes(content: content)
|
24
|
+
|
25
|
+
page
|
26
|
+
end
|
27
|
+
end
|
28
|
+
|
29
|
+
def path
|
30
|
+
if parent
|
31
|
+
Pathname.new(parent.path) + permalink
|
32
|
+
else
|
33
|
+
Pathname.new("/")
|
34
|
+
end.to_s
|
35
|
+
end
|
36
|
+
|
37
|
+
def walk(path, &block)
|
38
|
+
parts = Pathname.new(path.to_s).each_filename
|
39
|
+
|
40
|
+
parts.reduce(self, &block)
|
41
|
+
end
|
42
|
+
end
|
@@ -0,0 +1,19 @@
|
|
1
|
+
!!!
|
2
|
+
%html
|
3
|
+
%head
|
4
|
+
%meta{:content => "width=device-width, maximum-scale=1.0, initial-scale=1.0", :name => "viewport"}/
|
5
|
+
= csrf_meta_tags
|
6
|
+
%title Content Manager
|
7
|
+
= stylesheet_link_tag 'mercury_engine/mercury'
|
8
|
+
= javascript_include_tag 'jquery-1.7', 'mercury_engine/mercury'
|
9
|
+
%body
|
10
|
+
:javascript
|
11
|
+
// Set to the url that you want to save any given page to, leave null for default handling.
|
12
|
+
var saveUrl = null;
|
13
|
+
|
14
|
+
// Instantiate the PageEditor
|
15
|
+
new Mercury.PageEditor(saveUrl, {
|
16
|
+
saveStyle: 'form', // 'form', or 'json' (default json)
|
17
|
+
saveMethod: null, // 'PUT', or 'POST', (create, vs. update -- default PUT)
|
18
|
+
visible: true // boolean - if the interface should start visible or not
|
19
|
+
});
|
data/config/routes.rb
ADDED
@@ -0,0 +1,13 @@
|
|
1
|
+
class CreateVersions < ActiveRecord::Migration
|
2
|
+
def change
|
3
|
+
create_table :versions do |t|
|
4
|
+
t.string :item_type, :null => false
|
5
|
+
t.integer :item_id, :null => false
|
6
|
+
t.string :event, :null => false
|
7
|
+
t.string :whodunnit
|
8
|
+
t.text :object
|
9
|
+
t.datetime :created_at
|
10
|
+
end
|
11
|
+
add_index :versions, [:item_type, :item_id]
|
12
|
+
end
|
13
|
+
end
|
@@ -0,0 +1,23 @@
|
|
1
|
+
module MercuryEngine
|
2
|
+
module Generators
|
3
|
+
class InstallGenerator < Rails::Generators::Base
|
4
|
+
def add_route
|
5
|
+
route "mount MercuryEngine::Engine => '/'"
|
6
|
+
end
|
7
|
+
|
8
|
+
def add_migrations
|
9
|
+
run 'bundle exec rake railties:install:migrations FROM=mercury_engine'
|
10
|
+
end
|
11
|
+
|
12
|
+
def run_migrations
|
13
|
+
response = ask "Would you like to run the migrations now? [Y/n]"
|
14
|
+
|
15
|
+
if response == "" || response.downcase == "y"
|
16
|
+
run 'bundle exec rake db:migrate'
|
17
|
+
else
|
18
|
+
say "Skipping rake db:migrate, don't forget to run it!", :yellow
|
19
|
+
end
|
20
|
+
end
|
21
|
+
end
|
22
|
+
end
|
23
|
+
end
|
metadata
ADDED
@@ -0,0 +1,211 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: mercury_engine
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Josh Nussbaum
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2013-04-26 00:00:00.000000000 Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: rails
|
16
|
+
requirement: !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ~>
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: 3.2.13
|
22
|
+
type: :runtime
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: !ruby/object:Gem::Requirement
|
25
|
+
none: false
|
26
|
+
requirements:
|
27
|
+
- - ~>
|
28
|
+
- !ruby/object:Gem::Version
|
29
|
+
version: 3.2.13
|
30
|
+
- !ruby/object:Gem::Dependency
|
31
|
+
name: mercury-rails
|
32
|
+
requirement: !ruby/object:Gem::Requirement
|
33
|
+
none: false
|
34
|
+
requirements:
|
35
|
+
- - ! '>='
|
36
|
+
- !ruby/object:Gem::Version
|
37
|
+
version: '0'
|
38
|
+
type: :runtime
|
39
|
+
prerelease: false
|
40
|
+
version_requirements: !ruby/object:Gem::Requirement
|
41
|
+
none: false
|
42
|
+
requirements:
|
43
|
+
- - ! '>='
|
44
|
+
- !ruby/object:Gem::Version
|
45
|
+
version: '0'
|
46
|
+
- !ruby/object:Gem::Dependency
|
47
|
+
name: devise
|
48
|
+
requirement: !ruby/object:Gem::Requirement
|
49
|
+
none: false
|
50
|
+
requirements:
|
51
|
+
- - ! '>='
|
52
|
+
- !ruby/object:Gem::Version
|
53
|
+
version: '0'
|
54
|
+
type: :runtime
|
55
|
+
prerelease: false
|
56
|
+
version_requirements: !ruby/object:Gem::Requirement
|
57
|
+
none: false
|
58
|
+
requirements:
|
59
|
+
- - ! '>='
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '0'
|
62
|
+
- !ruby/object:Gem::Dependency
|
63
|
+
name: acts_as_tree
|
64
|
+
requirement: !ruby/object:Gem::Requirement
|
65
|
+
none: false
|
66
|
+
requirements:
|
67
|
+
- - ! '>='
|
68
|
+
- !ruby/object:Gem::Version
|
69
|
+
version: '0'
|
70
|
+
type: :runtime
|
71
|
+
prerelease: false
|
72
|
+
version_requirements: !ruby/object:Gem::Requirement
|
73
|
+
none: false
|
74
|
+
requirements:
|
75
|
+
- - ! '>='
|
76
|
+
- !ruby/object:Gem::Version
|
77
|
+
version: '0'
|
78
|
+
- !ruby/object:Gem::Dependency
|
79
|
+
name: paper_trail
|
80
|
+
requirement: !ruby/object:Gem::Requirement
|
81
|
+
none: false
|
82
|
+
requirements:
|
83
|
+
- - ! '>='
|
84
|
+
- !ruby/object:Gem::Version
|
85
|
+
version: '0'
|
86
|
+
type: :runtime
|
87
|
+
prerelease: false
|
88
|
+
version_requirements: !ruby/object:Gem::Requirement
|
89
|
+
none: false
|
90
|
+
requirements:
|
91
|
+
- - ! '>='
|
92
|
+
- !ruby/object:Gem::Version
|
93
|
+
version: '0'
|
94
|
+
- !ruby/object:Gem::Dependency
|
95
|
+
name: haml-rails
|
96
|
+
requirement: !ruby/object:Gem::Requirement
|
97
|
+
none: false
|
98
|
+
requirements:
|
99
|
+
- - ! '>='
|
100
|
+
- !ruby/object:Gem::Version
|
101
|
+
version: '0'
|
102
|
+
type: :runtime
|
103
|
+
prerelease: false
|
104
|
+
version_requirements: !ruby/object:Gem::Requirement
|
105
|
+
none: false
|
106
|
+
requirements:
|
107
|
+
- - ! '>='
|
108
|
+
- !ruby/object:Gem::Version
|
109
|
+
version: '0'
|
110
|
+
- !ruby/object:Gem::Dependency
|
111
|
+
name: sass-rails
|
112
|
+
requirement: !ruby/object:Gem::Requirement
|
113
|
+
none: false
|
114
|
+
requirements:
|
115
|
+
- - ! '>='
|
116
|
+
- !ruby/object:Gem::Version
|
117
|
+
version: '0'
|
118
|
+
type: :runtime
|
119
|
+
prerelease: false
|
120
|
+
version_requirements: !ruby/object:Gem::Requirement
|
121
|
+
none: false
|
122
|
+
requirements:
|
123
|
+
- - ! '>='
|
124
|
+
- !ruby/object:Gem::Version
|
125
|
+
version: '0'
|
126
|
+
- !ruby/object:Gem::Dependency
|
127
|
+
name: coffee-rails
|
128
|
+
requirement: !ruby/object:Gem::Requirement
|
129
|
+
none: false
|
130
|
+
requirements:
|
131
|
+
- - ! '>='
|
132
|
+
- !ruby/object:Gem::Version
|
133
|
+
version: '0'
|
134
|
+
type: :runtime
|
135
|
+
prerelease: false
|
136
|
+
version_requirements: !ruby/object:Gem::Requirement
|
137
|
+
none: false
|
138
|
+
requirements:
|
139
|
+
- - ! '>='
|
140
|
+
- !ruby/object:Gem::Version
|
141
|
+
version: '0'
|
142
|
+
- !ruby/object:Gem::Dependency
|
143
|
+
name: sqlite3
|
144
|
+
requirement: !ruby/object:Gem::Requirement
|
145
|
+
none: false
|
146
|
+
requirements:
|
147
|
+
- - ! '>='
|
148
|
+
- !ruby/object:Gem::Version
|
149
|
+
version: '0'
|
150
|
+
type: :development
|
151
|
+
prerelease: false
|
152
|
+
version_requirements: !ruby/object:Gem::Requirement
|
153
|
+
none: false
|
154
|
+
requirements:
|
155
|
+
- - ! '>='
|
156
|
+
- !ruby/object:Gem::Version
|
157
|
+
version: '0'
|
158
|
+
description: Adds models and helpers that make it easy to integrate Mercury Editor
|
159
|
+
into any rails app
|
160
|
+
email: joshnuss@gmail.com
|
161
|
+
executables: []
|
162
|
+
extensions: []
|
163
|
+
extra_rdoc_files: []
|
164
|
+
files:
|
165
|
+
- app/assets/stylesheets/mercury_engine/mercury.css
|
166
|
+
- app/assets/javascripts/mercury_engine/mercury.js.coffee
|
167
|
+
- app/views/mercury_engine/shared/_edit_link.html.haml
|
168
|
+
- app/views/layouts/mercury.html.haml
|
169
|
+
- app/models/page.rb
|
170
|
+
- app/helpers/pages_helper.rb
|
171
|
+
- app/controllers/pages_controller.rb
|
172
|
+
- config/routes.rb
|
173
|
+
- db/migrate/20130318125100_create_pages.rb
|
174
|
+
- db/migrate/20130318080000_create_versions.rb
|
175
|
+
- lib/tasks/mercury_engine_tasks.rake
|
176
|
+
- lib/mercury_engine.rb
|
177
|
+
- lib/generators/mercury_engine/install_generator.rb
|
178
|
+
- lib/mercury_engine/version.rb
|
179
|
+
- lib/mercury_engine/engine.rb
|
180
|
+
- lib/mercury/authentication.rb
|
181
|
+
- MIT-LICENSE
|
182
|
+
- Rakefile
|
183
|
+
- README.md
|
184
|
+
homepage: http://www.github.com/DynamoMTL/mercury-engine
|
185
|
+
licenses: []
|
186
|
+
post_install_message:
|
187
|
+
rdoc_options: []
|
188
|
+
require_paths:
|
189
|
+
- lib
|
190
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
191
|
+
none: false
|
192
|
+
requirements:
|
193
|
+
- - ! '>='
|
194
|
+
- !ruby/object:Gem::Version
|
195
|
+
version: 1.9.3
|
196
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
197
|
+
none: false
|
198
|
+
requirements:
|
199
|
+
- - ! '>='
|
200
|
+
- !ruby/object:Gem::Version
|
201
|
+
version: '0'
|
202
|
+
segments:
|
203
|
+
- 0
|
204
|
+
hash: -89830127
|
205
|
+
requirements: []
|
206
|
+
rubyforge_project:
|
207
|
+
rubygems_version: 1.8.25
|
208
|
+
signing_key:
|
209
|
+
specification_version: 3
|
210
|
+
summary: Content management in an engine using Mercury Editor
|
211
|
+
test_files: []
|