theme asset picker in progress
@ -7,6 +7,10 @@ module Admin
|
|||||||
assets = current_site.theme_assets.all
|
assets = current_site.theme_assets.all
|
||||||
@non_image_assets = assets.find_all { |a| a.stylesheet? || a.javascript? }
|
@non_image_assets = assets.find_all { |a| a.stylesheet? || a.javascript? }
|
||||||
@image_assets = assets.find_all { |a| a.image? }
|
@image_assets = assets.find_all { |a| a.image? }
|
||||||
|
|
||||||
|
if request.xhr?
|
||||||
|
render :action => 'images', :layout => false
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
def new
|
def new
|
||||||
@ -18,7 +22,17 @@ module Admin
|
|||||||
end
|
end
|
||||||
|
|
||||||
def create
|
def create
|
||||||
@asset = current_site.theme_assets.build(params[:theme_asset])
|
# logger.debug "request = #{request.inspect}"
|
||||||
|
# logger.debug "file size = #{request.env['rack.input'].inspect}"
|
||||||
|
|
||||||
|
# File.cp(request.env['rack.input'], '/Users/didier/Desktop/FOO')
|
||||||
|
|
||||||
|
if params[:file]
|
||||||
|
# params[:theme_asset][:source] = request.env['rack.input']
|
||||||
|
@asset = current_site.theme_assets.build(:source => params[:file])
|
||||||
|
else
|
||||||
|
@asset = current_site.theme_assets.build(params[:theme_asset])
|
||||||
|
end
|
||||||
|
|
||||||
if @asset.save
|
if @asset.save
|
||||||
flash_success!
|
flash_success!
|
||||||
|
@ -64,6 +64,7 @@ class ThemeAsset
|
|||||||
|
|
||||||
def filename
|
def filename
|
||||||
if not self.image?
|
if not self.image?
|
||||||
|
# TODO: fix that for handling not image / stylesheets / javascripts assets
|
||||||
"#{self.slug}.#{self.stylesheet? ? 'css' : 'js'}"
|
"#{self.slug}.#{self.stylesheet? ? 'css' : 'js'}"
|
||||||
else
|
else
|
||||||
"#{self.slug}#{File.extname(self.source.file.original_filename)}"
|
"#{self.slug}#{File.extname(self.source.file.original_filename)}"
|
||||||
|
@ -30,6 +30,8 @@ class AssetUploader < CarrierWave::Uploader::Base
|
|||||||
def set_content_type
|
def set_content_type
|
||||||
value = :other
|
value = :other
|
||||||
|
|
||||||
|
content_type = File.mime_type?(original_filename) if file.content_type == 'application/octet-stream'
|
||||||
|
|
||||||
self.class.content_types.each_pair do |type, rules|
|
self.class.content_types.each_pair do |type, rules|
|
||||||
rules.each do |rule|
|
rules.each do |rule|
|
||||||
case rule
|
case rule
|
||||||
|
@ -36,11 +36,13 @@ class ThemeAssetUploader < CarrierWave::Uploader::Base
|
|||||||
def set_content_type
|
def set_content_type
|
||||||
value = :other
|
value = :other
|
||||||
|
|
||||||
|
content_type = File.mime_type?(original_filename) if file.content_type == 'application/octet-stream'
|
||||||
|
|
||||||
self.class.content_types.each_pair do |type, rules|
|
self.class.content_types.each_pair do |type, rules|
|
||||||
rules.each do |rule|
|
rules.each do |rule|
|
||||||
case rule
|
case rule
|
||||||
when String then value = type if file.content_type == rule
|
when String then value = type if content_type == rule
|
||||||
when Regexp then value = type if (file.content_type =~ rule) == 0
|
when Regexp then value = type if (content_type =~ rule) == 0
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
= stylesheet_link_tag 'blueprint/screen', :media => 'screen'
|
= stylesheet_link_tag 'blueprint/screen', :media => 'screen'
|
||||||
/ [if IE]
|
/ [if IE]
|
||||||
= stylesheet_link_tag('blueprint/ie', :media => 'screen')
|
= stylesheet_link_tag 'blueprint/ie', :media => 'screen'
|
||||||
|
|
||||||
= stylesheet_link_tag 'admin/layout', 'admin/plugins/toggle', 'admin/menu', 'admin/buttons', 'admin/formtastic', 'admin/formtastic_changes', 'admin/application', :media => 'screen', :cache => Rails.env.production?
|
= stylesheet_link_tag 'admin/layout', 'admin/plugins/toggle', 'admin/menu', 'admin/buttons', 'admin/formtastic', 'admin/formtastic_changes', 'admin/application', :media => 'screen', :cache => Rails.env.production?
|
||||||
|
|
||||||
|
@ -1,7 +1,11 @@
|
|||||||
%li{ :class => "asset #{'last' if (asset_counter + 1) % 6 == 0}"}
|
- per_row ||= 6
|
||||||
%h4= link_to truncate(asset.slug, :length => 22), edit_admin_theme_asset_path(asset)
|
- edit_mode = defined?(edit).nil? ? true: edit
|
||||||
|
|
||||||
|
%li{ :class => "asset #{'last' if (asset_counter + 1) % per_row == 0}"}
|
||||||
|
%h4= link_to truncate(asset.slug, :length => 22), edit_mode ? edit_admin_theme_asset_path(asset) : asset.source.url
|
||||||
.image
|
.image
|
||||||
.inside
|
.inside
|
||||||
= vignette_tag(asset)
|
= vignette_tag(asset)
|
||||||
.actions
|
- if edit_mode
|
||||||
= link_to image_tag('admin/list/icons/cross.png'), admin_theme_asset_path(asset), :class => 'remove', :confirm => t('admin.messages.confirm'), :method => :delete
|
.actions
|
||||||
|
= link_to image_tag('admin/list/icons/cross.png'), admin_theme_asset_path(asset), :class => 'remove', :confirm => t('admin.messages.confirm'), :method => :delete
|
@ -1,5 +1,6 @@
|
|||||||
- content_for :head do
|
- content_for :head do
|
||||||
= javascript_include_tag 'admin/plugins/codemirror/codemirror', 'admin/theme_assets.js'
|
= javascript_include_tag 'admin/plugins/codemirror/codemirror', 'admin/plugins/fancybox', 'admin/plugins/plupload/plupload.full.js', 'admin/theme_assets.js'
|
||||||
|
= stylesheet_link_tag 'admin/plugins/fancybox', 'admin/box'
|
||||||
|
|
||||||
= f.hidden_field :performing_plain_text
|
= f.hidden_field :performing_plain_text
|
||||||
|
|
||||||
@ -22,12 +23,14 @@
|
|||||||
= f.select :content_type, ["stylesheet", "javascript"]
|
= f.select :content_type, ["stylesheet", "javascript"]
|
||||||
|
|
||||||
= f.custom_input :plain_text, :css => 'full', :with_label => false do
|
= f.custom_input :plain_text, :css => 'full', :with_label => false do
|
||||||
%code{ :class => (@asset.new_record? || (@asset.size && @asset.size > 40000) ? 'nude' : @asset.content_type) }
|
%code{ :class => (@asset.size && @asset.size > 40000 ? 'nude' : (@asset.content_type || 'stylesheet')) }
|
||||||
= f.text_area :plain_text
|
= f.text_area :plain_text
|
||||||
|
%li.more
|
||||||
|
= link_to t('.picker_link'), admin_theme_assets_path, :id => 'asset-picker-link'
|
||||||
|
|
||||||
%span.alt
|
%span.alt
|
||||||
= t('admin.theme_assets.form.choose_file')
|
= t('admin.theme_assets.form.choose_file')
|
||||||
|
|
||||||
- if @asset.image?
|
- if @asset.image?
|
||||||
= f.foldable_inputs :name => "#{t('formtastic.titles.preview')} #{image_dimensions_and_size(@asset)}", :class => 'preview' do
|
= f.foldable_inputs :name => "#{t('formtastic.titles.preview')} #{image_dimensions_and_size(@asset)}", :class => 'preview' do
|
||||||
%li
|
%li
|
||||||
|
12
app/views/admin/theme_assets/images.html.haml
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
#theme-images.asset-picker
|
||||||
|
%h2= t('.title')
|
||||||
|
|
||||||
|
.actions
|
||||||
|
= admin_button_tag t('admin.theme_assets.index.new'), admin_theme_assets_url(:json), :class => 'button small add', :id => 'upload-link'
|
||||||
|
|
||||||
|
- if @image_assets.empty?
|
||||||
|
%p.no-items= t('.no_items')
|
||||||
|
- else
|
||||||
|
%ul.assets
|
||||||
|
= render :partial => 'asset', :collection => @image_assets, :locals => { :per_row => 3, :edit => false }
|
||||||
|
%li.clear
|
@ -29,7 +29,7 @@ module Mongoid #:nodoc:
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
Rails.logger.debug "conditions = #{conditions.inspect} / #{options[:scope].inspect}"
|
# Rails.logger.debug "conditions = #{conditions.inspect} / #{options[:scope].inspect}"
|
||||||
|
|
||||||
return if document.class.where(conditions).empty?
|
return if document.class.where(conditions).empty?
|
||||||
|
|
||||||
|
18
doc/TODO
@ -1,16 +1,21 @@
|
|||||||
BOARD:
|
BOARD:
|
||||||
- theme assets
|
|
||||||
- theme assets picker (???)
|
- theme assets picker (???)
|
||||||
|
x lightbox (http://fancybox.net/api)
|
||||||
- assets uploader: remove old files if new one
|
x select it
|
||||||
|
- flash upload (http://www.plupload.com/example_custom.php)
|
||||||
- theme assets: disable version if not image
|
- theme assets: disable version if not image
|
||||||
|
- refactor theme assets / assets uploaders
|
||||||
|
|
||||||
BACKLOG:
|
BACKLOG:
|
||||||
|
|
||||||
- devise messages in French
|
- devise messages in French
|
||||||
- localize devise emails
|
- localize devise emails
|
||||||
|
|
||||||
|
- cucumber features for admin pages
|
||||||
- refactoring admin crud (pages + layouts + snippets)
|
- refactoring admin crud (pages + layouts + snippets)
|
||||||
|
|
||||||
|
- icons for mime type
|
||||||
|
|
||||||
- new types for custom field
|
- new types for custom field
|
||||||
- file
|
- file
|
||||||
- boolean
|
- boolean
|
||||||
@ -18,11 +23,14 @@ BACKLOG:
|
|||||||
- refactoring: CustomFields::CustomField => CustomFields::Field
|
- refactoring: CustomFields::CustomField => CustomFields::Field
|
||||||
- optimization custom_fields: use dynamic class for a collection instead of modifying the metaclass each time we build an item
|
- optimization custom_fields: use dynamic class for a collection instead of modifying the metaclass each time we build an item
|
||||||
|
|
||||||
|
- deploy on Heroku
|
||||||
|
|
||||||
BUGS:
|
BUGS:
|
||||||
- when assigning new layout, disabled parts show up :-( (js problem)
|
- when assigning new layout, disabled parts show up :-( (js problem)
|
||||||
|
|
||||||
NICE TO HAVE:
|
NICE TO HAVE:
|
||||||
- asset collections: custom resizing if image
|
- asset collections: custom resizing if image
|
||||||
|
- super_finder
|
||||||
|
|
||||||
DONE:
|
DONE:
|
||||||
x admin layout
|
x admin layout
|
||||||
@ -91,4 +99,6 @@ x liquid rendering engine
|
|||||||
x contents pagination
|
x contents pagination
|
||||||
x how to disable a page part in layout ? (BUG)
|
x how to disable a page part in layout ? (BUG)
|
||||||
x non published page (redirect to 404 ?)
|
x non published page (redirect to 404 ?)
|
||||||
x refactoring page.rb => create module pagetree
|
x refactoring page.rb => create module pagetree
|
||||||
|
! assets uploader: remove old files if new one (BUG non )
|
||||||
|
x CodeMirror: switch js -> css -> js .... (http://marijn.haverbeke.nl/codemirror/manual.html)
|
@ -3,6 +3,7 @@ require 'locomotive/configuration'
|
|||||||
require 'locomotive/liquid'
|
require 'locomotive/liquid'
|
||||||
require 'locomotive/mongoid'
|
require 'locomotive/mongoid'
|
||||||
|
|
||||||
|
|
||||||
module Locomotive
|
module Locomotive
|
||||||
|
|
||||||
class << self
|
class << self
|
||||||
|
@ -1 +1,2 @@
|
|||||||
require 'locomotive/mongoid/document'
|
require 'locomotive/mongoid/document'
|
||||||
|
require 'locomotive/mongoid/model_extensions'
|
BIN
public/images/admin/plugins/fancybox/blank.gif
Executable file
After Width: | Height: | Size: 43 B |
BIN
public/images/admin/plugins/fancybox/fancy_close.png
Executable file
After Width: | Height: | Size: 1.5 KiB |
BIN
public/images/admin/plugins/fancybox/fancy_loading.png
Executable file
After Width: | Height: | Size: 10 KiB |
BIN
public/images/admin/plugins/fancybox/fancy_nav_left.png
Executable file
After Width: | Height: | Size: 1.4 KiB |
BIN
public/images/admin/plugins/fancybox/fancy_nav_right.png
Executable file
After Width: | Height: | Size: 1.4 KiB |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_e.png
Executable file
After Width: | Height: | Size: 107 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_n.png
Executable file
After Width: | Height: | Size: 106 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_ne.png
Executable file
After Width: | Height: | Size: 347 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_nw.png
Executable file
After Width: | Height: | Size: 324 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_s.png
Executable file
After Width: | Height: | Size: 111 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_se.png
Executable file
After Width: | Height: | Size: 352 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_sw.png
Executable file
After Width: | Height: | Size: 340 B |
BIN
public/images/admin/plugins/fancybox/fancy_shadow_w.png
Executable file
After Width: | Height: | Size: 103 B |
BIN
public/images/admin/plugins/fancybox/fancy_title_left.png
Executable file
After Width: | Height: | Size: 503 B |
BIN
public/images/admin/plugins/fancybox/fancy_title_main.png
Executable file
After Width: | Height: | Size: 96 B |
BIN
public/images/admin/plugins/fancybox/fancy_title_over.png
Executable file
After Width: | Height: | Size: 70 B |
BIN
public/images/admin/plugins/fancybox/fancy_title_right.png
Executable file
After Width: | Height: | Size: 506 B |
BIN
public/images/admin/plugins/fancybox/fancybox-x.png
Executable file
After Width: | Height: | Size: 203 B |
BIN
public/images/admin/plugins/fancybox/fancybox-y.png
Executable file
After Width: | Height: | Size: 176 B |
BIN
public/images/admin/plugins/fancybox/fancybox.png
Executable file
After Width: | Height: | Size: 15 KiB |
@ -21,12 +21,15 @@ $.growl.settings.dockCss = {
|
|||||||
var addCodeMirrorEditor = function(type, el, parser) {
|
var addCodeMirrorEditor = function(type, el, parser) {
|
||||||
var parserfile = "parse" + type + ".js";
|
var parserfile = "parse" + type + ".js";
|
||||||
if (parser != undefined) parserfile = parser;
|
if (parser != undefined) parserfile = parser;
|
||||||
if (type == 'liquid') type = 'xml';
|
// if (type == 'liquid') type = 'xml';
|
||||||
|
|
||||||
var editor = CodeMirror.fromTextArea(el.attr('id'), {
|
var editor = CodeMirror.fromTextArea(el.attr('id'), {
|
||||||
height: "400px",
|
height: "400px",
|
||||||
parserfile: parserfile,
|
parserfile: parserfile,
|
||||||
stylesheet: ["/stylesheets/admin/plugins/codemirror/" + type + "colors.css", "/stylesheets/admin/plugins/codemirror/liquidcolors.css"],
|
stylesheet: [
|
||||||
|
"/stylesheets/admin/plugins/codemirror/csscolors.css",
|
||||||
|
"/stylesheets/admin/plugins/codemirror/javascriptcolors.css",
|
||||||
|
"/stylesheets/admin/plugins/codemirror/liquidcolors.css"],
|
||||||
path: "/javascripts/admin/plugins/codemirror/",
|
path: "/javascripts/admin/plugins/codemirror/",
|
||||||
continuousScanning: 500,
|
continuousScanning: 500,
|
||||||
reindentOnLoad: true,
|
reindentOnLoad: true,
|
||||||
|
@ -33,6 +33,8 @@ var CodeMirror = (function(){
|
|||||||
iframeClass: null,
|
iframeClass: null,
|
||||||
passDelay: 200,
|
passDelay: 200,
|
||||||
passTime: 50,
|
passTime: 50,
|
||||||
|
lineNumberDelay: 200,
|
||||||
|
lineNumberTime: 50,
|
||||||
continuousScanning: false,
|
continuousScanning: false,
|
||||||
saveFunction: null,
|
saveFunction: null,
|
||||||
onChange: null,
|
onChange: null,
|
||||||
@ -41,7 +43,7 @@ var CodeMirror = (function(){
|
|||||||
disableSpellcheck: true,
|
disableSpellcheck: true,
|
||||||
textWrapping: true,
|
textWrapping: true,
|
||||||
readOnly: false,
|
readOnly: false,
|
||||||
width: "100%",
|
width: "",
|
||||||
height: "300px",
|
height: "300px",
|
||||||
autoMatchParens: false,
|
autoMatchParens: false,
|
||||||
parserConfig: null,
|
parserConfig: null,
|
||||||
@ -50,67 +52,49 @@ var CodeMirror = (function(){
|
|||||||
activeTokens: null,
|
activeTokens: null,
|
||||||
cursorActivity: null,
|
cursorActivity: null,
|
||||||
lineNumbers: false,
|
lineNumbers: false,
|
||||||
indentUnit: 2
|
indentUnit: 2,
|
||||||
|
domain: null
|
||||||
});
|
});
|
||||||
|
|
||||||
function wrapLineNumberDiv(place) {
|
function addLineNumberDiv(container) {
|
||||||
return function(node) {
|
var nums = document.createElement("DIV"),
|
||||||
var container = document.createElement("DIV"),
|
scroller = document.createElement("DIV");
|
||||||
nums = document.createElement("DIV"),
|
nums.style.position = "absolute";
|
||||||
scroller = document.createElement("DIV");
|
nums.style.height = "100%";
|
||||||
container.style.position = "relative";
|
if (nums.style.setExpression) {
|
||||||
nums.style.position = "absolute";
|
try {nums.style.setExpression("height", "this.previousSibling.offsetHeight + 'px'");}
|
||||||
nums.style.height = "100%";
|
catch(e) {} // Seems to throw 'Not Implemented' on some IE8 versions
|
||||||
if (nums.style.setExpression) {
|
|
||||||
try {nums.style.setExpression("height", "this.previousSibling.offsetHeight + 'px'");}
|
|
||||||
catch(e) {} // Seems to throw 'Not Implemented' on some IE8 versions
|
|
||||||
}
|
|
||||||
nums.style.top = "0px";
|
|
||||||
nums.style.overflow = "hidden";
|
|
||||||
place(container);
|
|
||||||
container.appendChild(node);
|
|
||||||
container.appendChild(nums);
|
|
||||||
scroller.className = "CodeMirror-line-numbers";
|
|
||||||
nums.appendChild(scroller);
|
|
||||||
}
|
}
|
||||||
|
nums.style.top = "0px";
|
||||||
|
nums.style.overflow = "hidden";
|
||||||
|
container.appendChild(nums);
|
||||||
|
scroller.className = "CodeMirror-line-numbers";
|
||||||
|
nums.appendChild(scroller);
|
||||||
|
return nums;
|
||||||
}
|
}
|
||||||
|
|
||||||
function applyLineNumbers(frame) {
|
function frameHTML(options) {
|
||||||
var win = frame.contentWindow, doc = win.document,
|
if (typeof options.parserfile == "string")
|
||||||
nums = frame.nextSibling, scroller = nums.firstChild;
|
options.parserfile = [options.parserfile];
|
||||||
|
if (typeof options.stylesheet == "string")
|
||||||
|
options.stylesheet = [options.stylesheet];
|
||||||
|
|
||||||
var nextNum = 1, barWidth = null;
|
var html = ["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\" \"http://www.w3.org/TR/html4/loose.dtd\"><html><head>"];
|
||||||
function sizeBar() {
|
// Hack to work around a bunch of IE8-specific problems.
|
||||||
if (!frame.offsetWidth || !win.Editor) {
|
html.push("<meta http-equiv=\"X-UA-Compatible\" content=\"IE=EmulateIE7\"/>");
|
||||||
for (var cur = frame; cur.parentNode; cur = cur.parentNode) {
|
forEach(options.stylesheet, function(file) {
|
||||||
if (cur != document) {
|
html.push("<link rel=\"stylesheet\" type=\"text/css\" href=\"" + file + "\"/>");
|
||||||
clearInterval(sizeInterval);
|
});
|
||||||
return;
|
forEach(options.basefiles.concat(options.parserfile), function(file) {
|
||||||
}
|
html.push("<script type=\"text/javascript\" src=\"" + options.path + file + "\"><" + "/script>");
|
||||||
}
|
});
|
||||||
}
|
html.push("</head><body style=\"border-width: 0;\" class=\"editbox\" spellcheck=\"" +
|
||||||
|
(options.disableSpellcheck ? "false" : "true") + "\"></body></html>");
|
||||||
if (nums.offsetWidth != barWidth) {
|
return html.join("");
|
||||||
barWidth = nums.offsetWidth;
|
|
||||||
nums.style.left = "-" + (frame.parentNode.style.marginLeft = barWidth + "px");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
function update() {
|
|
||||||
var diff = 20 + Math.max(doc.body.offsetHeight, frame.offsetHeight) - scroller.offsetHeight;
|
|
||||||
for (var n = Math.ceil(diff / 10); n > 0; n--) {
|
|
||||||
var div = document.createElement("DIV");
|
|
||||||
div.appendChild(document.createTextNode(nextNum++));
|
|
||||||
scroller.appendChild(div);
|
|
||||||
}
|
|
||||||
nums.scrollTop = doc.body.scrollTop || doc.documentElement.scrollTop || 0;
|
|
||||||
}
|
|
||||||
sizeBar();
|
|
||||||
update();
|
|
||||||
win.addEventHandler(win, "scroll", update);
|
|
||||||
win.addEventHandler(win, "resize", update);
|
|
||||||
var sizeInterval = setInterval(sizeBar, 500);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var internetExplorer = document.selection && window.ActiveXObject && /MSIE/.test(navigator.userAgent);
|
||||||
|
|
||||||
function CodeMirror(place, options) {
|
function CodeMirror(place, options) {
|
||||||
// Backward compatibility for deprecated options.
|
// Backward compatibility for deprecated options.
|
||||||
if (options.dumbTabs) options.tabMode = "spaces";
|
if (options.dumbTabs) options.tabMode = "spaces";
|
||||||
@ -123,66 +107,72 @@ var CodeMirror = (function(){
|
|||||||
var frame = this.frame = document.createElement("IFRAME");
|
var frame = this.frame = document.createElement("IFRAME");
|
||||||
if (options.iframeClass) frame.className = options.iframeClass;
|
if (options.iframeClass) frame.className = options.iframeClass;
|
||||||
frame.frameBorder = 0;
|
frame.frameBorder = 0;
|
||||||
frame.src = "javascript:false;";
|
|
||||||
frame.style.border = "0";
|
frame.style.border = "0";
|
||||||
frame.style.width = options.width;
|
frame.style.width = '100%';
|
||||||
frame.style.height = options.height;
|
frame.style.height = '100%';
|
||||||
// display: block occasionally suppresses some Firefox bugs, so we
|
// display: block occasionally suppresses some Firefox bugs, so we
|
||||||
// always add it, redundant as it sounds.
|
// always add it, redundant as it sounds.
|
||||||
frame.style.display = "block";
|
frame.style.display = "block";
|
||||||
|
|
||||||
if (place.appendChild) {
|
var div = this.wrapping = document.createElement("DIV");
|
||||||
var node = place;
|
div.style.position = "relative";
|
||||||
place = function(n){node.appendChild(n);};
|
div.className = "CodeMirror-wrapping";
|
||||||
}
|
div.style.width = options.width;
|
||||||
if (options.lineNumbers) place = wrapLineNumberDiv(place);
|
div.style.height = options.height;
|
||||||
place(frame);
|
// This is used by Editor.reroutePasteEvent
|
||||||
|
var teHack = this.textareaHack = document.createElement("TEXTAREA");
|
||||||
|
div.appendChild(teHack);
|
||||||
|
teHack.style.position = "absolute";
|
||||||
|
teHack.style.left = "-10000px";
|
||||||
|
teHack.style.width = "10px";
|
||||||
|
|
||||||
// Link back to this object, so that the editor can fetch options
|
// Link back to this object, so that the editor can fetch options
|
||||||
// and add a reference to itself.
|
// and add a reference to itself.
|
||||||
frame.CodeMirror = this;
|
frame.CodeMirror = this;
|
||||||
|
if (options.domain && internetExplorer) {
|
||||||
|
this.html = frameHTML(options);
|
||||||
|
frame.src = "javascript:(function(){document.open();" +
|
||||||
|
(options.domain ? "document.domain=\"" + options.domain + "\";" : "") +
|
||||||
|
"document.write(window.frameElement.CodeMirror.html);document.close();})()";
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
frame.src = "javascript:false";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (place.appendChild) place.appendChild(div);
|
||||||
|
else place(div);
|
||||||
|
div.appendChild(frame);
|
||||||
|
if (options.lineNumbers) this.lineNumbers = addLineNumberDiv(div);
|
||||||
|
|
||||||
this.win = frame.contentWindow;
|
this.win = frame.contentWindow;
|
||||||
|
if (!options.domain || !internetExplorer) {
|
||||||
if (typeof options.parserfile == "string")
|
this.win.document.open();
|
||||||
options.parserfile = [options.parserfile];
|
this.win.document.write(frameHTML(options));
|
||||||
if (typeof options.stylesheet == "string")
|
this.win.document.close();
|
||||||
options.stylesheet = [options.stylesheet];
|
}
|
||||||
|
|
||||||
var html = ["<!DOCTYPE HTML PUBLIC \"-//W3C//DTD HTML 4.0 Transitional//EN\" \"http://www.w3.org/TR/html4/loose.dtd\"><html><head>"];
|
|
||||||
// Hack to work around a bunch of IE8-specific problems.
|
|
||||||
html.push("<meta http-equiv=\"X-UA-Compatible\" content=\"IE=EmulateIE7\"/>");
|
|
||||||
forEach(options.stylesheet, function(file) {
|
|
||||||
html.push("<link rel=\"stylesheet\" type=\"text/css\" href=\"" + file + "\"/>");
|
|
||||||
});
|
|
||||||
forEach(options.basefiles.concat(options.parserfile), function(file) {
|
|
||||||
html.push("<script type=\"text/javascript\" src=\"" + options.path + file + "\"></script>");
|
|
||||||
});
|
|
||||||
html.push("</head><body style=\"border-width: 0;\" class=\"editbox\" spellcheck=\"" +
|
|
||||||
(options.disableSpellcheck ? "false" : "true") + "\"></body></html>");
|
|
||||||
|
|
||||||
var doc = this.win.document;
|
|
||||||
doc.open();
|
|
||||||
doc.write(html.join(""));
|
|
||||||
doc.close();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CodeMirror.prototype = {
|
CodeMirror.prototype = {
|
||||||
init: function() {
|
init: function() {
|
||||||
if (this.options.initCallback) this.options.initCallback(this);
|
if (this.options.initCallback) this.options.initCallback(this);
|
||||||
if (this.options.lineNumbers) applyLineNumbers(this.frame);
|
if (this.options.lineNumbers) this.activateLineNumbers();
|
||||||
if (this.options.reindentOnLoad) this.reindent();
|
if (this.options.reindentOnLoad) this.reindent();
|
||||||
},
|
},
|
||||||
|
|
||||||
getCode: function() {return this.editor.getCode();},
|
getCode: function() {return this.editor.getCode();},
|
||||||
setCode: function(code) {this.editor.importCode(code);},
|
setCode: function(code) {this.editor.importCode(code);},
|
||||||
selection: function() {return this.editor.selectedText();},
|
selection: function() {this.focusIfIE(); return this.editor.selectedText();},
|
||||||
reindent: function() {this.editor.reindent();},
|
reindent: function() {this.editor.reindent();},
|
||||||
reindentSelection: function() {this.editor.reindentSelection(null);},
|
reindentSelection: function() {this.focusIfIE(); this.editor.reindentSelection(null);},
|
||||||
|
|
||||||
|
focusIfIE: function() {
|
||||||
|
// in IE, a lot of selection-related functionality only works when the frame is focused
|
||||||
|
if (this.win.select.ie_selection) this.focus();
|
||||||
|
},
|
||||||
focus: function() {
|
focus: function() {
|
||||||
this.win.focus();
|
this.win.focus();
|
||||||
if (this.editor.selectionSnapshot) // IE hack
|
if (this.editor.selectionSnapshot) // IE hack
|
||||||
this.win.select.selectCoords(this.win, this.editor.selectionSnapshot);
|
this.win.select.setBookmark(this.win.document.body, this.editor.selectionSnapshot);
|
||||||
},
|
},
|
||||||
replaceSelection: function(text) {
|
replaceSelection: function(text) {
|
||||||
this.focus();
|
this.focus();
|
||||||
@ -192,8 +182,8 @@ var CodeMirror = (function(){
|
|||||||
replaceChars: function(text, start, end) {
|
replaceChars: function(text, start, end) {
|
||||||
this.editor.replaceChars(text, start, end);
|
this.editor.replaceChars(text, start, end);
|
||||||
},
|
},
|
||||||
getSearchCursor: function(string, fromCursor) {
|
getSearchCursor: function(string, fromCursor, caseFold) {
|
||||||
return this.editor.getSearchCursor(string, fromCursor);
|
return this.editor.getSearchCursor(string, fromCursor, caseFold);
|
||||||
},
|
},
|
||||||
|
|
||||||
undo: function() {this.editor.history.undo();},
|
undo: function() {this.editor.history.undo();},
|
||||||
@ -204,18 +194,76 @@ var CodeMirror = (function(){
|
|||||||
grabKeys: function(callback, filter) {this.editor.grabKeys(callback, filter);},
|
grabKeys: function(callback, filter) {this.editor.grabKeys(callback, filter);},
|
||||||
ungrabKeys: function() {this.editor.ungrabKeys();},
|
ungrabKeys: function() {this.editor.ungrabKeys();},
|
||||||
|
|
||||||
setParser: function(name) {this.editor.setParser(name);},
|
setParser: function(name) { this.editor.setParser(name); },
|
||||||
|
setSpellcheck: function(on) {this.win.document.body.spellcheck = on;},
|
||||||
cursorPosition: function(start) {
|
setStylesheet: function(names) {
|
||||||
if (this.win.select.ie_selection) this.focus();
|
if (typeof names === "string") names = [names];
|
||||||
return this.editor.cursorPosition(start);
|
var activeStylesheets = {};
|
||||||
|
var matchedNames = {};
|
||||||
|
var links = this.win.document.getElementsByTagName("link");
|
||||||
|
// Create hashes of active stylesheets and matched names.
|
||||||
|
// This is O(n^2) but n is expected to be very small.
|
||||||
|
for (var x = 0, link; link = links[x]; x++) {
|
||||||
|
if (link.rel.indexOf("stylesheet") !== -1) {
|
||||||
|
for (var y = 0; y < names.length; y++) {
|
||||||
|
var name = names[y];
|
||||||
|
if (link.href.substring(link.href.length - name.length) === name) {
|
||||||
|
activeStylesheets[link.href] = true;
|
||||||
|
matchedNames[name] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Activate the selected stylesheets and disable the rest.
|
||||||
|
for (var x = 0, link; link = links[x]; x++) {
|
||||||
|
if (link.rel.indexOf("stylesheet") !== -1) {
|
||||||
|
link.disabled = !(link.href in activeStylesheets);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// Create any new stylesheets.
|
||||||
|
for (var y = 0; y < names.length; y++) {
|
||||||
|
var name = names[y];
|
||||||
|
if (!(name in matchedNames)) {
|
||||||
|
var link = this.win.document.createElement("link");
|
||||||
|
link.rel = "stylesheet";
|
||||||
|
link.type = "text/css";
|
||||||
|
link.href = name;
|
||||||
|
this.win.document.getElementsByTagName('head')[0].appendChild(link);
|
||||||
|
}
|
||||||
|
}
|
||||||
},
|
},
|
||||||
|
setTextWrapping: function(on) {
|
||||||
|
if (on == this.options.textWrapping) return;
|
||||||
|
this.win.document.body.style.whiteSpace = on ? "" : "nowrap";
|
||||||
|
this.options.textWrapping = on;
|
||||||
|
if (this.lineNumbers) {
|
||||||
|
this.setLineNumbers(false);
|
||||||
|
this.setLineNumbers(true);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
setIndentUnit: function(unit) {this.win.indentUnit = unit;},
|
||||||
|
setUndoDepth: function(depth) {this.editor.history.maxDepth = depth;},
|
||||||
|
setTabMode: function(mode) {this.options.tabMode = mode;},
|
||||||
|
setLineNumbers: function(on) {
|
||||||
|
if (on && !this.lineNumbers) {
|
||||||
|
this.lineNumbers = addLineNumberDiv(this.wrapping);
|
||||||
|
this.activateLineNumbers();
|
||||||
|
}
|
||||||
|
else if (!on && this.lineNumbers) {
|
||||||
|
this.wrapping.removeChild(this.lineNumbers);
|
||||||
|
this.wrapping.style.marginLeft = "";
|
||||||
|
this.lineNumbers = null;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
cursorPosition: function(start) {this.focusIfIE(); return this.editor.cursorPosition(start);},
|
||||||
firstLine: function() {return this.editor.firstLine();},
|
firstLine: function() {return this.editor.firstLine();},
|
||||||
lastLine: function() {return this.editor.lastLine();},
|
lastLine: function() {return this.editor.lastLine();},
|
||||||
nextLine: function(line) {return this.editor.nextLine(line);},
|
nextLine: function(line) {return this.editor.nextLine(line);},
|
||||||
prevLine: function(line) {return this.editor.prevLine(line);},
|
prevLine: function(line) {return this.editor.prevLine(line);},
|
||||||
lineContent: function(line) {return this.editor.lineContent(line);},
|
lineContent: function(line) {return this.editor.lineContent(line);},
|
||||||
setLineContent: function(line, content) {this.editor.setLineContent(line, content);},
|
setLineContent: function(line, content) {this.editor.setLineContent(line, content);},
|
||||||
|
removeLine: function(line){this.editor.removeLine(line);},
|
||||||
insertIntoLine: function(line, position, content) {this.editor.insertIntoLine(line, position, content);},
|
insertIntoLine: function(line, position, content) {this.editor.insertIntoLine(line, position, content);},
|
||||||
selectLines: function(startLine, startOffset, endLine, endOffset) {
|
selectLines: function(startLine, startOffset, endLine, endOffset) {
|
||||||
this.win.focus();
|
this.win.focus();
|
||||||
@ -235,14 +283,125 @@ var CodeMirror = (function(){
|
|||||||
}
|
}
|
||||||
return num;
|
return num;
|
||||||
},
|
},
|
||||||
|
jumpToLine: function(line) {
|
||||||
// Old number-based line interface
|
if (typeof line == "number") line = this.nthLine(line);
|
||||||
jumpToLine: function(n) {
|
this.selectLines(line, 0);
|
||||||
this.selectLines(this.nthLine(n), 0);
|
|
||||||
this.win.focus();
|
this.win.focus();
|
||||||
},
|
},
|
||||||
currentLine: function() {
|
currentLine: function() { // Deprecated, but still there for backward compatibility
|
||||||
return this.lineNumber(this.cursorPosition().line);
|
return this.lineNumber(this.cursorLine());
|
||||||
|
},
|
||||||
|
cursorLine: function() {
|
||||||
|
return this.cursorPosition().line;
|
||||||
|
},
|
||||||
|
|
||||||
|
activateLineNumbers: function() {
|
||||||
|
var frame = this.frame, win = frame.contentWindow, doc = win.document, body = doc.body,
|
||||||
|
nums = this.lineNumbers, scroller = nums.firstChild, self = this;
|
||||||
|
var barWidth = null;
|
||||||
|
|
||||||
|
function sizeBar() {
|
||||||
|
if (frame.offsetWidth == 0) return;
|
||||||
|
for (var root = frame; root.parentNode; root = root.parentNode);
|
||||||
|
if (!nums.parentNode || root != document || !win.Editor) {
|
||||||
|
// Clear event handlers (their nodes might already be collected, so try/catch)
|
||||||
|
try{clear();}catch(e){}
|
||||||
|
clearInterval(sizeInterval);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (nums.offsetWidth != barWidth) {
|
||||||
|
barWidth = nums.offsetWidth;
|
||||||
|
nums.style.left = "-" + (frame.parentNode.style.marginLeft = barWidth + "px");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function doScroll() {
|
||||||
|
nums.scrollTop = body.scrollTop || doc.documentElement.scrollTop || 0;
|
||||||
|
}
|
||||||
|
// Cleanup function, registered by nonWrapping and wrapping.
|
||||||
|
var clear = function(){};
|
||||||
|
sizeBar();
|
||||||
|
var sizeInterval = setInterval(sizeBar, 500);
|
||||||
|
|
||||||
|
function nonWrapping() {
|
||||||
|
var nextNum = 1, pending;
|
||||||
|
function update() {
|
||||||
|
var target = 50 + Math.max(body.offsetHeight, Math.max(frame.offsetHeight, body.scrollHeight || 0));
|
||||||
|
var endTime = new Date().getTime() + self.options.lineNumberTime;
|
||||||
|
while (scroller.offsetHeight < target && (!scroller.firstChild || scroller.offsetHeight)) {
|
||||||
|
scroller.appendChild(document.createElement("DIV"));
|
||||||
|
scroller.lastChild.innerHTML = nextNum++;
|
||||||
|
if (new Date().getTime() > endTime) {
|
||||||
|
if (pending) clearTimeout(pending);
|
||||||
|
pending = setTimeout(update, self.options.lineNumberDelay);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
doScroll();
|
||||||
|
}
|
||||||
|
var onScroll = win.addEventHandler(win, "scroll", update, true),
|
||||||
|
onResize = win.addEventHandler(win, "resize", update, true);
|
||||||
|
clear = function(){onScroll(); onResize(); if (pending) clearTimeout(pending);};
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
function wrapping() {
|
||||||
|
var node, lineNum, next, pos;
|
||||||
|
|
||||||
|
function addNum(n) {
|
||||||
|
if (!lineNum) lineNum = scroller.appendChild(document.createElement("DIV"));
|
||||||
|
lineNum.innerHTML = n;
|
||||||
|
pos = lineNum.offsetHeight + lineNum.offsetTop;
|
||||||
|
lineNum = lineNum.nextSibling;
|
||||||
|
}
|
||||||
|
function work() {
|
||||||
|
if (!scroller.parentNode || scroller.parentNode != self.lineNumbers) return;
|
||||||
|
|
||||||
|
var endTime = new Date().getTime() + self.options.lineNumberTime;
|
||||||
|
while (node) {
|
||||||
|
addNum(next++);
|
||||||
|
for (; node && !win.isBR(node); node = node.nextSibling) {
|
||||||
|
var bott = node.offsetTop + node.offsetHeight;
|
||||||
|
while (scroller.offsetHeight && bott - 3 > pos) addNum(" ");
|
||||||
|
}
|
||||||
|
if (node) node = node.nextSibling;
|
||||||
|
if (new Date().getTime() > endTime) {
|
||||||
|
pending = setTimeout(work, self.options.lineNumberDelay);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// While there are un-processed number DIVs, or the scroller is smaller than the frame...
|
||||||
|
var target = 50 + Math.max(body.offsetHeight, Math.max(frame.offsetHeight, body.scrollHeight || 0));
|
||||||
|
while (lineNum || (scroller.offsetHeight < target && (!scroller.firstChild || scroller.offsetHeight)))
|
||||||
|
addNum(next++);
|
||||||
|
doScroll();
|
||||||
|
}
|
||||||
|
function start() {
|
||||||
|
doScroll();
|
||||||
|
node = body.firstChild;
|
||||||
|
lineNum = scroller.firstChild;
|
||||||
|
pos = 0;
|
||||||
|
next = 1;
|
||||||
|
work();
|
||||||
|
}
|
||||||
|
|
||||||
|
start();
|
||||||
|
var pending = null;
|
||||||
|
function update() {
|
||||||
|
if (pending) clearTimeout(pending);
|
||||||
|
if (self.editor.allClean()) start();
|
||||||
|
else pending = setTimeout(update, 200);
|
||||||
|
}
|
||||||
|
self.updateNumbers = update;
|
||||||
|
var onScroll = win.addEventHandler(win, "scroll", doScroll, true),
|
||||||
|
onResize = win.addEventHandler(win, "resize", update, true);
|
||||||
|
clear = function(){
|
||||||
|
if (pending) clearTimeout(pending);
|
||||||
|
if (self.updateNumbers == update) self.updateNumbers = null;
|
||||||
|
onScroll();
|
||||||
|
onResize();
|
||||||
|
};
|
||||||
|
}
|
||||||
|
(this.options.textWrapping ? wrapping : nonWrapping)();
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -4,6 +4,11 @@
|
|||||||
* plain sequences of <span> and <br> elements
|
* plain sequences of <span> and <br> elements
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
var internetExplorer = document.selection && window.ActiveXObject && /MSIE/.test(navigator.userAgent);
|
||||||
|
var webkit = /AppleWebKit/.test(navigator.userAgent);
|
||||||
|
var safari = /Apple Computers, Inc/.test(navigator.vendor);
|
||||||
|
var gecko = /gecko\/(\d{8})/i.test(navigator.userAgent);
|
||||||
|
|
||||||
// Make sure a string does not contain two consecutive 'collapseable'
|
// Make sure a string does not contain two consecutive 'collapseable'
|
||||||
// whitespace characters.
|
// whitespace characters.
|
||||||
function makeWhiteSpace(n) {
|
function makeWhiteSpace(n) {
|
||||||
@ -80,13 +85,13 @@ var Editor = (function(){
|
|||||||
if (text.length) leaving = false;
|
if (text.length) leaving = false;
|
||||||
result.push(node);
|
result.push(node);
|
||||||
}
|
}
|
||||||
else if (node.nodeName == "BR" && node.childNodes.length == 0) {
|
else if (isBR(node) && node.childNodes.length == 0) {
|
||||||
leaving = true;
|
leaving = true;
|
||||||
result.push(node);
|
result.push(node);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
forEach(node.childNodes, simplifyNode);
|
forEach(node.childNodes, simplifyNode);
|
||||||
if (!leaving && newlineElements.hasOwnProperty(node.nodeName)) {
|
if (!leaving && newlineElements.hasOwnProperty(node.nodeName.toUpperCase())) {
|
||||||
leaving = true;
|
leaving = true;
|
||||||
if (!atEnd || !top)
|
if (!atEnd || !top)
|
||||||
result.push(doc.createElement("BR"));
|
result.push(doc.createElement("BR"));
|
||||||
@ -106,7 +111,7 @@ var Editor = (function(){
|
|||||||
// See the story.html file for some short remarks about the use of
|
// See the story.html file for some short remarks about the use of
|
||||||
// continuation-passing style in this iterator.
|
// continuation-passing style in this iterator.
|
||||||
function traverseDOM(start){
|
function traverseDOM(start){
|
||||||
function yield(value, c){cc = c; return value;}
|
function _yield(value, c){cc = c; return value;}
|
||||||
function push(fun, arg, c){return function(){return fun(arg, c);};}
|
function push(fun, arg, c){return function(){return fun(arg, c);};}
|
||||||
function stop(){cc = stop; throw StopIteration;};
|
function stop(){cc = stop; throw StopIteration;};
|
||||||
var cc = push(scanNode, start, stop);
|
var cc = push(scanNode, start, stop);
|
||||||
@ -124,6 +129,11 @@ var Editor = (function(){
|
|||||||
}
|
}
|
||||||
var point = null;
|
var point = null;
|
||||||
|
|
||||||
|
// This an Opera-specific hack -- always insert an empty span
|
||||||
|
// between two BRs, because Opera's cursor code gets terribly
|
||||||
|
// confused when the cursor is between two BRs.
|
||||||
|
var afterBR = true;
|
||||||
|
|
||||||
// Insert a normalized node at the current point. If it is a text
|
// Insert a normalized node at the current point. If it is a text
|
||||||
// node, wrap it in a <span>, and give that span a currentText
|
// node, wrap it in a <span>, and give that span a currentText
|
||||||
// property -- this is used to cache the nodeValue, because
|
// property -- this is used to cache the nodeValue, because
|
||||||
@ -136,6 +146,12 @@ var Editor = (function(){
|
|||||||
select.snapshotChanged();
|
select.snapshotChanged();
|
||||||
part = makePartSpan(part, owner);
|
part = makePartSpan(part, owner);
|
||||||
text = part.currentText;
|
text = part.currentText;
|
||||||
|
afterBR = false;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if (afterBR && window.opera)
|
||||||
|
point(makePartSpan("", owner));
|
||||||
|
afterBR = true;
|
||||||
}
|
}
|
||||||
part.dirty = true;
|
part.dirty = true;
|
||||||
nodeQueue.push(part);
|
nodeQueue.push(part);
|
||||||
@ -151,7 +167,7 @@ var Editor = (function(){
|
|||||||
forEach(simplifyDOM(node, end), function(part) {
|
forEach(simplifyDOM(node, end), function(part) {
|
||||||
toYield.push(insertPart(part));
|
toYield.push(insertPart(part));
|
||||||
});
|
});
|
||||||
return yield(toYield.join(""), c);
|
return _yield(toYield.join(""), c);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check whether a node is a normalized <span> element.
|
// Check whether a node is a normalized <span> element.
|
||||||
@ -173,11 +189,15 @@ var Editor = (function(){
|
|||||||
|
|
||||||
if (partNode(node)){
|
if (partNode(node)){
|
||||||
nodeQueue.push(node);
|
nodeQueue.push(node);
|
||||||
return yield(node.currentText, c);
|
afterBR = false;
|
||||||
|
return _yield(node.currentText, c);
|
||||||
}
|
}
|
||||||
else if (node.nodeName == "BR") {
|
else if (isBR(node)) {
|
||||||
|
if (afterBR && window.opera)
|
||||||
|
node.parentNode.insertBefore(makePartSpan("", owner), node);
|
||||||
nodeQueue.push(node);
|
nodeQueue.push(node);
|
||||||
return yield("\n", c);
|
afterBR = true;
|
||||||
|
return _yield("\n", c);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
var end = !node.nextSibling;
|
var end = !node.nextSibling;
|
||||||
@ -195,23 +215,20 @@ var Editor = (function(){
|
|||||||
|
|
||||||
// Determine the text size of a processed node.
|
// Determine the text size of a processed node.
|
||||||
function nodeSize(node) {
|
function nodeSize(node) {
|
||||||
if (node.nodeName == "BR")
|
return isBR(node) ? 1 : node.currentText.length;
|
||||||
return 1;
|
|
||||||
else
|
|
||||||
return node.currentText.length;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Search backwards through the top-level nodes until the next BR or
|
// Search backwards through the top-level nodes until the next BR or
|
||||||
// the start of the frame.
|
// the start of the frame.
|
||||||
function startOfLine(node) {
|
function startOfLine(node) {
|
||||||
while (node && node.nodeName != "BR") node = node.previousSibling;
|
while (node && !isBR(node)) node = node.previousSibling;
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
function endOfLine(node, container) {
|
function endOfLine(node, container) {
|
||||||
if (!node) node = container.firstChild;
|
if (!node) node = container.firstChild;
|
||||||
else if (node.nodeName == "BR") node = node.nextSibling;
|
else if (isBR(node)) node = node.nextSibling;
|
||||||
|
|
||||||
while (node && node.nodeName != "BR") node = node.nextSibling;
|
while (node && !isBR(node)) node = node.nextSibling;
|
||||||
return node;
|
return node;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -223,8 +240,10 @@ var Editor = (function(){
|
|||||||
// indicating whether anything was found, and can be called again to
|
// indicating whether anything was found, and can be called again to
|
||||||
// skip to the next find. Use the select and replace methods to
|
// skip to the next find. Use the select and replace methods to
|
||||||
// actually do something with the found locations.
|
// actually do something with the found locations.
|
||||||
function SearchCursor(editor, string, fromCursor) {
|
function SearchCursor(editor, string, fromCursor, caseFold) {
|
||||||
this.editor = editor;
|
this.editor = editor;
|
||||||
|
this.caseFold = caseFold;
|
||||||
|
if (caseFold) string = string.toLowerCase();
|
||||||
this.history = editor.history;
|
this.history = editor.history;
|
||||||
this.history.commit();
|
this.history.commit();
|
||||||
|
|
||||||
@ -252,7 +271,8 @@ var Editor = (function(){
|
|||||||
// For one-line strings, searching can be done simply by calling
|
// For one-line strings, searching can be done simply by calling
|
||||||
// indexOf on the current line.
|
// indexOf on the current line.
|
||||||
function() {
|
function() {
|
||||||
var match = cleanText(self.history.textAfter(self.line).slice(self.offset)).indexOf(string);
|
var line = cleanText(self.history.textAfter(self.line).slice(self.offset));
|
||||||
|
var match = (self.caseFold ? line.toLowerCase() : line).indexOf(string);
|
||||||
if (match > -1)
|
if (match > -1)
|
||||||
return {from: {node: self.line, offset: self.offset + match},
|
return {from: {node: self.line, offset: self.offset + match},
|
||||||
to: {node: self.line, offset: self.offset + match + string.length}};
|
to: {node: self.line, offset: self.offset + match + string.length}};
|
||||||
@ -262,19 +282,21 @@ var Editor = (function(){
|
|||||||
// end of the line and the last match starts at the start.
|
// end of the line and the last match starts at the start.
|
||||||
function() {
|
function() {
|
||||||
var firstLine = cleanText(self.history.textAfter(self.line).slice(self.offset));
|
var firstLine = cleanText(self.history.textAfter(self.line).slice(self.offset));
|
||||||
var match = firstLine.lastIndexOf(target[0]);
|
var match = (self.caseFold ? firstLine.toLowerCase() : firstLine).lastIndexOf(target[0]);
|
||||||
if (match == -1 || match != firstLine.length - target[0].length)
|
if (match == -1 || match != firstLine.length - target[0].length)
|
||||||
return false;
|
return false;
|
||||||
var startOffset = self.offset + match;
|
var startOffset = self.offset + match;
|
||||||
|
|
||||||
var line = self.history.nodeAfter(self.line);
|
var line = self.history.nodeAfter(self.line);
|
||||||
for (var i = 1; i < target.length - 1; i++) {
|
for (var i = 1; i < target.length - 1; i++) {
|
||||||
if (cleanText(self.history.textAfter(line)) != target[i])
|
var lineText = cleanText(self.history.textAfter(line));
|
||||||
|
if ((self.caseFold ? lineText.toLowerCase() : lineText) != target[i])
|
||||||
return false;
|
return false;
|
||||||
line = self.history.nodeAfter(line);
|
line = self.history.nodeAfter(line);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (cleanText(self.history.textAfter(line)).indexOf(target[target.length - 1]) != 0)
|
var lastLine = cleanText(self.history.textAfter(line));
|
||||||
|
if ((self.caseFold ? lastLine.toLowerCase() : lastLine).indexOf(target[target.length - 1]) != 0)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
return {from: {node: self.line, offset: startOffset},
|
return {from: {node: self.line, offset: startOffset},
|
||||||
@ -351,8 +373,7 @@ var Editor = (function(){
|
|||||||
this.doc = document;
|
this.doc = document;
|
||||||
var container = this.container = this.doc.body;
|
var container = this.container = this.doc.body;
|
||||||
this.win = window;
|
this.win = window;
|
||||||
this.history = new History(container, options.undoDepth, options.undoDelay,
|
this.history = new History(container, options.undoDepth, options.undoDelay, this);
|
||||||
this, options.onChange);
|
|
||||||
var self = this;
|
var self = this;
|
||||||
|
|
||||||
if (!Editor.Parser)
|
if (!Editor.Parser)
|
||||||
@ -364,10 +385,8 @@ var Editor = (function(){
|
|||||||
select.setCursorPos(container, {node: null, offset: 0});
|
select.setCursorPos(container, {node: null, offset: 0});
|
||||||
|
|
||||||
this.dirty = [];
|
this.dirty = [];
|
||||||
if (options.content)
|
this.importCode(options.content || "");
|
||||||
this.importCode(options.content);
|
this.history.onChange = options.onChange;
|
||||||
else // FF acts weird when the editable document is completely empty
|
|
||||||
container.appendChild(this.doc.createElement("BR"));
|
|
||||||
|
|
||||||
if (!options.readOnly) {
|
if (!options.readOnly) {
|
||||||
if (options.continuousScanning !== false) {
|
if (options.continuousScanning !== false) {
|
||||||
@ -409,6 +428,11 @@ var Editor = (function(){
|
|||||||
addEventHandler(document.body, "mouseup", cursorActivity);
|
addEventHandler(document.body, "mouseup", cursorActivity);
|
||||||
addEventHandler(document.body, "cut", cursorActivity);
|
addEventHandler(document.body, "cut", cursorActivity);
|
||||||
|
|
||||||
|
// workaround for a gecko bug [?] where going forward and then
|
||||||
|
// back again breaks designmode (no more cursor)
|
||||||
|
if (gecko)
|
||||||
|
addEventHandler(this.win, "pagehide", function(){self.unloaded = true;});
|
||||||
|
|
||||||
addEventHandler(document.body, "paste", function(event) {
|
addEventHandler(document.body, "paste", function(event) {
|
||||||
cursorActivity();
|
cursorActivity();
|
||||||
var text = null;
|
var text = null;
|
||||||
@ -418,15 +442,14 @@ var Editor = (function(){
|
|||||||
}
|
}
|
||||||
catch(e) {}
|
catch(e) {}
|
||||||
if (text !== null) {
|
if (text !== null) {
|
||||||
self.replaceSelection(text);
|
|
||||||
event.stop();
|
event.stop();
|
||||||
|
self.replaceSelection(text);
|
||||||
|
select.scrollToCursor(self.container);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
addEventHandler(document.body, "beforepaste", method(this, "reroutePasteEvent"));
|
|
||||||
|
|
||||||
if (this.options.autoMatchParens)
|
if (this.options.autoMatchParens)
|
||||||
addEventHandler(document.body, "click", method(this, "scheduleParenBlink"));
|
addEventHandler(document.body, "click", method(this, "scheduleParenHighlight"));
|
||||||
}
|
}
|
||||||
else if (!options.textWrapping) {
|
else if (!options.textWrapping) {
|
||||||
container.style.whiteSpace = "nowrap";
|
container.style.whiteSpace = "nowrap";
|
||||||
@ -491,6 +514,16 @@ var Editor = (function(){
|
|||||||
return startOfLine(line.previousSibling);
|
return startOfLine(line.previousSibling);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
visibleLineCount: function() {
|
||||||
|
var line = this.container.firstChild;
|
||||||
|
while (line && isBR(line)) line = line.nextSibling; // BR heights are unreliable
|
||||||
|
if (!line) return false;
|
||||||
|
var innerHeight = (window.innerHeight
|
||||||
|
|| document.documentElement.clientHeight
|
||||||
|
|| document.body.clientHeight);
|
||||||
|
return Math.floor(innerHeight / line.offsetHeight);
|
||||||
|
},
|
||||||
|
|
||||||
selectLines: function(startLine, startOffset, endLine, endOffset) {
|
selectLines: function(startLine, startOffset, endLine, endOffset) {
|
||||||
this.checkLine(startLine);
|
this.checkLine(startLine);
|
||||||
var start = {node: startLine, offset: startOffset}, end = null;
|
var start = {node: startLine, offset: startOffset}, end = null;
|
||||||
@ -503,10 +536,9 @@ var Editor = (function(){
|
|||||||
},
|
},
|
||||||
|
|
||||||
lineContent: function(line) {
|
lineContent: function(line) {
|
||||||
this.checkLine(line);
|
|
||||||
var accum = [];
|
var accum = [];
|
||||||
for (line = line ? line.nextSibling : this.container.firstChild;
|
for (line = line ? line.nextSibling : this.container.firstChild;
|
||||||
line && line.nodeName != "BR"; line = line.nextSibling)
|
line && !isBR(line); line = line.nextSibling)
|
||||||
accum.push(nodeText(line));
|
accum.push(nodeText(line));
|
||||||
return cleanText(accum.join(""));
|
return cleanText(accum.join(""));
|
||||||
},
|
},
|
||||||
@ -520,6 +552,18 @@ var Editor = (function(){
|
|||||||
this.scheduleHighlight();
|
this.scheduleHighlight();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
removeLine: function(line) {
|
||||||
|
var node = line ? line.nextSibling : this.container.firstChild;
|
||||||
|
while (node) {
|
||||||
|
var next = node.nextSibling;
|
||||||
|
removeElement(node);
|
||||||
|
if (isBR(node)) break;
|
||||||
|
node = next;
|
||||||
|
}
|
||||||
|
this.addDirtyNode(line);
|
||||||
|
this.scheduleHighlight();
|
||||||
|
},
|
||||||
|
|
||||||
insertIntoLine: function(line, position, content) {
|
insertIntoLine: function(line, position, content) {
|
||||||
var before = null;
|
var before = null;
|
||||||
if (position == "end") {
|
if (position == "end") {
|
||||||
@ -531,7 +575,7 @@ var Editor = (function(){
|
|||||||
before = cur;
|
before = cur;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
var text = (cur.innerText || cur.textContent || cur.nodeValue || "");
|
var text = nodeText(cur);
|
||||||
if (text.length > position) {
|
if (text.length > position) {
|
||||||
before = cur.nextSibling;
|
before = cur.nextSibling;
|
||||||
content = text.slice(0, position) + content + text.slice(position);
|
content = text.slice(0, position) + content + text.slice(position);
|
||||||
@ -586,13 +630,9 @@ var Editor = (function(){
|
|||||||
reroutePasteEvent: function() {
|
reroutePasteEvent: function() {
|
||||||
if (this.capturingPaste || window.opera) return;
|
if (this.capturingPaste || window.opera) return;
|
||||||
this.capturingPaste = true;
|
this.capturingPaste = true;
|
||||||
var te = parent.document.createElement("TEXTAREA");
|
var te = window.frameElement.CodeMirror.textareaHack;
|
||||||
te.style.position = "absolute";
|
|
||||||
te.style.left = "-500px";
|
|
||||||
te.style.width = "10px";
|
|
||||||
te.style.top = nodeTop(frameElement) + "px";
|
|
||||||
parent.document.body.appendChild(te);
|
|
||||||
parent.focus();
|
parent.focus();
|
||||||
|
te.value = "";
|
||||||
te.focus();
|
te.focus();
|
||||||
|
|
||||||
var self = this;
|
var self = this;
|
||||||
@ -600,10 +640,12 @@ var Editor = (function(){
|
|||||||
self.capturingPaste = false;
|
self.capturingPaste = false;
|
||||||
self.win.focus();
|
self.win.focus();
|
||||||
if (self.selectionSnapshot) // IE hack
|
if (self.selectionSnapshot) // IE hack
|
||||||
self.win.select.selectCoords(self.win, self.selectionSnapshot);
|
self.win.select.setBookmark(self.container, self.selectionSnapshot);
|
||||||
var text = te.value;
|
var text = te.value;
|
||||||
if (text) self.replaceSelection(text);
|
if (text) {
|
||||||
removeElement(te);
|
self.replaceSelection(text);
|
||||||
|
select.scrollToCursor(self.container);
|
||||||
|
}
|
||||||
}, 10);
|
}, 10);
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -618,8 +660,8 @@ var Editor = (function(){
|
|||||||
offset: lastLine.length};
|
offset: lastLine.length};
|
||||||
},
|
},
|
||||||
|
|
||||||
getSearchCursor: function(string, fromCursor) {
|
getSearchCursor: function(string, fromCursor, caseFold) {
|
||||||
return new SearchCursor(this, string, fromCursor);
|
return new SearchCursor(this, string, fromCursor, caseFold);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Re-indent the whole buffer
|
// Re-indent the whole buffer
|
||||||
@ -655,7 +697,7 @@ var Editor = (function(){
|
|||||||
forEach(this.container.childNodes, function(n) {
|
forEach(this.container.childNodes, function(n) {
|
||||||
if (n.nodeType != 3) n.dirty = true;
|
if (n.nodeType != 3) n.dirty = true;
|
||||||
});
|
});
|
||||||
this.addDirtyNode(this.firstChild);
|
this.addDirtyNode(this.container.firstChild);
|
||||||
this.scheduleHighlight();
|
this.scheduleHighlight();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -663,7 +705,7 @@ var Editor = (function(){
|
|||||||
// Intercept enter and tab, and assign their new functions.
|
// Intercept enter and tab, and assign their new functions.
|
||||||
keyDown: function(event) {
|
keyDown: function(event) {
|
||||||
if (this.frozen == "leave") this.frozen = null;
|
if (this.frozen == "leave") this.frozen = null;
|
||||||
if (this.frozen && (!this.keyFilter || this.keyFilter(event.keyCode))) {
|
if (this.frozen && (!this.keyFilter || this.keyFilter(event.keyCode, event))) {
|
||||||
event.stop();
|
event.stop();
|
||||||
this.frozen(event);
|
this.frozen(event);
|
||||||
return;
|
return;
|
||||||
@ -674,7 +716,7 @@ var Editor = (function(){
|
|||||||
this.delayScanning();
|
this.delayScanning();
|
||||||
// Schedule a paren-highlight event, if configured.
|
// Schedule a paren-highlight event, if configured.
|
||||||
if (this.options.autoMatchParens)
|
if (this.options.autoMatchParens)
|
||||||
this.scheduleParenBlink();
|
this.scheduleParenHighlight();
|
||||||
|
|
||||||
// The various checks for !altKey are there because AltGr sets both
|
// The various checks for !altKey are there because AltGr sets both
|
||||||
// ctrlKey and altKey to true, and should not be recognised as
|
// ctrlKey and altKey to true, and should not be recognised as
|
||||||
@ -690,8 +732,8 @@ var Editor = (function(){
|
|||||||
}
|
}
|
||||||
event.stop();
|
event.stop();
|
||||||
}
|
}
|
||||||
else if (code == 9 && this.options.tabMode != "default") { // tab
|
else if (code == 9 && this.options.tabMode != "default" && !event.ctrlKey) { // tab
|
||||||
this.handleTab(!event.ctrlKey && !event.shiftKey);
|
this.handleTab(!event.shiftKey);
|
||||||
event.stop();
|
event.stop();
|
||||||
}
|
}
|
||||||
else if (code == 32 && event.shiftKey && this.options.tabMode == "default") { // space
|
else if (code == 32 && event.shiftKey && this.options.tabMode == "default") { // space
|
||||||
@ -699,11 +741,20 @@ var Editor = (function(){
|
|||||||
event.stop();
|
event.stop();
|
||||||
}
|
}
|
||||||
else if (code == 36 && !event.shiftKey && !event.ctrlKey) { // home
|
else if (code == 36 && !event.shiftKey && !event.ctrlKey) { // home
|
||||||
if (this.home())
|
if (this.home()) event.stop();
|
||||||
event.stop();
|
}
|
||||||
|
else if (code == 35 && !event.shiftKey && !event.ctrlKey) { // end
|
||||||
|
if (this.end()) event.stop();
|
||||||
|
}
|
||||||
|
// Only in Firefox is the default behavior for PgUp/PgDn correct.
|
||||||
|
else if (code == 33 && !event.shiftKey && !event.ctrlKey && !gecko) { // PgUp
|
||||||
|
if (this.pageUp()) event.stop();
|
||||||
|
}
|
||||||
|
else if (code == 34 && !event.shiftKey && !event.ctrlKey && !gecko) { // PgDn
|
||||||
|
if (this.pageDown()) event.stop();
|
||||||
}
|
}
|
||||||
else if ((code == 219 || code == 221) && event.ctrlKey && !event.altKey) { // [, ]
|
else if ((code == 219 || code == 221) && event.ctrlKey && !event.altKey) { // [, ]
|
||||||
this.blinkParens(event.shiftKey);
|
this.highlightParens(event.shiftKey, true);
|
||||||
event.stop();
|
event.stop();
|
||||||
}
|
}
|
||||||
else if (event.metaKey && !event.shiftKey && (code == 37 || code == 39)) { // Meta-left/right
|
else if (event.metaKey && !event.shiftKey && (code == 37 || code == 39)) { // Meta-left/right
|
||||||
@ -730,6 +781,9 @@ var Editor = (function(){
|
|||||||
this.options.saveFunction();
|
this.options.saveFunction();
|
||||||
event.stop();
|
event.stop();
|
||||||
}
|
}
|
||||||
|
else if (internetExplorer && code == 86) {
|
||||||
|
this.reroutePasteEvent();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -741,7 +795,7 @@ var Editor = (function(){
|
|||||||
// keydown event does not prevent the associated keypress event
|
// keydown event does not prevent the associated keypress event
|
||||||
// from happening, so we have to cancel enter and tab again
|
// from happening, so we have to cancel enter and tab again
|
||||||
// here.
|
// here.
|
||||||
if ((this.frozen && (!this.keyFilter || this.keyFilter(event.keyCode))) ||
|
if ((this.frozen && (!this.keyFilter || this.keyFilter(event.keyCode, event))) ||
|
||||||
event.code == 13 || (event.code == 9 && this.options.tabMode != "default") ||
|
event.code == 13 || (event.code == 9 && this.options.tabMode != "default") ||
|
||||||
(event.keyCode == 32 && event.shiftKey && this.options.tabMode == "default"))
|
(event.keyCode == 32 && event.shiftKey && this.options.tabMode == "default"))
|
||||||
event.stop();
|
event.stop();
|
||||||
@ -812,17 +866,17 @@ var Editor = (function(){
|
|||||||
if (start) insertAfter(whiteSpace, start);
|
if (start) insertAfter(whiteSpace, start);
|
||||||
else this.container.insertBefore(whiteSpace, this.container.firstChild);
|
else this.container.insertBefore(whiteSpace, this.container.firstChild);
|
||||||
}
|
}
|
||||||
if (firstText) select.snapshotMove(firstText.firstChild, whiteSpace.firstChild, curIndent, false, true);
|
var fromNode = firstText && (firstText.firstChild || firstText);
|
||||||
|
select.snapshotMove(fromNode, whiteSpace.firstChild, newIndent, false, true);
|
||||||
}
|
}
|
||||||
if (indentDiff != 0) this.addDirtyNode(start);
|
if (indentDiff != 0) this.addDirtyNode(start);
|
||||||
return whiteSpace;
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// Re-highlight the selected part of the document.
|
// Re-highlight the selected part of the document.
|
||||||
highlightAtCursor: function() {
|
highlightAtCursor: function() {
|
||||||
var pos = select.selectionTopNode(this.container, true);
|
var pos = select.selectionTopNode(this.container, true);
|
||||||
var to = select.selectionTopNode(this.container, false);
|
var to = select.selectionTopNode(this.container, false);
|
||||||
if (pos === false || to === false) return;
|
if (pos === false || to === false) return false;
|
||||||
|
|
||||||
select.markSelection(this.win);
|
select.markSelection(this.win);
|
||||||
if (this.highlight(pos, endOfLine(to, this.container), true, 20) === false)
|
if (this.highlight(pos, endOfLine(to, this.container), true, 20) === false)
|
||||||
@ -841,12 +895,14 @@ var Editor = (function(){
|
|||||||
this.reindentSelection(direction);
|
this.reindentSelection(direction);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
// Custom home behaviour that doesn't land the cursor in front of
|
||||||
|
// leading whitespace unless pressed twice.
|
||||||
home: function() {
|
home: function() {
|
||||||
var cur = select.selectionTopNode(this.container, true), start = cur;
|
var cur = select.selectionTopNode(this.container, true), start = cur;
|
||||||
if (cur === false || !(!cur || cur.isPart || cur.nodeName == "BR") || !this.container.firstChild)
|
if (cur === false || !(!cur || cur.isPart || isBR(cur)) || !this.container.firstChild)
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
while (cur && cur.nodeName != "BR") cur = cur.previousSibling;
|
while (cur && !isBR(cur)) cur = cur.previousSibling;
|
||||||
var next = cur ? cur.nextSibling : this.container.firstChild;
|
var next = cur ? cur.nextSibling : this.container.firstChild;
|
||||||
if (next && next != start && next.isPart && hasClass(next, "whitespace"))
|
if (next && next != start && next.isPart && hasClass(next, "whitespace"))
|
||||||
select.focusAfterNode(next, this.container);
|
select.focusAfterNode(next, this.container);
|
||||||
@ -857,18 +913,88 @@ var Editor = (function(){
|
|||||||
return true;
|
return true;
|
||||||
},
|
},
|
||||||
|
|
||||||
// Delay (or initiate) the next paren blink event.
|
// Some browsers (Opera) don't manage to handle the end key
|
||||||
scheduleParenBlink: function() {
|
// properly in the face of vertical scrolling.
|
||||||
|
end: function() {
|
||||||
|
var cur = select.selectionTopNode(this.container, true);
|
||||||
|
if (cur === false) return false;
|
||||||
|
cur = endOfLine(cur, this.container);
|
||||||
|
if (!cur) return false;
|
||||||
|
select.focusAfterNode(cur.previousSibling, this.container);
|
||||||
|
select.scrollToCursor(this.container);
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
pageUp: function() {
|
||||||
|
var line = this.cursorPosition().line, scrollAmount = this.visibleLineCount();
|
||||||
|
if (line === false || scrollAmount === false) return false;
|
||||||
|
// Try to keep one line on the screen.
|
||||||
|
scrollAmount -= 2;
|
||||||
|
for (var i = 0; i < scrollAmount; i++) {
|
||||||
|
line = this.prevLine(line);
|
||||||
|
if (line === false) break;
|
||||||
|
}
|
||||||
|
if (i == 0) return false; // Already at first line
|
||||||
|
select.setCursorPos(this.container, {node: line, offset: 0});
|
||||||
|
select.scrollToCursor(this.container);
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
pageDown: function() {
|
||||||
|
var line = this.cursorPosition().line, scrollAmount = this.visibleLineCount();
|
||||||
|
if (line === false || scrollAmount === false) return false;
|
||||||
|
// Try to move to the last line of the current page.
|
||||||
|
scrollAmount -= 2;
|
||||||
|
for (var i = 0; i < scrollAmount; i++) {
|
||||||
|
var nextLine = this.nextLine(line);
|
||||||
|
if (nextLine === false) break;
|
||||||
|
line = nextLine;
|
||||||
|
}
|
||||||
|
if (i == 0) return false; // Already at last line
|
||||||
|
select.setCursorPos(this.container, {node: line, offset: 0});
|
||||||
|
select.scrollToCursor(this.container);
|
||||||
|
return true;
|
||||||
|
},
|
||||||
|
|
||||||
|
// Delay (or initiate) the next paren highlight event.
|
||||||
|
scheduleParenHighlight: function() {
|
||||||
if (this.parenEvent) this.parent.clearTimeout(this.parenEvent);
|
if (this.parenEvent) this.parent.clearTimeout(this.parenEvent);
|
||||||
var self = this;
|
var self = this;
|
||||||
this.parenEvent = this.parent.setTimeout(function(){self.blinkParens();}, 300);
|
this.parenEvent = this.parent.setTimeout(function(){self.highlightParens();}, 300);
|
||||||
},
|
},
|
||||||
|
|
||||||
// Take the token before the cursor. If it contains a character in
|
// Take the token before the cursor. If it contains a character in
|
||||||
// '()[]{}', search for the matching paren/brace/bracket, and
|
// '()[]{}', search for the matching paren/brace/bracket, and
|
||||||
// highlight them in green for a moment, or red if no proper match
|
// highlight them in green for a moment, or red if no proper match
|
||||||
// was found.
|
// was found.
|
||||||
blinkParens: function(jump) {
|
highlightParens: function(jump, fromKey) {
|
||||||
|
var self = this;
|
||||||
|
// give the relevant nodes a colour.
|
||||||
|
function highlight(node, ok) {
|
||||||
|
if (!node) return;
|
||||||
|
if (self.options.markParen) {
|
||||||
|
self.options.markParen(node, ok);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
node.style.fontWeight = "bold";
|
||||||
|
node.style.color = ok ? "#8F8" : "#F88";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function unhighlight(node) {
|
||||||
|
if (!node) return;
|
||||||
|
if (self.options.unmarkParen) {
|
||||||
|
self.options.unmarkParen(node);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
node.style.fontWeight = "";
|
||||||
|
node.style.color = "";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (!fromKey && self.highlighted) {
|
||||||
|
unhighlight(self.highlighted[0]);
|
||||||
|
unhighlight(self.highlighted[1]);
|
||||||
|
}
|
||||||
|
|
||||||
if (!window.select) return;
|
if (!window.select) return;
|
||||||
// Clear the event property.
|
// Clear the event property.
|
||||||
if (this.parenEvent) this.parent.clearTimeout(this.parenEvent);
|
if (this.parenEvent) this.parent.clearTimeout(this.parenEvent);
|
||||||
@ -886,7 +1012,7 @@ var Editor = (function(){
|
|||||||
return /[\(\[\{]/.test(ch);
|
return /[\(\[\{]/.test(ch);
|
||||||
}
|
}
|
||||||
|
|
||||||
var ch, self = this, cursor = select.selectionTopNode(this.container, true);
|
var ch, cursor = select.selectionTopNode(this.container, true);
|
||||||
if (!cursor || !this.highlightAtCursor()) return;
|
if (!cursor || !this.highlightAtCursor()) return;
|
||||||
cursor = select.selectionTopNode(this.container, true);
|
cursor = select.selectionTopNode(this.container, true);
|
||||||
if (!(cursor && ((ch = paren(cursor)) || (cursor = cursor.nextSibling) && (ch = paren(cursor)))))
|
if (!(cursor && ((ch = paren(cursor)) || (cursor = cursor.nextSibling) && (ch = paren(cursor)))))
|
||||||
@ -899,9 +1025,9 @@ var Editor = (function(){
|
|||||||
// have to scan, we just try, and when we find dirty nodes we
|
// have to scan, we just try, and when we find dirty nodes we
|
||||||
// abort, parse them, and re-try.
|
// abort, parse them, and re-try.
|
||||||
function tryFindMatch() {
|
function tryFindMatch() {
|
||||||
var stack = [], ch, ok = true;;
|
var stack = [], ch, ok = true;
|
||||||
for (var runner = cursor; runner; runner = dir ? runner.nextSibling : runner.previousSibling) {
|
for (var runner = cursor; runner; runner = dir ? runner.nextSibling : runner.previousSibling) {
|
||||||
if (runner.className == className && runner.nodeName == "SPAN" && (ch = paren(runner))) {
|
if (runner.className == className && isSpan(runner) && (ch = paren(runner))) {
|
||||||
if (forward(ch) == dir)
|
if (forward(ch) == dir)
|
||||||
stack.push(ch);
|
stack.push(ch);
|
||||||
else if (!stack.length)
|
else if (!stack.length)
|
||||||
@ -910,18 +1036,12 @@ var Editor = (function(){
|
|||||||
ok = false;
|
ok = false;
|
||||||
if (!stack.length) break;
|
if (!stack.length) break;
|
||||||
}
|
}
|
||||||
else if (runner.dirty || runner.nodeName != "SPAN" && runner.nodeName != "BR") {
|
else if (runner.dirty || !isSpan(runner) && !isBR(runner)) {
|
||||||
return {node: runner, status: "dirty"};
|
return {node: runner, status: "dirty"};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return {node: runner, status: runner && ok};
|
return {node: runner, status: runner && ok};
|
||||||
}
|
}
|
||||||
// Temporarily give the relevant nodes a colour.
|
|
||||||
function blink(node, ok) {
|
|
||||||
node.style.fontWeight = "bold";
|
|
||||||
node.style.color = ok ? "#8F8" : "#F88";
|
|
||||||
self.parent.setTimeout(function() {node.style.fontWeight = ""; node.style.color = "";}, 500);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (true) {
|
while (true) {
|
||||||
var found = tryFindMatch();
|
var found = tryFindMatch();
|
||||||
@ -933,11 +1053,14 @@ var Editor = (function(){
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
blink(cursor, found.status);
|
highlight(cursor, found.status);
|
||||||
if (found.node) {
|
highlight(found.node, found.status);
|
||||||
blink(found.node, found.status);
|
if (fromKey)
|
||||||
if (jump) select.focusAfterNode(found.node.previousSibling, this.container);
|
self.parent.setTimeout(function() {unhighlight(cursor); unhighlight(found.node);}, 500);
|
||||||
}
|
else
|
||||||
|
self.highlighted = [cursor, found.node];
|
||||||
|
if (jump && found.node)
|
||||||
|
select.focusAfterNode(found.node.previousSibling, this.container);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -955,20 +1078,17 @@ var Editor = (function(){
|
|||||||
// there's nothing to indent.
|
// there's nothing to indent.
|
||||||
if (cursor === false)
|
if (cursor === false)
|
||||||
return;
|
return;
|
||||||
var lineStart = startOfLine(cursor);
|
select.markSelection(this.win);
|
||||||
var whiteSpace = this.indentLineAfter(lineStart, direction);
|
this.indentLineAfter(startOfLine(cursor), direction);
|
||||||
if (cursor == lineStart && whiteSpace)
|
select.selectMarked();
|
||||||
cursor = whiteSpace;
|
|
||||||
// This means the indentation has probably messed up the cursor.
|
|
||||||
if (cursor == whiteSpace)
|
|
||||||
select.focusAfterNode(cursor, this.container);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// Indent all lines whose start falls inside of the current
|
// Indent all lines whose start falls inside of the current
|
||||||
// selection.
|
// selection.
|
||||||
indentRegion: function(start, end, direction) {
|
indentRegion: function(start, end, direction) {
|
||||||
var current = (start = startOfLine(start)), before = start && startOfLine(start.previousSibling);
|
var current = (start = startOfLine(start)), before = start && startOfLine(start.previousSibling);
|
||||||
if (end.nodeName != "BR") end = endOfLine(end, this.container);
|
if (!isBR(end)) end = endOfLine(end, this.container);
|
||||||
|
this.addDirtyNode(start);
|
||||||
|
|
||||||
do {
|
do {
|
||||||
var next = endOfLine(current, this.container);
|
var next = endOfLine(current, this.container);
|
||||||
@ -983,9 +1103,16 @@ var Editor = (function(){
|
|||||||
// Find the node that the cursor is in, mark it as dirty, and make
|
// Find the node that the cursor is in, mark it as dirty, and make
|
||||||
// sure a highlight pass is scheduled.
|
// sure a highlight pass is scheduled.
|
||||||
cursorActivity: function(safe) {
|
cursorActivity: function(safe) {
|
||||||
|
// pagehide event hack above
|
||||||
|
if (this.unloaded) {
|
||||||
|
this.win.document.designMode = "off";
|
||||||
|
this.win.document.designMode = "on";
|
||||||
|
this.unloaded = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (internetExplorer) {
|
if (internetExplorer) {
|
||||||
this.container.createTextRange().execCommand("unlink");
|
this.container.createTextRange().execCommand("unlink");
|
||||||
this.selectionSnapshot = select.selectionCoords(this.win);
|
this.selectionSnapshot = select.getBookmark(this.container);
|
||||||
}
|
}
|
||||||
|
|
||||||
var activity = this.options.cursorActivity;
|
var activity = this.options.cursorActivity;
|
||||||
@ -1021,6 +1148,10 @@ var Editor = (function(){
|
|||||||
this.dirty.push(node);
|
this.dirty.push(node);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
allClean: function() {
|
||||||
|
return !this.dirty.length;
|
||||||
|
},
|
||||||
|
|
||||||
// Cause a highlight pass to happen in options.passDelay
|
// Cause a highlight pass to happen in options.passDelay
|
||||||
// milliseconds. Clear the existing timeout, if one exists. This
|
// milliseconds. Clear the existing timeout, if one exists. This
|
||||||
// way, the passes do not happen while the user is typing, and
|
// way, the passes do not happen while the user is typing, and
|
||||||
@ -1043,7 +1174,7 @@ var Editor = (function(){
|
|||||||
// If the node has been coloured in the meantime, or is no
|
// If the node has been coloured in the meantime, or is no
|
||||||
// longer in the document, it should not be returned.
|
// longer in the document, it should not be returned.
|
||||||
while (found && found.parentNode != this.container)
|
while (found && found.parentNode != this.container)
|
||||||
found = found.parentNode
|
found = found.parentNode;
|
||||||
if (found && (found.dirty || found.nodeType == 3))
|
if (found && (found.dirty || found.nodeType == 3))
|
||||||
return found;
|
return found;
|
||||||
} catch (e) {}
|
} catch (e) {}
|
||||||
@ -1060,7 +1191,7 @@ var Editor = (function(){
|
|||||||
highlightDirty: function(force) {
|
highlightDirty: function(force) {
|
||||||
// Prevent FF from raising an error when it is firing timeouts
|
// Prevent FF from raising an error when it is firing timeouts
|
||||||
// on a page that's no longer loaded.
|
// on a page that's no longer loaded.
|
||||||
if (!window.select) return;
|
if (!window.select) return false;
|
||||||
|
|
||||||
if (!this.options.readOnly) select.markSelection(this.win);
|
if (!this.options.readOnly) select.markSelection(this.win);
|
||||||
var start, endTime = force ? null : time() + this.options.passTime;
|
var start, endTime = force ? null : time() + this.options.passTime;
|
||||||
@ -1123,17 +1254,17 @@ var Editor = (function(){
|
|||||||
var endTime = (typeof target == "number" ? target : null);
|
var endTime = (typeof target == "number" ? target : null);
|
||||||
|
|
||||||
if (!container.firstChild)
|
if (!container.firstChild)
|
||||||
return;
|
return false;
|
||||||
// Backtrack to the first node before from that has a partial
|
// Backtrack to the first node before from that has a partial
|
||||||
// parse stored.
|
// parse stored.
|
||||||
while (from && (!from.parserFromHere || from.dirty)) {
|
while (from && (!from.parserFromHere || from.dirty)) {
|
||||||
if (maxBacktrack != null && from.nodeName == "BR" && (--maxBacktrack) < 0)
|
if (maxBacktrack != null && isBR(from) && (--maxBacktrack) < 0)
|
||||||
return false;
|
return false;
|
||||||
from = from.previousSibling;
|
from = from.previousSibling;
|
||||||
}
|
}
|
||||||
// If we are at the end of the document, do nothing.
|
// If we are at the end of the document, do nothing.
|
||||||
if (from && !from.nextSibling)
|
if (from && !from.nextSibling)
|
||||||
return;
|
return false;
|
||||||
|
|
||||||
// Check whether a part (<span> node) and the corresponding token
|
// Check whether a part (<span> node) and the corresponding token
|
||||||
// match.
|
// match.
|
||||||
@ -1178,6 +1309,11 @@ var Editor = (function(){
|
|||||||
stream = stringStream(traversal),
|
stream = stringStream(traversal),
|
||||||
parsed = from ? from.parserFromHere(stream) : Editor.Parser.make(stream);
|
parsed = from ? from.parserFromHere(stream) : Editor.Parser.make(stream);
|
||||||
|
|
||||||
|
function surroundedByBRs(node) {
|
||||||
|
return (node.previousSibling == null || isBR(node.previousSibling)) &&
|
||||||
|
(node.nextSibling == null || isBR(node.nextSibling));
|
||||||
|
}
|
||||||
|
|
||||||
// parts is an interface to make it possible to 'delay' fetching
|
// parts is an interface to make it possible to 'delay' fetching
|
||||||
// the next DOM node until we are completely done with the one
|
// the next DOM node until we are completely done with the one
|
||||||
// before it. This is necessary because often the next node is
|
// before it. This is necessary because often the next node is
|
||||||
@ -1208,13 +1344,23 @@ var Editor = (function(){
|
|||||||
// Allow empty nodes when they are alone on a line, needed
|
// Allow empty nodes when they are alone on a line, needed
|
||||||
// for the FF cursor bug workaround (see select.js,
|
// for the FF cursor bug workaround (see select.js,
|
||||||
// insertNewlineAtCursor).
|
// insertNewlineAtCursor).
|
||||||
while (part && part.nodeName == "SPAN" && part.currentText == "") {
|
while (part && isSpan(part) && part.currentText == "") {
|
||||||
var old = part;
|
// Leave empty nodes that are alone on a line alone in
|
||||||
this.remove();
|
// Opera, since that browsers doesn't deal well with
|
||||||
part = this.get();
|
// having 2 BRs in a row.
|
||||||
// Adjust selection information, if any. See select.js for details.
|
if (window.opera && surroundedByBRs(part)) {
|
||||||
select.snapshotMove(old.firstChild, part && (part.firstChild || part), 0);
|
this.next();
|
||||||
|
part = this.get();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
var old = part;
|
||||||
|
this.remove();
|
||||||
|
part = this.get();
|
||||||
|
// Adjust selection information, if any. See select.js for details.
|
||||||
|
select.snapshotMove(old.firstChild, part && (part.firstChild || part), 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return part;
|
return part;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
@ -1230,7 +1376,7 @@ var Editor = (function(){
|
|||||||
if (token.value == "\n"){
|
if (token.value == "\n"){
|
||||||
// The idea of the two streams actually staying synchronized
|
// The idea of the two streams actually staying synchronized
|
||||||
// is such a long shot that we explicitly check.
|
// is such a long shot that we explicitly check.
|
||||||
if (part.nodeName != "BR")
|
if (!isBR(part))
|
||||||
throw "Parser out of sync. Expected BR.";
|
throw "Parser out of sync. Expected BR.";
|
||||||
|
|
||||||
if (part.dirty || !part.indentation) lineDirty = true;
|
if (part.dirty || !part.indentation) lineDirty = true;
|
||||||
@ -1258,7 +1404,7 @@ var Editor = (function(){
|
|||||||
parts.next();
|
parts.next();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
if (part.nodeName != "SPAN")
|
if (!isSpan(part))
|
||||||
throw "Parser out of sync. Expected SPAN.";
|
throw "Parser out of sync. Expected SPAN.";
|
||||||
if (part.dirty)
|
if (part.dirty)
|
||||||
lineDirty = true;
|
lineDirty = true;
|
||||||
@ -1314,6 +1460,6 @@ var Editor = (function(){
|
|||||||
|
|
||||||
addEventHandler(window, "load", function() {
|
addEventHandler(window, "load", function() {
|
||||||
var CodeMirror = window.frameElement.CodeMirror;
|
var CodeMirror = window.frameElement.CodeMirror;
|
||||||
CodeMirror.editor = new Editor(CodeMirror.options);
|
var e = CodeMirror.editor = new Editor(CodeMirror.options);
|
||||||
this.parent.setTimeout(method(CodeMirror, "init"), 0);
|
this.parent.setTimeout(method(CodeMirror, "init"), 0);
|
||||||
});
|
});
|
||||||
|
@ -34,7 +34,7 @@ var indentUnit = 2;
|
|||||||
}
|
}
|
||||||
|
|
||||||
window.highlightText = function(string, callback, parser) {
|
window.highlightText = function(string, callback, parser) {
|
||||||
var parser = (parser || Editor.Parser).make(stringStream(normaliseString(string)));
|
parser = (parser || Editor.Parser).make(stringStream(normaliseString(string)));
|
||||||
var line = [];
|
var line = [];
|
||||||
if (callback.nodeType == 1) {
|
if (callback.nodeType == 1) {
|
||||||
var node = callback;
|
var node = callback;
|
||||||
|
@ -36,7 +36,7 @@ MirrorFrame.prototype = {
|
|||||||
|
|
||||||
var first = true;
|
var first = true;
|
||||||
do {
|
do {
|
||||||
var cursor = this.mirror.getSearchCursor(text, first);
|
var cursor = this.mirror.getSearchCursor(text, first, true);
|
||||||
first = false;
|
first = false;
|
||||||
while (cursor.findNext()) {
|
while (cursor.findNext()) {
|
||||||
cursor.select();
|
cursor.select();
|
||||||
|
@ -10,6 +10,8 @@
|
|||||||
var JSParser = Editor.Parser = (function() {
|
var JSParser = Editor.Parser = (function() {
|
||||||
// Token types that can be considered to be atoms.
|
// Token types that can be considered to be atoms.
|
||||||
var atomicTypes = {"atom": true, "number": true, "variable": true, "string": true, "regexp": true};
|
var atomicTypes = {"atom": true, "number": true, "variable": true, "string": true, "regexp": true};
|
||||||
|
// Setting that can be used to have JSON data indent properly.
|
||||||
|
var json = false;
|
||||||
// Constructor for the lexical context objects.
|
// Constructor for the lexical context objects.
|
||||||
function JSLexical(indented, column, type, align, prev, info) {
|
function JSLexical(indented, column, type, align, prev, info) {
|
||||||
// indentation at start of this line
|
// indentation at start of this line
|
||||||
@ -58,7 +60,7 @@ var JSParser = Editor.Parser = (function() {
|
|||||||
// semicolon. Actions at the end of the stack go first. It is
|
// semicolon. Actions at the end of the stack go first. It is
|
||||||
// initialized with an infinitely looping action that consumes
|
// initialized with an infinitely looping action that consumes
|
||||||
// whole statements.
|
// whole statements.
|
||||||
var cc = [statements];
|
var cc = [json ? singleExpr : statements];
|
||||||
// Context contains information about the current local scope, the
|
// Context contains information about the current local scope, the
|
||||||
// variables defined in that, and the scopes above it.
|
// variables defined in that, and the scopes above it.
|
||||||
var context = null;
|
var context = null;
|
||||||
@ -224,6 +226,9 @@ var JSParser = Editor.Parser = (function() {
|
|||||||
function statements(type){
|
function statements(type){
|
||||||
return pass(statement, statements);
|
return pass(statement, statements);
|
||||||
}
|
}
|
||||||
|
function singleExpr(type){
|
||||||
|
return pass(expression, statements);
|
||||||
|
}
|
||||||
// Dispatches various types of statements based on the type of the
|
// Dispatches various types of statements based on the type of the
|
||||||
// current token.
|
// current token.
|
||||||
function statement(type){
|
function statement(type){
|
||||||
@ -282,7 +287,7 @@ var JSParser = Editor.Parser = (function() {
|
|||||||
if (type == ",") cont(what, proceed);
|
if (type == ",") cont(what, proceed);
|
||||||
else if (type == end) cont();
|
else if (type == end) cont();
|
||||||
else cont(expect(end));
|
else cont(expect(end));
|
||||||
};
|
}
|
||||||
return function commaSeparated(type) {
|
return function commaSeparated(type) {
|
||||||
if (type == end) cont();
|
if (type == end) cont();
|
||||||
else pass(what, proceed);
|
else pass(what, proceed);
|
||||||
@ -337,5 +342,11 @@ var JSParser = Editor.Parser = (function() {
|
|||||||
return parser;
|
return parser;
|
||||||
}
|
}
|
||||||
|
|
||||||
return {make: parseJS, electricChars: "{}:"};
|
return {
|
||||||
|
make: parseJS,
|
||||||
|
electricChars: "{}:",
|
||||||
|
configure: function(obj) {
|
||||||
|
if (obj.json != null) json = obj.json;
|
||||||
|
}
|
||||||
|
};
|
||||||
})();
|
})();
|
||||||
|
@ -125,11 +125,11 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
// parseJavaScript in parsejavascript.js (there is actually a bit more
|
// parseJavaScript in parsejavascript.js (there is actually a bit more
|
||||||
// shared code than I'd like), but it is quite a bit simpler.
|
// shared code than I'd like), but it is quite a bit simpler.
|
||||||
function parseXML(source) {
|
function parseXML(source) {
|
||||||
var tokens = tokenizeXML(source);
|
var tokens = tokenizeXML(source), token;
|
||||||
var cc = [base];
|
var cc = [base];
|
||||||
var tokenNr = 0, indented = 0;
|
var tokenNr = 0, indented = 0;
|
||||||
var currentTag = null, context = null;
|
var currentTag = null, context = null;
|
||||||
var consume, marked;
|
var consume;
|
||||||
|
|
||||||
function push(fs) {
|
function push(fs) {
|
||||||
for (var i = fs.length - 1; i >= 0; i--)
|
for (var i = fs.length - 1; i >= 0; i--)
|
||||||
@ -144,13 +144,13 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
consume = false;
|
consume = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
function mark(style) {
|
function markErr() {
|
||||||
marked = style;
|
token.style += " xml-error";
|
||||||
}
|
}
|
||||||
function expect(text) {
|
function expect(text) {
|
||||||
return function(style, content) {
|
return function(style, content) {
|
||||||
if (content == text) cont();
|
if (content == text) cont();
|
||||||
else mark("xml-error") || cont(arguments.callee);
|
else {markErr(); cont(arguments.callee);}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -192,12 +192,12 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
cont();
|
cont();
|
||||||
}
|
}
|
||||||
else if (harmlessTokens.hasOwnProperty(style)) cont();
|
else if (harmlessTokens.hasOwnProperty(style)) cont();
|
||||||
else mark("xml-error") || cont();
|
else {markErr(); cont();}
|
||||||
}
|
}
|
||||||
function tagname(style, content) {
|
function tagname(style, content) {
|
||||||
if (style == "xml-name") {
|
if (style == "xml-name") {
|
||||||
currentTag = content.toLowerCase();
|
currentTag = content.toLowerCase();
|
||||||
mark("xml-tagname");
|
token.style = "xml-tagname";
|
||||||
cont();
|
cont();
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -206,24 +206,22 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
function closetagname(style, content) {
|
function closetagname(style, content) {
|
||||||
if (style == "xml-name" && context && content.toLowerCase() == context.name) {
|
if (style == "xml-name") {
|
||||||
popContext();
|
token.style = "xml-tagname";
|
||||||
mark("xml-tagname");
|
if (context && content.toLowerCase() == context.name) popContext();
|
||||||
}
|
else markErr();
|
||||||
else {
|
|
||||||
mark("xml-error");
|
|
||||||
}
|
}
|
||||||
cont();
|
cont();
|
||||||
}
|
}
|
||||||
function endtag(startOfLine) {
|
function endtag(startOfLine) {
|
||||||
return function(style, content) {
|
return function(style, content) {
|
||||||
if (content == "/>" || (content == ">" && UseKludges.autoSelfClosers.hasOwnProperty(currentTag))) cont();
|
if (content == "/>" || (content == ">" && UseKludges.autoSelfClosers.hasOwnProperty(currentTag))) cont();
|
||||||
else if (content == ">") pushContext(currentTag, startOfLine) || cont();
|
else if (content == ">") {pushContext(currentTag, startOfLine); cont();}
|
||||||
else mark("xml-error") || cont(arguments.callee);
|
else {markErr(); cont(arguments.callee);}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
function attributes(style) {
|
function attributes(style) {
|
||||||
if (style == "xml-name") mark("xml-attname") || cont(attribute, attributes);
|
if (style == "xml-name") {token.style = "xml-attname"; cont(attribute, attributes);}
|
||||||
else pass();
|
else pass();
|
||||||
}
|
}
|
||||||
function attribute(style, content) {
|
function attribute(style, content) {
|
||||||
@ -240,7 +238,7 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
indentation: function() {return indented;},
|
indentation: function() {return indented;},
|
||||||
|
|
||||||
next: function(){
|
next: function(){
|
||||||
var token = tokens.next();
|
token = tokens.next();
|
||||||
if (token.style == "whitespace" && tokenNr == 0)
|
if (token.style == "whitespace" && tokenNr == 0)
|
||||||
indented = token.value.length;
|
indented = token.value.length;
|
||||||
else
|
else
|
||||||
@ -254,13 +252,9 @@ var XMLParser = Editor.Parser = (function() {
|
|||||||
return token;
|
return token;
|
||||||
|
|
||||||
while(true){
|
while(true){
|
||||||
consume = marked = false;
|
consume = false;
|
||||||
cc.pop()(token.style, token.content);
|
cc.pop()(token.style, token.content);
|
||||||
if (consume){
|
if (consume) return token;
|
||||||
if (marked)
|
|
||||||
token.style = marked;
|
|
||||||
return token;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -37,7 +37,7 @@ var select = {};
|
|||||||
atEnd = !element.nextSibling || !element.nextSibling.nextSibling
|
atEnd = !element.nextSibling || !element.nextSibling.nextSibling
|
||||||
|| !element.nextSibling.nextSibling.nextSibling;
|
|| !element.nextSibling.nextSibling.nextSibling;
|
||||||
// In Opera (and recent Webkit versions), BR elements *always*
|
// In Opera (and recent Webkit versions), BR elements *always*
|
||||||
// have a scrollTop property of zero.
|
// have a offsetTop property of zero.
|
||||||
var compensateHack = 0;
|
var compensateHack = 0;
|
||||||
while (element && !element.offsetTop) {
|
while (element && !element.offsetTop) {
|
||||||
compensateHack++;
|
compensateHack++;
|
||||||
@ -52,7 +52,7 @@ var select = {};
|
|||||||
while (pos && pos.offsetParent) {
|
while (pos && pos.offsetParent) {
|
||||||
y += pos.offsetTop;
|
y += pos.offsetTop;
|
||||||
// Don't count X offset for <br> nodes
|
// Don't count X offset for <br> nodes
|
||||||
if (pos.nodeName != "BR")
|
if (!isBR(pos))
|
||||||
x += pos.offsetLeft;
|
x += pos.offsetLeft;
|
||||||
pos = pos.offsetParent;
|
pos = pos.offsetParent;
|
||||||
}
|
}
|
||||||
@ -66,7 +66,7 @@ var select = {};
|
|||||||
scroll = true;
|
scroll = true;
|
||||||
}
|
}
|
||||||
if (screen_y < 0 || atEnd || screen_y > (win.innerHeight || html.clientHeight || 0) - 50) {
|
if (screen_y < 0 || atEnd || screen_y > (win.innerHeight || html.clientHeight || 0) - 50) {
|
||||||
scroll_y = atEnd ? 1e10 : y;
|
scroll_y = atEnd ? 1e6 : y;
|
||||||
scroll = true;
|
scroll = true;
|
||||||
}
|
}
|
||||||
if (scroll) win.scrollTo(scroll_x, scroll_y);
|
if (scroll) win.scrollTo(scroll_x, scroll_y);
|
||||||
@ -247,13 +247,17 @@ var select = {};
|
|||||||
}
|
}
|
||||||
if (cur) {
|
if (cur) {
|
||||||
try{range.moveToElementText(cur);}
|
try{range.moveToElementText(cur);}
|
||||||
catch(e){}
|
catch(e){return false;}
|
||||||
range.collapse(false);
|
range.collapse(false);
|
||||||
}
|
}
|
||||||
else range.moveToElementText(node.parentNode);
|
else range.moveToElementText(node.parentNode);
|
||||||
if (count) range.move("character", count);
|
if (count) range.move("character", count);
|
||||||
}
|
}
|
||||||
else range.moveToElementText(node);
|
else {
|
||||||
|
try{range.moveToElementText(node);}
|
||||||
|
catch(e){return false;}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Do a binary search through the container object, comparing
|
// Do a binary search through the container object, comparing
|
||||||
@ -262,7 +266,7 @@ var select = {};
|
|||||||
while (start < end) {
|
while (start < end) {
|
||||||
var middle = Math.ceil((end + start) / 2), node = container.childNodes[middle];
|
var middle = Math.ceil((end + start) / 2), node = container.childNodes[middle];
|
||||||
if (!node) return false; // Don't ask. IE6 manages this sometimes.
|
if (!node) return false; // Don't ask. IE6 manages this sometimes.
|
||||||
moveToNodeStart(range2, node);
|
if (!moveToNodeStart(range2, node)) return false;
|
||||||
if (range.compareEndPoints("StartToStart", range2) == 1)
|
if (range.compareEndPoints("StartToStart", range2) == 1)
|
||||||
start = middle;
|
start = middle;
|
||||||
else
|
else
|
||||||
@ -314,7 +318,7 @@ var select = {};
|
|||||||
if (!selection) return null;
|
if (!selection) return null;
|
||||||
|
|
||||||
var topNode = select.selectionTopNode(container, start);
|
var topNode = select.selectionTopNode(container, start);
|
||||||
while (topNode && topNode.nodeName != "BR")
|
while (topNode && !isBR(topNode))
|
||||||
topNode = topNode.previousSibling;
|
topNode = topNode.previousSibling;
|
||||||
|
|
||||||
var range = selection.createRange(), range2 = range.duplicate();
|
var range = selection.createRange(), range2 = range.duplicate();
|
||||||
@ -356,32 +360,15 @@ var select = {};
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Some hacks for storing and re-storing the selection when the editor loses and regains focus.
|
// Some hacks for storing and re-storing the selection when the editor loses and regains focus.
|
||||||
select.selectionCoords = function (win) {
|
select.getBookmark = function (container) {
|
||||||
var selection = win.document.selection;
|
var from = select.cursorPos(container, true), to = select.cursorPos(container, false);
|
||||||
if (!selection) return null;
|
if (from && to) return {from: from, to: to};
|
||||||
var start = selection.createRange(), end = start.duplicate();
|
|
||||||
start.collapse(true);
|
|
||||||
end.collapse(false);
|
|
||||||
|
|
||||||
var body = win.document.body;
|
|
||||||
return {start: {x: start.boundingLeft + body.scrollLeft - 1,
|
|
||||||
y: start.boundingTop + body.scrollTop},
|
|
||||||
end: {x: end.boundingLeft + body.scrollLeft - 1,
|
|
||||||
y: end.boundingTop + body.scrollTop}};
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// Restore a stored selection.
|
// Restore a stored selection.
|
||||||
select.selectCoords = function(win, coords) {
|
select.setBookmark = function(container, mark) {
|
||||||
if (!coords) return;
|
if (!mark) return;
|
||||||
|
select.setCursorPos(container, mark.from, mark.to);
|
||||||
var range1 = win.document.body.createTextRange(), range2 = range1.duplicate();
|
|
||||||
// This can fail for various hard-to-handle reasons.
|
|
||||||
try {
|
|
||||||
range1.moveToPoint(coords.start.x, coords.start.y);
|
|
||||||
range2.moveToPoint(coords.end.x, coords.end.y);
|
|
||||||
range1.setEndPoint("EndToStart", range2);
|
|
||||||
range1.select();
|
|
||||||
} catch(e) {}
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
// W3C model
|
// W3C model
|
||||||
@ -407,7 +394,7 @@ var select = {};
|
|||||||
// ancestors with a suitable offset. This goes down the DOM tree
|
// ancestors with a suitable offset. This goes down the DOM tree
|
||||||
// until a 'leaf' is reached (or is it *up* the DOM tree?).
|
// until a 'leaf' is reached (or is it *up* the DOM tree?).
|
||||||
function normalize(point){
|
function normalize(point){
|
||||||
while (point.node.nodeType != 3 && point.node.nodeName != "BR") {
|
while (point.node.nodeType != 3 && !isBR(point.node)) {
|
||||||
var newNode = point.node.childNodes[point.offset] || point.node.nextSibling;
|
var newNode = point.node.childNodes[point.offset] || point.node.nextSibling;
|
||||||
point.offset = 0;
|
point.offset = 0;
|
||||||
while (!newNode && point.node.parentNode) {
|
while (!newNode && point.node.parentNode) {
|
||||||
@ -425,8 +412,17 @@ var select = {};
|
|||||||
};
|
};
|
||||||
|
|
||||||
select.selectMarked = function () {
|
select.selectMarked = function () {
|
||||||
if (!currentSelection || !currentSelection.changed) return;
|
var cs = currentSelection;
|
||||||
var win = currentSelection.window, range = win.document.createRange();
|
// on webkit-based browsers, it is apparently possible that the
|
||||||
|
// selection gets reset even when a node that is not one of the
|
||||||
|
// endpoints get messed with. the most common situation where
|
||||||
|
// this occurs is when a selection is deleted or overwitten. we
|
||||||
|
// check for that here.
|
||||||
|
function focusIssue() {
|
||||||
|
return cs.start.node == cs.end.node && cs.start.offset == 0 && cs.end.offset == 0;
|
||||||
|
}
|
||||||
|
if (!cs || !(cs.changed || (webkit && focusIssue()))) return;
|
||||||
|
var win = cs.window, range = win.document.createRange();
|
||||||
|
|
||||||
function setPoint(point, which) {
|
function setPoint(point, which) {
|
||||||
if (point.node) {
|
if (point.node) {
|
||||||
@ -442,8 +438,8 @@ var select = {};
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
setPoint(currentSelection.end, "End");
|
setPoint(cs.end, "End");
|
||||||
setPoint(currentSelection.start, "Start");
|
setPoint(cs.start, "Start");
|
||||||
selectRange(range, win);
|
selectRange(range, win);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -452,7 +448,7 @@ var select = {};
|
|||||||
var selection = window.getSelection();
|
var selection = window.getSelection();
|
||||||
selection.removeAllRanges();
|
selection.removeAllRanges();
|
||||||
selection.addRange(range);
|
selection.addRange(range);
|
||||||
};
|
}
|
||||||
function selectionRange(window) {
|
function selectionRange(window) {
|
||||||
var selection = window.getSelection();
|
var selection = window.getSelection();
|
||||||
if (!selection || selection.rangeCount == 0)
|
if (!selection || selection.rangeCount == 0)
|
||||||
@ -471,7 +467,7 @@ var select = {};
|
|||||||
var offset = start ? range.startOffset : range.endOffset;
|
var offset = start ? range.startOffset : range.endOffset;
|
||||||
// Work around (yet another) bug in Opera's selection model.
|
// Work around (yet another) bug in Opera's selection model.
|
||||||
if (window.opera && !start && range.endContainer == container && range.endOffset == range.startOffset + 1 &&
|
if (window.opera && !start && range.endContainer == container && range.endOffset == range.startOffset + 1 &&
|
||||||
container.childNodes[range.startOffset] && container.childNodes[range.startOffset].nodeName == "BR")
|
container.childNodes[range.startOffset] && isBR(container.childNodes[range.startOffset]))
|
||||||
offset--;
|
offset--;
|
||||||
|
|
||||||
// For text nodes, we look at the node itself if the cursor is
|
// For text nodes, we look at the node itself if the cursor is
|
||||||
@ -486,7 +482,7 @@ var select = {};
|
|||||||
// Occasionally, browsers will return the HTML node as
|
// Occasionally, browsers will return the HTML node as
|
||||||
// selection. If the offset is 0, we take the start of the frame
|
// selection. If the offset is 0, we take the start of the frame
|
||||||
// ('after null'), otherwise, we take the last node.
|
// ('after null'), otherwise, we take the last node.
|
||||||
else if (node.nodeName == "HTML") {
|
else if (node.nodeName.toUpperCase() == "HTML") {
|
||||||
return (offset == 1 ? null : container.lastChild);
|
return (offset == 1 ? null : container.lastChild);
|
||||||
}
|
}
|
||||||
// If the given node is our 'container', we just look up the
|
// If the given node is our 'container', we just look up the
|
||||||
@ -557,7 +553,7 @@ var select = {};
|
|||||||
if (!range) return;
|
if (!range) return;
|
||||||
|
|
||||||
var topNode = select.selectionTopNode(container, start);
|
var topNode = select.selectionTopNode(container, start);
|
||||||
while (topNode && topNode.nodeName != "BR")
|
while (topNode && !isBR(topNode))
|
||||||
topNode = topNode.previousSibling;
|
topNode = topNode.previousSibling;
|
||||||
|
|
||||||
range = range.cloneRange();
|
range = range.cloneRange();
|
||||||
@ -574,13 +570,17 @@ var select = {};
|
|||||||
range = win.document.createRange();
|
range = win.document.createRange();
|
||||||
|
|
||||||
function setPoint(node, offset, side) {
|
function setPoint(node, offset, side) {
|
||||||
|
if (offset == 0 && node && !node.nextSibling) {
|
||||||
|
range["set" + side + "After"](node);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
if (!node)
|
if (!node)
|
||||||
node = container.firstChild;
|
node = container.firstChild;
|
||||||
else
|
else
|
||||||
node = node.nextSibling;
|
node = node.nextSibling;
|
||||||
|
|
||||||
if (!node)
|
if (!node) return;
|
||||||
return;
|
|
||||||
|
|
||||||
if (offset == 0) {
|
if (offset == 0) {
|
||||||
range["set" + side + "Before"](node);
|
range["set" + side + "Before"](node);
|
||||||
|
@ -14,7 +14,7 @@
|
|||||||
// Make a stringstream stream out of an iterator that returns strings.
|
// Make a stringstream stream out of an iterator that returns strings.
|
||||||
// This is applied to the result of traverseDOM (see codemirror.js),
|
// This is applied to the result of traverseDOM (see codemirror.js),
|
||||||
// and the resulting stream is fed to the parser.
|
// and the resulting stream is fed to the parser.
|
||||||
window.stringStream = function(source){
|
var stringStream = function(source){
|
||||||
// String that's currently being iterated over.
|
// String that's currently being iterated over.
|
||||||
var current = "";
|
var current = "";
|
||||||
// Position in that string.
|
// Position in that string.
|
||||||
|
@ -5,7 +5,6 @@ var tokenizeJavaScript = (function() {
|
|||||||
// backslash) is encountered, or the end of the line is reached.
|
// backslash) is encountered, or the end of the line is reached.
|
||||||
function nextUntilUnescaped(source, end) {
|
function nextUntilUnescaped(source, end) {
|
||||||
var escaped = false;
|
var escaped = false;
|
||||||
var next;
|
|
||||||
while (!source.endOfLine()) {
|
while (!source.endOfLine()) {
|
||||||
var next = source.next();
|
var next = source.next();
|
||||||
if (next == end && !escaped)
|
if (next == end && !escaped)
|
||||||
@ -48,7 +47,7 @@ var tokenizeJavaScript = (function() {
|
|||||||
}();
|
}();
|
||||||
|
|
||||||
// Some helper regexps
|
// Some helper regexps
|
||||||
var isOperatorChar = /[+\-*&%\/=<>!?|]/;
|
var isOperatorChar = /[+\-*&%=<>!?|]/;
|
||||||
var isHexDigit = /[0-9A-Fa-f]/;
|
var isHexDigit = /[0-9A-Fa-f]/;
|
||||||
var isWordChar = /[\w\$_]/;
|
var isWordChar = /[\w\$_]/;
|
||||||
|
|
||||||
@ -66,7 +65,7 @@ var tokenizeJavaScript = (function() {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
// The token reader, inteded to be used by the tokenizer from
|
// The token reader, intended to be used by the tokenizer from
|
||||||
// tokenize.js (through jsTokenState). Advances the source stream
|
// tokenize.js (through jsTokenState). Advances the source stream
|
||||||
// over a token, and returns an object containing the type and style
|
// over a token, and returns an object containing the type and style
|
||||||
// of that token.
|
// of that token.
|
||||||
|
@ -26,11 +26,10 @@
|
|||||||
// delay (of no input) after which it commits a set of changes, and,
|
// delay (of no input) after which it commits a set of changes, and,
|
||||||
// unfortunately, the 'parent' window -- a window that is not in
|
// unfortunately, the 'parent' window -- a window that is not in
|
||||||
// designMode, and on which setTimeout works in every browser.
|
// designMode, and on which setTimeout works in every browser.
|
||||||
function History(container, maxDepth, commitDelay, editor, onChange) {
|
function History(container, maxDepth, commitDelay, editor) {
|
||||||
this.container = container;
|
this.container = container;
|
||||||
this.maxDepth = maxDepth; this.commitDelay = commitDelay;
|
this.maxDepth = maxDepth; this.commitDelay = commitDelay;
|
||||||
this.editor = editor; this.parent = editor.parent;
|
this.editor = editor; this.parent = editor.parent;
|
||||||
this.onChange = onChange;
|
|
||||||
// This line object represents the initial, empty editor.
|
// This line object represents the initial, empty editor.
|
||||||
var initial = {text: "", from: null, to: null};
|
var initial = {text: "", from: null, to: null};
|
||||||
// As the borders between lines are represented by BR elements, the
|
// As the borders between lines are represented by BR elements, the
|
||||||
@ -73,7 +72,7 @@ History.prototype = {
|
|||||||
// shadow in the redo history.
|
// shadow in the redo history.
|
||||||
var item = this.history.pop();
|
var item = this.history.pop();
|
||||||
this.redoHistory.push(this.updateTo(item, "applyChain"));
|
this.redoHistory.push(this.updateTo(item, "applyChain"));
|
||||||
if (this.onChange) this.onChange();
|
this.notifyEnvironment();
|
||||||
return this.chainNode(item);
|
return this.chainNode(item);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -85,7 +84,7 @@ History.prototype = {
|
|||||||
// The inverse of undo, basically.
|
// The inverse of undo, basically.
|
||||||
var item = this.redoHistory.pop();
|
var item = this.redoHistory.pop();
|
||||||
this.addUndoLevel(this.updateTo(item, "applyChain"));
|
this.addUndoLevel(this.updateTo(item, "applyChain"));
|
||||||
if (this.onChange) this.onChange();
|
this.notifyEnvironment();
|
||||||
return this.chainNode(item);
|
return this.chainNode(item);
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
@ -109,6 +108,7 @@ History.prototype = {
|
|||||||
from = end;
|
from = end;
|
||||||
}
|
}
|
||||||
this.pushChains([chain], from == null && to == null);
|
this.pushChains([chain], from == null && to == null);
|
||||||
|
this.notifyEnvironment();
|
||||||
},
|
},
|
||||||
|
|
||||||
pushChains: function(chains, doNotHighlight) {
|
pushChains: function(chains, doNotHighlight) {
|
||||||
@ -162,7 +162,7 @@ History.prototype = {
|
|||||||
if (chains.length) {
|
if (chains.length) {
|
||||||
this.addUndoLevel(this.updateTo(chains, "linkChain"));
|
this.addUndoLevel(this.updateTo(chains, "linkChain"));
|
||||||
this.redoHistory = [];
|
this.redoHistory = [];
|
||||||
if (this.onChange) this.onChange();
|
this.notifyEnvironment();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -191,6 +191,13 @@ History.prototype = {
|
|||||||
this.editor.scheduleHighlight();
|
this.editor.scheduleHighlight();
|
||||||
},
|
},
|
||||||
|
|
||||||
|
notifyEnvironment: function() {
|
||||||
|
// Used by the line-wrapping line-numbering code.
|
||||||
|
if (window.frameElement && window.frameElement.CodeMirror.updateNumbers)
|
||||||
|
window.frameElement.CodeMirror.updateNumbers();
|
||||||
|
if (this.onChange) this.onChange();
|
||||||
|
},
|
||||||
|
|
||||||
// Link a chain into the DOM nodes (or the first/last links for null
|
// Link a chain into the DOM nodes (or the first/last links for null
|
||||||
// nodes).
|
// nodes).
|
||||||
linkChain: function(chain) {
|
linkChain: function(chain) {
|
||||||
@ -250,7 +257,7 @@ History.prototype = {
|
|||||||
function buildLine(node) {
|
function buildLine(node) {
|
||||||
var text = [];
|
var text = [];
|
||||||
for (var cur = node ? node.nextSibling : self.container.firstChild;
|
for (var cur = node ? node.nextSibling : self.container.firstChild;
|
||||||
cur && cur.nodeName != "BR"; cur = cur.nextSibling)
|
cur && !isBR(cur); cur = cur.nextSibling)
|
||||||
if (cur.currentText) text.push(cur.currentText);
|
if (cur.currentText) text.push(cur.currentText);
|
||||||
return {from: node, to: cur, text: cleanText(text.join(""))};
|
return {from: node, to: cur, text: cleanText(text.join(""))};
|
||||||
}
|
}
|
||||||
@ -275,7 +282,7 @@ History.prototype = {
|
|||||||
// Get the BR element after/before the given node.
|
// Get the BR element after/before the given node.
|
||||||
function nextBR(node, dir) {
|
function nextBR(node, dir) {
|
||||||
var link = dir + "Sibling", search = node[link];
|
var link = dir + "Sibling", search = node[link];
|
||||||
while (search && search.nodeName != "BR")
|
while (search && !isBR(search))
|
||||||
search = search[link];
|
search = search[link];
|
||||||
return search;
|
return search;
|
||||||
}
|
}
|
||||||
|
@ -1,9 +1,5 @@
|
|||||||
/* A few useful utility functions. */
|
/* A few useful utility functions. */
|
||||||
|
|
||||||
var internetExplorer = document.selection && window.ActiveXObject && /MSIE/.test(navigator.userAgent);
|
|
||||||
var webkit = /AppleWebKit/.test(navigator.userAgent);
|
|
||||||
var safari = /Apple Computers, Inc/.test(navigator.vendor);
|
|
||||||
|
|
||||||
// Capture a method on an object.
|
// Capture a method on an object.
|
||||||
function method(obj, name) {
|
function method(obj, name) {
|
||||||
return function() {obj[name].apply(obj, arguments);};
|
return function() {obj[name].apply(obj, arguments);};
|
||||||
@ -112,7 +108,7 @@ function addEventHandler(node, type, handler, removeFunc) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function nodeText(node) {
|
function nodeText(node) {
|
||||||
return node.innerText || node.textContent || node.nodeValue || "";
|
return node.textContent || node.innerText || node.nodeValue || "";
|
||||||
}
|
}
|
||||||
|
|
||||||
function nodeTop(node) {
|
function nodeTop(node) {
|
||||||
@ -123,3 +119,12 @@ function nodeTop(node) {
|
|||||||
}
|
}
|
||||||
return top;
|
return top;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function isBR(node) {
|
||||||
|
var nn = node.nodeName;
|
||||||
|
return nn == "BR" || nn == "br";
|
||||||
|
}
|
||||||
|
function isSpan(node) {
|
||||||
|
var nn = node.nodeName;
|
||||||
|
return nn == "SPAN" || nn == "span";
|
||||||
|
}
|
||||||
|
1077
public/javascripts/admin/plugins/fancybox.js
Executable file
86
public/javascripts/admin/plugins/plupload/gears_init.js
Normal file
@ -0,0 +1,86 @@
|
|||||||
|
// Copyright 2007, Google Inc.
|
||||||
|
//
|
||||||
|
// Redistribution and use in source and binary forms, with or without
|
||||||
|
// modification, are permitted provided that the following conditions are met:
|
||||||
|
//
|
||||||
|
// 1. Redistributions of source code must retain the above copyright notice,
|
||||||
|
// this list of conditions and the following disclaimer.
|
||||||
|
// 2. Redistributions in binary form must reproduce the above copyright notice,
|
||||||
|
// this list of conditions and the following disclaimer in the documentation
|
||||||
|
// and/or other materials provided with the distribution.
|
||||||
|
// 3. Neither the name of Google Inc. nor the names of its contributors may be
|
||||||
|
// used to endorse or promote products derived from this software without
|
||||||
|
// specific prior written permission.
|
||||||
|
//
|
||||||
|
// THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
|
||||||
|
// WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
|
||||||
|
// MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
|
||||||
|
// EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
||||||
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
// PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
|
||||||
|
// OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
|
||||||
|
// WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
|
||||||
|
// OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
|
||||||
|
// ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
//
|
||||||
|
// Sets up google.gears.*, which is *the only* supported way to access Gears.
|
||||||
|
//
|
||||||
|
// Circumvent this file at your own risk!
|
||||||
|
//
|
||||||
|
// In the future, Gears may automatically define google.gears.* without this
|
||||||
|
// file. Gears may use these objects to transparently fix bugs and compatibility
|
||||||
|
// issues. Applications that use the code below will continue to work seamlessly
|
||||||
|
// when that happens.
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
// We are already defined. Hooray!
|
||||||
|
if (window.google && google.gears) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var factory = null;
|
||||||
|
|
||||||
|
// Firefox
|
||||||
|
if (typeof GearsFactory != 'undefined') {
|
||||||
|
factory = new GearsFactory();
|
||||||
|
} else {
|
||||||
|
// IE
|
||||||
|
try {
|
||||||
|
factory = new ActiveXObject('Gears.Factory');
|
||||||
|
// privateSetGlobalObject is only required and supported on WinCE.
|
||||||
|
if (factory.getBuildInfo().indexOf('ie_mobile') != -1) {
|
||||||
|
factory.privateSetGlobalObject(this);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Safari
|
||||||
|
if ((typeof navigator.mimeTypes != 'undefined')
|
||||||
|
&& navigator.mimeTypes["application/x-googlegears"]) {
|
||||||
|
factory = document.createElement("object");
|
||||||
|
factory.style.display = "none";
|
||||||
|
factory.width = 0;
|
||||||
|
factory.height = 0;
|
||||||
|
factory.type = "application/x-googlegears";
|
||||||
|
document.documentElement.appendChild(factory);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// *Do not* define any objects if Gears is not installed. This mimics the
|
||||||
|
// behavior of Gears defining the objects in the future.
|
||||||
|
if (!factory) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Now set up the objects, being careful not to overwrite anything.
|
||||||
|
//
|
||||||
|
// Note: In Internet Explorer for Windows Mobile, you can't add properties to
|
||||||
|
// the window object. However, global objects are automatically added as
|
||||||
|
// properties of the window object in all browsers.
|
||||||
|
if (!window.google) {
|
||||||
|
google = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!google.gears) {
|
||||||
|
google.gears = {factory: factory};
|
||||||
|
}
|
||||||
|
})();
|
1
public/javascripts/admin/plugins/plupload/jquery.plupload.queue.min.js
vendored
Normal file
1
public/javascripts/admin/plugins/plupload/plupload.browserplus.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(a){a.runtimes.BrowserPlus=a.addRuntime("browserplus",{getFeatures:function(){return{dragdrop:true,jpgresize:true,pngresize:true,chunks:true,progress:true,multipart:true}},init:function(g,i){var e=window.BrowserPlus,h={},d=g.settings,c=d.resize;function f(n){var m,l,j=[],k,o;for(l=0;l<n.length;l++){k=n[l];o=a.guid();h[o]=k;j.push(new a.File(o,k.name,k.size))}if(l){g.trigger("FilesAdded",j)}}function b(){g.bind("PostInit",function(){var m,k=d.drop_element,o=g.id+"_droptarget",j=document.getElementById(k),l;function p(r,q){e.DragAndDrop.AddDropTarget({id:r},function(s){e.DragAndDrop.AttachCallbacks({id:r,hover:function(t){if(!t&&q){q()}},drop:function(t){if(q){q()}f(t)}},function(){})})}function n(){document.getElementById(o).style.top="-1000px"}if(j){if(document.attachEvent&&(/MSIE/gi).test(navigator.userAgent)){m=document.createElement("div");m.setAttribute("id",o);a.extend(m.style,{position:"absolute",top:"-1000px",background:"red",filter:"alpha(opacity=0)",opacity:0});document.body.appendChild(m);a.addEvent(j,"dragenter",function(r){var q,s;q=document.getElementById(k);s=a.getPos(q);a.extend(document.getElementById(o).style,{top:s.y+"px",left:s.x+"px",width:q.offsetWidth+"px",height:q.offsetHeight+"px"})});p(o,n)}else{p(k)}}a.addEvent(document.getElementById(d.browse_button),"click",function(v){var t=[],r,q,u=d.filters,s;v.preventDefault();for(r=0;r<u.length;r++){s=u[r].extensions.split(",");for(q=0;q<s.length;q++){t.push(a.mimeTypes[s[q]])}}e.FileBrowse.OpenBrowseDialog({mimeTypes:t},function(w){if(w.success){f(w.value)}})});j=m=null});g.bind("UploadFile",function(n,k){var m=h[k.id],j={},l=n.settings.chunk_size,o,p=[];function r(s,u){var t;if(k.status==a.FAILED){return}j.name=k.target_name||k.name;if(l){j.chunk=s;j.chunks=u}t=p.shift();e.Uploader.upload({url:a.buildUrl(n.settings.url,j),files:{file:t},cookies:document.cookies,postvars:n.settings.multipart_params,progressCallback:function(x){var w,v=0;o[s]=parseInt(x.filePercent*t.size/100,10);for(w=0;w<o.length;w++){v+=o[w]}k.loaded=v;n.trigger("UploadProgress",k)}},function(w){var v,x;if(w.success){v=w.value.statusCode;if(l){n.trigger("ChunkUploaded",k,{chunk:s,chunks:u,response:w.value.body,status:v})}if(p.length>0){r(++s,u)}else{k.status=a.DONE;n.trigger("FileUploaded",k,{response:w.value.body,status:v});if(v>=400){n.trigger("Error",{code:a.HTTP_ERROR,message:"HTTP Error.",file:k,status:v})}}}else{n.trigger("Error",{code:a.GENERIC_ERROR,message:"Generic Error.",file:k,details:w.error})}})}function q(s){k.size=s.size;if(l){e.FileAccess.chunk({file:s,chunkSize:l},function(v){if(v.success){var w=v.value,t=w.length;o=Array(t);for(var u=0;u<t;u++){o[u]=0;p.push(w[u])}r(0,t)}})}else{o=Array(1);p.push(s);r(0,1)}}if(c&&/\.(png|jpg|jpeg)$/i.test(k.name)){BrowserPlus.ImageAlter.transform({file:m,quality:c.quality||90,actions:[{scale:{maxwidth:c.width,maxheight:c.height}}]},function(s){if(s.success){q(s.value.file)}})}else{q(m)}});i({success:true})}if(e){e.init(function(k){var j=[{service:"Uploader",version:"3"},{service:"DragAndDrop",version:"1"},{service:"FileBrowse",version:"1"},{service:"FileAccess",version:"2"}];if(c){j.push({service:"ImageAlter",version:"4"})}if(k.success){e.require({services:j},function(l){if(l.success){b()}else{i()}})}else{i()}})}else{i()}}})})(plupload);
|
1
public/javascripts/admin/plugins/plupload/plupload.flash.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(c){var a={};function b(){var d;try{d=navigator.plugins["Shockwave Flash"];d=d.description}catch(f){try{d=new ActiveXObject("ShockwaveFlash.ShockwaveFlash").GetVariable("$version")}catch(e){d="0.0"}}d=d.match(/\d+/g);return parseFloat(d[0]+"."+d[1])}c.flash={trigger:function(f,d,e){setTimeout(function(){var j=a[f],h,g;if(j){j.trigger("Flash:"+d,e)}},0)}};c.runtimes.Flash=c.addRuntime("flash",{getFeatures:function(){return{jpgresize:true,pngresize:true,chunks:true,progress:true,multipart:true}},init:function(g,l){var k,f,h,e,m=0,d=document.body;if(b()<10){l({success:false});return}a[g.id]=g;k=document.getElementById(g.settings.browse_button);f=document.createElement("div");f.id=g.id+"_flash_container";c.extend(f.style,{position:"absolute",top:"0px",background:g.settings.shim_bgcolor||"transparent",zIndex:99999,width:"100%",height:"100%"});f.className="plupload flash";if(g.settings.container){d=document.getElementById(g.settings.container);d.style.position="relative"}d.appendChild(f);h="id="+escape(g.id);f.innerHTML='<object id="'+g.id+'_flash" width="100%" height="100%" style="outline:0" type="application/x-shockwave-flash" data="'+g.settings.flash_swf_url+'"><param name="movie" value="'+g.settings.flash_swf_url+'" /><param name="flashvars" value="'+h+'" /><param name="wmode" value="transparent" /><param name="allowscriptaccess" value="always" /></object>';function j(){return document.getElementById(g.id+"_flash")}function i(){if(m++>5000){l({success:false});return}if(!e){setTimeout(i,1)}}i();k=f=null;g.bind("Flash:Init",function(){var p={},o,n=g.settings.resize||{};e=true;j().setFileFilters(g.settings.filters,g.settings.multi_selection);g.bind("UploadFile",function(q,r){var s=q.settings;j().uploadFile(p[r.id],c.buildUrl(s.url,{name:r.target_name||r.name}),{chunk_size:s.chunk_size,width:n.width,height:n.height,quality:n.quality||90,multipart:s.multipart,multipart_params:s.multipart_params,file_data_name:s.file_data_name,format:/\.(jpg|jpeg)$/i.test(r.name)?"jpg":"png",headers:s.headers})});g.bind("Flash:UploadProcess",function(r,q){var s=r.getFile(p[q.id]);if(s.status!=c.FAILED){s.loaded=q.loaded;s.size=q.size;r.trigger("UploadProgress",s)}});g.bind("Flash:UploadChunkComplete",function(q,s){var t,r=q.getFile(p[s.id]);t={chunk:s.chunk,chunks:s.chunks,response:s.text};q.trigger("ChunkUploaded",r,t);if(r.status!=c.FAILED){j().uploadNextChunk()}if(s.chunk==s.chunks-1){r.status=c.DONE;q.trigger("FileUploaded",r,{response:s.text})}});g.bind("Flash:SelectFiles",function(q,t){var s,r,u=[],v;for(r=0;r<t.length;r++){s=t[r];v=c.guid();p[v]=s.id;p[s.id]=v;u.push(new c.File(v,s.name,s.size))}if(u.length){g.trigger("FilesAdded",u)}});g.bind("Flash:SecurityError",function(q,r){g.trigger("Error",{code:c.SECURITY_ERROR,message:"Security error.",details:r.message,file:g.getFile(p[r.id])})});g.bind("Flash:GenericError",function(q,r){g.trigger("Error",{code:c.GENERIC_ERROR,message:"Generic error.",details:r.message,file:g.getFile(p[r.id])})});g.bind("Flash:IOError",function(q,r){g.trigger("Error",{code:c.IO_ERROR,message:"IO error.",details:r.message,file:g.getFile(p[r.id])})});g.bind("QueueChanged",function(q){g.refresh()});g.bind("FilesRemoved",function(q,s){var r;for(r=0;r<s.length;r++){j().removeFile(p[s[r].id])}});g.bind("StateChanged",function(q){g.refresh()});g.bind("Refresh",function(q){var r,s,t;j().setFileFilters(g.settings.filters,g.settings.multi_selection);r=document.getElementById(q.settings.browse_button);s=c.getPos(r,document.getElementById(q.settings.container));t=c.getSize(r);c.extend(document.getElementById(q.id+"_flash_container").style,{top:s.y+"px",left:s.x+"px",width:t.w+"px",height:t.h+"px"})});l({success:true})})}})})(plupload);
|
BIN
public/javascripts/admin/plugins/plupload/plupload.flash.swf
Normal file
1
public/javascripts/admin/plugins/plupload/plupload.gears.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(b){var c={};function a(i,e,k,j,d){var l,g,f,h;g=google.gears.factory.create("beta.canvas");g.decode(i);h=Math.min(e/g.width,k/g.height);if(h<1){e=Math.round(g.width*h);k=Math.round(g.height*h)}else{e=g.width;k=g.height}g.resize(e,k);return g.encode(d,{quality:j/100})}b.runtimes.Gears=b.addRuntime("gears",{getFeatures:function(){return{dragdrop:true,jpgresize:true,pngresize:true,chunks:true,progress:true,multipart:true}},init:function(g,i){var h;if(!window.google||!google.gears){return i({success:false})}try{h=google.gears.factory.create("beta.desktop")}catch(f){return i({success:false})}function d(k){var j,e,l=[],m;for(e=0;e<k.length;e++){j=k[e];m=b.guid();c[m]=j.blob;l.push(new b.File(m,j.name,j.blob.length))}g.trigger("FilesAdded",l)}g.bind("PostInit",function(){var j=g.settings,e=document.getElementById(j.drop_element);if(e){b.addEvent(e,"dragover",function(k){h.setDropEffect(k,"copy");k.preventDefault()});b.addEvent(e,"drop",function(l){var k=h.getDragData(l,"application/x-gears-files");if(k){d(k.files)}l.preventDefault()});e=0}b.addEvent(document.getElementById(j.browse_button),"click",function(o){var n=[],l,k,m;o.preventDefault();for(l=0;l<j.filters.length;l++){m=j.filters[l].extensions.split(",");for(k=0;k<m.length;k++){n.push("."+m[k])}}h.openFiles(d,{singleFile:!j.multi_selection,filter:n})})});g.bind("UploadFile",function(o,l){var q=0,p,m,n=0,k=o.settings.resize,e;m=o.settings.chunk_size;e=m>0;p=Math.ceil(l.size/m);if(!e){m=l.size;p=1}if(k&&/\.(png|jpg|jpeg)$/i.test(l.name)){c[l.id]=a(c[l.id],k.width,k.height,k.quality||90,/\.png$/i.test(l.name)?"image/png":"image/jpeg")}l.size=c[l.id].length;function j(){var u,w,s=o.settings.multipart,r=0,v={name:l.target_name||l.name};function t(y){var x,C="----pluploadboundary"+b.guid(),A="--",B="\r\n",z;if(s){u.setRequestHeader("Content-Type","multipart/form-data; boundary="+C);x=google.gears.factory.create("beta.blobbuilder");b.each(o.settings.multipart_params,function(E,D){x.append(A+C+B+'Content-Disposition: form-data; name="'+D+'"'+B+B);x.append(E+B)});x.append(A+C+B+'Content-Disposition: form-data; name="'+o.settings.file_data_name+'"; filename="'+l.name+'"'+B+"Content-Type: application/octet-stream"+B+B);x.append(y);x.append(B+A+C+A+B);z=x.getAsBlob();r=z.length-y.length;y=z}u.send(y)}if(l.status==b.DONE||l.status==b.FAILED||o.state==b.STOPPED){return}if(e){v.chunk=q;v.chunks=p}w=Math.min(m,l.size-(q*m));u=google.gears.factory.create("beta.httprequest");u.open("POST",b.buildUrl(o.settings.url,v));if(!s){u.setRequestHeader("Content-Disposition",'attachment; filename="'+l.name+'"');u.setRequestHeader("Content-Type","application/octet-stream")}b.each(o.settings.headers,function(y,x){u.setRequestHeader(x,y)});u.upload.onprogress=function(x){l.loaded=n+x.loaded-r;o.trigger("UploadProgress",l)};u.onreadystatechange=function(){var x;if(u.readyState==4){if(u.status==200){x={chunk:q,chunks:p,response:u.responseText,status:u.status};o.trigger("ChunkUploaded",l,x);if(x.cancelled){l.status=b.FAILED;return}n+=w;if(++q>=p){l.status=b.DONE;o.trigger("FileUploaded",l,{response:u.responseText,status:u.status})}else{j()}}else{o.trigger("Error",{code:b.HTTP_ERROR,message:"HTTP Error.",file:l,chunk:q,chunks:p,status:u.status})}}};if(q<p){t(c[l.id].slice(q*m,w))}}j()});i({success:true})}})})(plupload);
|
1
public/javascripts/admin/plugins/plupload/plupload.html4.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(a){a.runtimes.Html4=a.addRuntime("html4",{getFeatures:function(){return{multipart:true}},init:function(f,g){var d={},c,b;function e(l){var k,j,m=[],n,h;h=l.value.replace(/\\/g,"/");h=h.substring(h.length,h.lastIndexOf("/")+1);n=a.guid();k=new a.File(n,h);d[n]=k;k.input=l;m.push(k);if(m.length){f.trigger("FilesAdded",m)}}f.bind("Init",function(p){var h,x,v,t=[],o,u,m=p.settings.filters,l,s,r=/MSIE/.test(navigator.userAgent),k="javascript",w,j=document.body,n;if(f.settings.container){j=document.getElementById(f.settings.container);j.style.position="relative"}c=(typeof p.settings.form=="string")?document.getElementById(p.settings.form):p.settings.form;if(!c){n=document.getElementById(f.settings.browse_button);for(;n;n=n.parentNode){if(n.nodeName=="FORM"){c=n}}}if(!c){c=document.createElement("form");c.style.display="inline";n=document.getElementById(f.settings.container);n.parentNode.insertBefore(c,n);c.appendChild(n)}c.setAttribute("method","post");c.setAttribute("enctype","multipart/form-data");a.each(p.settings.multipart_params,function(z,y){var i=document.createElement("input");a.extend(i,{type:"hidden",name:y,value:z});c.appendChild(i)});b=document.createElement("iframe");b.setAttribute("src",k+':""');b.setAttribute("name",p.id+"_iframe");b.setAttribute("id",p.id+"_iframe");b.style.display="none";a.addEvent(b,"load",function(B){var C=B.target,z=f.currentfile,A;try{A=C.contentWindow.document||C.contentDocument||window.frames[C.id].document}catch(y){p.trigger("Error",{code:a.SECURITY_ERROR,message:"Security error.",file:z});return}if(A.location.href=="about:blank"||!z){return}var i=A.documentElement.innerText||A.documentElement.textContent;if(i!=""){z.status=a.DONE;z.loaded=1025;z.percent=100;if(z.input){z.input.removeAttribute("name")}p.trigger("UploadProgress",z);p.trigger("FileUploaded",z,{response:i});if(c.tmpAction){c.setAttribute("action",c.tmpAction)}if(c.tmpTarget){c.setAttribute("target",c.tmpTarget)}}});c.appendChild(b);if(r){window.frames[b.id].name=b.name}x=document.createElement("div");x.id=p.id+"_iframe_container";for(o=0;o<m.length;o++){l=m[o].extensions.split(/,/);for(u=0;u<l.length;u++){s=a.mimeTypes[l[u]];if(s){t.push(s)}}}a.extend(x.style,{position:"absolute",background:"transparent",width:"100px",height:"100px",overflow:"hidden",zIndex:99999,opacity:0});w=f.settings.shim_bgcolor;if(w){a.extend(x.style,{background:w,opacity:1})}x.className="plupload_iframe";j.appendChild(x);function q(){v=document.createElement("input");v.setAttribute("type","file");v.setAttribute("accept",t.join(","));v.setAttribute("size",1);a.extend(v.style,{width:"100%",height:"100%",opacity:0});if(r){a.extend(v.style,{filter:"alpha(opacity=0)"})}a.addEvent(v,"change",function(i){var y=i.target;if(y.value){q();y.style.display="none";e(y)}});x.appendChild(v);return true}q()});f.bind("Refresh",function(h){var i,j,k;i=document.getElementById(f.settings.browse_button);j=a.getPos(i,document.getElementById(h.settings.container));k=a.getSize(i);a.extend(document.getElementById(f.id+"_iframe_container").style,{top:j.y+"px",left:j.x+"px",width:k.w+"px",height:k.h+"px"})});f.bind("UploadFile",function(h,i){if(i.status==a.DONE||i.status==a.FAILED||h.state==a.STOPPED){return}if(!i.input){i.status=a.ERROR;return}i.input.setAttribute("name",h.settings.file_data_name);c.tmpAction=c.getAttribute("action");c.setAttribute("action",a.buildUrl(h.settings.url,{name:i.target_name||i.name}));c.tmpTarget=c.getAttribute("target");c.setAttribute("target",b.name);this.currentfile=i;c.submit()});f.bind("FilesRemoved",function(h,k){var j,l;for(j=0;j<k.length;j++){l=k[j].input;if(l){l.parentNode.removeChild(l)}}});g({success:true})}})})(plupload);
|
1
public/javascripts/admin/plugins/plupload/plupload.html5.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(b){function a(i,l,j,c,k){var e,d,h,g,f;e=document.createElement("canvas");e.style.display="none";document.body.appendChild(e);d=e.getContext("2d");h=new Image();h.onload=function(){var o,m,n;f=Math.min(l/h.width,j/h.height);if(f<1){o=Math.round(h.width*f);m=Math.round(h.height*f)}else{o=h.width;m=h.height}e.width=o;e.height=m;d.drawImage(h,0,0,o,m);g=e.toDataURL(c);g=g.substring(g.indexOf("base64,")+7);g=atob(g);e.parentNode.removeChild(e);k({success:true,data:g})};h.src=i}b.runtimes.Html5=b.addRuntime("html5",{getFeatures:function(){var g,d,f,e,c;d=f=e=c=false;if(window.XMLHttpRequest){g=new XMLHttpRequest();f=!!g.upload;d=!!(g.sendAsBinary||g.upload)}if(d){e=!!(File&&File.prototype.getAsDataURL);c=!!(File&&File.prototype.slice)}return{html5:d,dragdrop:window.mozInnerScreenX!==undefined||c,jpgresize:e,pngresize:e,multipart:e,progress:f}},init:function(e,f){var c={};function d(k){var h,g,j=[],l;for(g=0;g<k.length;g++){h=k[g];l=b.guid();c[l]=h;j.push(new b.File(l,h.fileName,h.fileSize))}if(j.length){e.trigger("FilesAdded",j)}}if(!this.getFeatures().html5){f({success:false});return}e.bind("Init",function(l){var p,n=[],k,o,h=l.settings.filters,j,m,g=document.body;p=document.createElement("div");p.id=l.id+"_html5_container";for(k=0;k<h.length;k++){j=h[k].extensions.split(/,/);for(o=0;o<j.length;o++){m=b.mimeTypes[j[o]];if(m){n.push(m)}}}b.extend(p.style,{position:"absolute",background:e.settings.shim_bgcolor||"transparent",width:"100px",height:"100px",overflow:"hidden",zIndex:99999,opacity:e.settings.shim_bgcolor?"":0});p.className="plupload html5";if(e.settings.container){g=document.getElementById(e.settings.container);g.style.position="relative"}g.appendChild(p);p.innerHTML='<input id="'+e.id+'_html5" style="width:100%;" type="file" accept="'+n.join(",")+'" '+(e.settings.multi_selection?'multiple="multiple"':"")+" />";document.getElementById(e.id+"_html5").onchange=function(){d(this.files);this.value=""}});e.bind("PostInit",function(){var g=document.getElementById(e.settings.drop_element);if(g){b.addEvent(g,"dragover",function(h){h.preventDefault()});b.addEvent(g,"drop",function(i){var h=i.dataTransfer;if(h&&h.files){d(h.files)}i.preventDefault()})}});e.bind("Refresh",function(g){var h,i,j;h=document.getElementById(e.settings.browse_button);i=b.getPos(h,document.getElementById(g.settings.container));j=b.getSize(h);b.extend(document.getElementById(e.id+"_html5_container").style,{top:i.y+"px",left:i.x+"px",width:j.w+"px",height:j.h+"px"})});e.bind("UploadFile",function(g,j){var n=new XMLHttpRequest(),i=n.upload,h=g.settings.resize,m,l=0;function k(o){var s="----pluploadboundary"+b.guid(),q="--",r="\r\n",p="";if(g.settings.multipart){n.setRequestHeader("Content-Type","multipart/form-data; boundary="+s);b.each(g.settings.multipart_params,function(u,t){p+=q+s+r+'Content-Disposition: form-data; name="'+t+'"'+r+r;p+=u+r});p+=q+s+r+'Content-Disposition: form-data; name="'+g.settings.file_data_name+'"; filename="'+j.name+'"'+r+"Content-Type: application/octet-stream"+r+r+o+r+q+s+q+r;l=p.length-o.length;o=p}n.sendAsBinary(o)}if(j.status==b.DONE||j.status==b.FAILED||g.state==b.STOPPED){return}if(i){i.onprogress=function(o){j.loaded=o.loaded-l;g.trigger("UploadProgress",j)}}n.onreadystatechange=function(){var o;if(n.readyState==4){try{o=n.status}catch(p){o=0}j.status=b.DONE;j.loaded=j.size;g.trigger("UploadProgress",j);g.trigger("FileUploaded",j,{response:n.responseText,status:o});if(o>=400){g.trigger("Error",{code:b.HTTP_ERROR,message:"HTTP Error.",file:j,status:o})}}};n.open("post",b.buildUrl(g.settings.url,{name:j.target_name||j.name}),true);n.setRequestHeader("Content-Type","application/octet-stream");b.each(g.settings.headers,function(p,o){n.setRequestHeader(o,p)});m=c[j.id];if(n.sendAsBinary){if(h&&/\.(png|jpg|jpeg)$/i.test(j.name)){a(m.getAsDataURL(),h.width,h.height,/\.png$/i.test(j.name)?"image/png":"image/jpeg",function(o){if(o.success){j.size=o.data.length;k(o.data)}else{k(m.getAsBinary())}})}else{k(m.getAsBinary())}}else{n.send(m)}});f({success:true})}})})(plupload);
|
1
public/javascripts/admin/plugins/plupload/plupload.js
Normal file
1
public/javascripts/admin/plugins/plupload/plupload.silverlight.min.js
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
(function(c){var a={};function b(l){var k,j=typeof l,h,e,g,f;if(j==="string"){k="\bb\tt\nn\ff\rr\"\"''\\\\";return'"'+l.replace(/([\u0080-\uFFFF\x00-\x1f\"])/g,function(n,m){var i=k.indexOf(m);if(i+1){return"\\"+k.charAt(i+1)}n=m.charCodeAt().toString(16);return"\\u"+"0000".substring(n.length)+n})+'"'}if(j=="object"){e=l.length!==h;k="";if(e){for(g=0;g<l.length;g++){if(k){k+=","}k+=b(l[g])}k="["+k+"]"}else{for(f in l){if(l.hasOwnProperty(f)){if(k){k+=","}k+=b(f)+":"+b(l[f])}}k="{"+k+"}"}return k}if(l===h){return"null"}return""+l}function d(o){var r=false,f=null,k=null,g,h,i,q,j,m=0;try{try{k=new ActiveXObject("AgControl.AgControl");if(k.IsVersionSupported(o)){r=true}k=null}catch(n){var l=navigator.plugins["Silverlight Plug-In"];if(l){g=l.description;if(g==="1.0.30226.2"){g="2.0.30226.2"}h=g.split(".");while(h.length>3){h.pop()}while(h.length<4){h.push(0)}i=o.split(".");while(i.length>4){i.pop()}do{q=parseInt(i[m],10);j=parseInt(h[m],10);m++}while(m<i.length&&q===j);if(q<=j&&!isNaN(q)){r=true}}}}catch(p){r=false}return r}c.silverlight={trigger:function(j,f){var h=a[j],g,e;if(h){e=c.toArray(arguments).slice(1);e[0]="Silverlight:"+f;setTimeout(function(){h.trigger.apply(h,e)},0)}}};c.runtimes.Silverlight=c.addRuntime("silverlight",{getFeatures:function(){return{jpgresize:true,pngresize:true,chunks:true,progress:true,multipart:true}},init:function(l,m){var k,h="",j=l.settings.filters,g,f=document.body;if(!d("2.0.31005.0")||(window.opera&&window.opera.buildNumber)){m({success:false});return}a[l.id]=l;k=document.createElement("div");k.id=l.id+"_silverlight_container";c.extend(k.style,{position:"absolute",top:"0px",background:l.settings.shim_bgcolor||"transparent",zIndex:99999,width:"100px",height:"100px",overflow:"hidden",opacity:l.settings.shim_bgcolor?"":0.01});k.className="plupload silverlight";if(l.settings.container){f=document.getElementById(l.settings.container);f.style.position="relative"}f.appendChild(k);for(g=0;g<j.length;g++){h+=(h!=""?"|":"")+j[g].title+" | *."+j[g].extensions.replace(/,/g,";*.")}k.innerHTML='<object id="'+l.id+'_silverlight" data="data:application/x-silverlight," type="application/x-silverlight-2" style="outline:none;" width="1024" height="1024"><param name="source" value="'+l.settings.silverlight_xap_url+'"/><param name="background" value="Transparent"/><param name="windowless" value="true"/><param name="initParams" value="id='+l.id+",filter="+h+'"/></object>';function e(){return document.getElementById(l.id+"_silverlight").content.Upload}l.bind("Silverlight:Init",function(){var i,n={};l.bind("Silverlight:StartSelectFiles",function(o){i=[]});l.bind("Silverlight:SelectFile",function(o,r,p,q){var s;s=c.guid();n[s]=r;n[r]=s;i.push(new c.File(s,p,q))});l.bind("Silverlight:SelectSuccessful",function(){if(i.length){l.trigger("FilesAdded",i)}});l.bind("Silverlight:UploadChunkError",function(o,r,p,s,q){l.trigger("Error",{code:c.IO_ERROR,message:"IO Error.",details:q,file:o.getFile(n[r])})});l.bind("Silverlight:UploadFileProgress",function(o,s,p,r){var q=o.getFile(n[s]);if(q.status!=c.FAILED){q.size=r;q.loaded=p;o.trigger("UploadProgress",q)}});l.bind("Refresh",function(o){var p,q,r;p=document.getElementById(o.settings.browse_button);q=c.getPos(p,document.getElementById(o.settings.container));r=c.getSize(p);c.extend(document.getElementById(o.id+"_silverlight_container").style,{top:q.y+"px",left:q.x+"px",width:r.w+"px",height:r.h+"px"})});l.bind("Silverlight:UploadChunkSuccessful",function(o,r,p,u,t){var s,q=o.getFile(n[r]);s={chunk:p,chunks:u,response:t};o.trigger("ChunkUploaded",q,s);if(q.status!=c.FAILED){e().UploadNextChunk()}if(p==u-1){q.status=c.DONE;o.trigger("FileUploaded",q,{response:t})}});l.bind("Silverlight:UploadSuccessful",function(o,r,p){var q=o.getFile(n[r]);q.status=c.DONE;o.trigger("FileUploaded",q,{response:p})});l.bind("FilesRemoved",function(o,q){var p;for(p=0;p<q.length;p++){e().RemoveFile(n[q[p].id])}});l.bind("UploadFile",function(o,q){var r=o.settings,p=r.resize||{};e().UploadFile(n[q.id],c.buildUrl(o.settings.url,{name:q.target_name||q.name}),b({chunk_size:r.chunk_size,image_width:p.width,image_height:p.height,image_quality:p.quality||90,multipart:!!r.multipart,multipart_params:r.multipart_params||{},headers:r.headers}))});m({success:true})})}})})(plupload);
|
@ -18,14 +18,63 @@ var enableFileOrTextToggling = function() {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var setupUploader = function() {
|
||||||
|
var multipartParams = {};
|
||||||
|
multipartParams[$('meta[name=csrf-param]').attr('content')] = $('meta[name=csrf-token]').attr('content');
|
||||||
|
|
||||||
|
var uploader = new plupload.Uploader({
|
||||||
|
runtimes : 'html5,flash',
|
||||||
|
container: 'theme-images',
|
||||||
|
browse_button : 'upload-link',
|
||||||
|
max_file_size : '5mb',
|
||||||
|
url : $('a#upload-link').attr('href'),
|
||||||
|
flash_swf_url : '/javascripts/admin/plugins/plupload/plupload.flash.swf',
|
||||||
|
multipart: true,
|
||||||
|
multipart_params: multipartParams
|
||||||
|
});
|
||||||
|
|
||||||
|
uploader.bind('QueueChanged', function() {
|
||||||
|
uploader.start();
|
||||||
|
});
|
||||||
|
|
||||||
|
uploader.init();
|
||||||
}
|
}
|
||||||
|
|
||||||
$(document).ready(function() {
|
$(document).ready(function() {
|
||||||
enableFileOrTextToggling();
|
enableFileOrTextToggling();
|
||||||
|
|
||||||
$('code.stylesheet textarea').each(function() { addCodeMirrorEditor('css', $(this)); });
|
// $('code.stylesheet textarea').each(function() {
|
||||||
$('code.javascript textarea').each(function() {
|
// addCodeMirrorEditor(null, $(this), ["tokenizejavascript.js", "parsejavascript.js", "parsecss.js"]);
|
||||||
addCodeMirrorEditor('javascript', $(this), ["tokenizejavascript.js", "parsejavascript.js"]);
|
// });
|
||||||
|
// $('code.javascript textarea').each(function() {
|
||||||
|
// addCodeMirrorEditor(null, $(this), ["parsecss.js", "tokenizejavascript.js", "parsejavascript.js"]);
|
||||||
|
// });
|
||||||
|
|
||||||
|
$('select#theme_asset_content_type').bind('change', function() {
|
||||||
|
var editor = CodeMirrorEditors[0].editor;
|
||||||
|
editor.setParser($(this).val() == 'stylesheet' ? 'CSSParser' : 'JSParser');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
$('a#asset-picker-link').fancybox({
|
||||||
|
'onComplete': function() {
|
||||||
|
setupUploader();
|
||||||
|
|
||||||
|
$('ul.assets h4 a').bind('click', function(e) {
|
||||||
|
var editor = CodeMirrorEditors[0].editor;
|
||||||
|
var handle = editor.cursorLine(), position = editor.cursorPosition(handle).character;
|
||||||
|
var text = 'url("' + $(this).attr('href') + '")';
|
||||||
|
|
||||||
|
editor.insertIntoLine(handle, position, text);
|
||||||
|
|
||||||
|
e.stopPropagation();
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
$.fancybox.close();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
$('a#asset-picker-link').click();
|
||||||
});
|
});
|
@ -121,10 +121,10 @@ ul.assets li.asset h4 { margin: 0px; height: 30px; }
|
|||||||
|
|
||||||
ul.assets li.asset h4 a {
|
ul.assets li.asset h4 a {
|
||||||
position: relative;
|
position: relative;
|
||||||
top: 6px;
|
top: 7px;
|
||||||
left: 12px;
|
left: 12px;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
font-size: 0.7em;
|
font-size: 0.6em;
|
||||||
color: #1f82bc;
|
color: #1f82bc;
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
}
|
}
|
||||||
|
13
public/stylesheets/admin/box.css
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
/* custom styles for fancybox */
|
||||||
|
|
||||||
|
div.asset-picker { width: 470px; position: relative; }
|
||||||
|
div.asset-picker .actions { position: absolute; right: 4px; top: 0px; }
|
||||||
|
div.asset-picker h2 {
|
||||||
|
border-bottom:1px dotted #BBBBBD;
|
||||||
|
color:#1E1F26;
|
||||||
|
font-size:1.1em;
|
||||||
|
font-weight:bold;
|
||||||
|
padding-bottom:10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
div.asset-picker ul { overflow: auto; height: 471px; }
|
@ -54,6 +54,11 @@
|
|||||||
color: #8B8D9A;
|
color: #8B8D9A;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.button.small.add {
|
||||||
|
padding-left: 24px;
|
||||||
|
background-position: 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
.button.remove, .button.remove span {
|
.button.remove, .button.remove span {
|
||||||
color: #ff092c !important;
|
color: #ff092c !important;
|
||||||
font-size: 1.1em;
|
font-size: 1.1em;
|
||||||
|
@ -137,6 +137,10 @@ form.formtastic fieldset ol li p.inline-errors {
|
|||||||
font-size: 0.7em !important;
|
font-size: 0.7em !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
form.formtastic fieldset ol li.more { text-align: right; width: auto; margin-right: 20px; line-height: 0.6em; }
|
||||||
|
form.formtastic fieldset ol li.more a { text-decoration: none; color: #787A89; font-size: 0.7em; }
|
||||||
|
form.formtastic fieldset ol li.more a:hover { text-decoration: underline; }
|
||||||
|
|
||||||
|
|
||||||
/*form.formtastic hr { border-top: 2px solid #ccc; }*/
|
/*form.formtastic hr { border-top: 2px solid #ccc; }*/
|
||||||
|
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
|
html {
|
||||||
|
cursor: text;
|
||||||
|
}
|
||||||
|
|
||||||
.editbox {
|
.editbox {
|
||||||
margin: .4em;
|
margin: .4em;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
font-size: 10pt;
|
font-size: 10pt;
|
||||||
color: black;
|
color: black;
|
||||||
background: white url(/images/admin/form/field.png) repeat-x 0 0 !important;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pre.code, .editbox {
|
pre.code, .editbox {
|
||||||
|
@ -17,6 +17,10 @@ h2 {
|
|||||||
font-size: 14pt;
|
font-size: 14pt;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
h3 {
|
||||||
|
font-size: 12pt;
|
||||||
|
}
|
||||||
|
|
||||||
p.rel {
|
p.rel {
|
||||||
padding-left: 2em;
|
padding-left: 2em;
|
||||||
text-indent: -2em;
|
text-indent: -2em;
|
||||||
|
59
public/stylesheets/admin/plugins/codemirror/jscolors.css
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
html {
|
||||||
|
cursor: text;
|
||||||
|
}
|
||||||
|
|
||||||
|
.editbox {
|
||||||
|
margin: .4em;
|
||||||
|
padding: 0;
|
||||||
|
font-family: monospace;
|
||||||
|
font-size: 10pt;
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
pre.code, .editbox {
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
.editbox p {
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-punctuation {
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-operator {
|
||||||
|
color: #666666;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-keyword {
|
||||||
|
color: #770088;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-atom {
|
||||||
|
color: #228811;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-variable {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-variabledef {
|
||||||
|
color: #0000FF;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-localvariable {
|
||||||
|
color: #004499;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-property {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-comment {
|
||||||
|
color: #AA7700;
|
||||||
|
}
|
||||||
|
|
||||||
|
span.js-string {
|
||||||
|
color: #AA2222;
|
||||||
|
}
|
@ -1,3 +1,7 @@
|
|||||||
|
html {
|
||||||
|
cursor: text;
|
||||||
|
}
|
||||||
|
|
||||||
.editbox {
|
.editbox {
|
||||||
margin: .4em;
|
margin: .4em;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
|
@ -1,10 +1,13 @@
|
|||||||
|
html {
|
||||||
|
cursor: text;
|
||||||
|
}
|
||||||
|
|
||||||
.editbox {
|
.editbox {
|
||||||
margin: .4em;
|
margin: .4em;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
font-size: 10pt;
|
font-size: 10pt;
|
||||||
color: black;
|
color: black;
|
||||||
background: white url(/images/admin/form/field.png) repeat-x 0 0 !important;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.editbox p {
|
.editbox p {
|
||||||
@ -44,7 +47,7 @@ span.xml-entity {
|
|||||||
}
|
}
|
||||||
|
|
||||||
span.xml-error {
|
span.xml-error {
|
||||||
color: #F00;
|
color: #F00 !important;
|
||||||
}
|
}
|
||||||
|
|
||||||
span.xml-text {
|
span.xml-text {
|
||||||
|
363
public/stylesheets/admin/plugins/fancybox.css
Executable file
@ -0,0 +1,363 @@
|
|||||||
|
/*
|
||||||
|
* FancyBox - jQuery Plugin
|
||||||
|
* Simple and fancy lightbox alternative
|
||||||
|
*
|
||||||
|
* Examples and documentation at: http://fancybox.net
|
||||||
|
*
|
||||||
|
* Copyright (c) 2008 - 2010 Janis Skarnelis
|
||||||
|
*
|
||||||
|
* Version: 1.3.1 (05/03/2010)
|
||||||
|
* Requires: jQuery v1.3+
|
||||||
|
*
|
||||||
|
* Dual licensed under the MIT and GPL licenses:
|
||||||
|
* http://www.opensource.org/licenses/mit-license.php
|
||||||
|
* http://www.gnu.org/licenses/gpl.html
|
||||||
|
*/
|
||||||
|
|
||||||
|
#fancybox-loading {
|
||||||
|
position: fixed;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
height: 40px;
|
||||||
|
width: 40px;
|
||||||
|
margin-top: -20px;
|
||||||
|
margin-left: -20px;
|
||||||
|
cursor: pointer;
|
||||||
|
overflow: hidden;
|
||||||
|
z-index: 1104;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
* html #fancybox-loading { /* IE6 */
|
||||||
|
position: absolute;
|
||||||
|
margin-top: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-loading div {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 40px;
|
||||||
|
height: 480px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-overlay {
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
bottom: 0;
|
||||||
|
right: 0;
|
||||||
|
background: #000;
|
||||||
|
z-index: 1100;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
* html #fancybox-overlay { /* IE6 */
|
||||||
|
position: absolute;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-tmp {
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
border: 0;
|
||||||
|
overflow: auto;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-wrap {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
margin: 0;
|
||||||
|
padding: 20px;
|
||||||
|
z-index: 1101;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-outer {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: #FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-inner {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 1px;
|
||||||
|
height: 1px;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
outline: none;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-hide-sel-frame {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
background: transparent;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-close {
|
||||||
|
position: absolute;
|
||||||
|
top: -15px;
|
||||||
|
right: -15px;
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px 0px;
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 1103;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox_error {
|
||||||
|
color: #444;
|
||||||
|
font: normal 12px/20px Arial;
|
||||||
|
padding: 7px;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-content {
|
||||||
|
height: auto;
|
||||||
|
width: auto;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
border: none;
|
||||||
|
outline: none;
|
||||||
|
line-height: 0;
|
||||||
|
vertical-align: top;
|
||||||
|
-ms-interpolation-mode: bicubic;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-frame {
|
||||||
|
position: relative;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: none;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 0;
|
||||||
|
font-family: Arial;
|
||||||
|
font-size: 12px;
|
||||||
|
z-index: 1102;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fancybox-title-inside {
|
||||||
|
padding: 10px 0;
|
||||||
|
text-align: center;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fancybox-title-outside {
|
||||||
|
padding-top: 5px;
|
||||||
|
color: #FFF;
|
||||||
|
text-align: center;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fancybox-title-over {
|
||||||
|
color: #FFF;
|
||||||
|
text-align: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-over {
|
||||||
|
padding: 10px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancy_title_over.png');
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-wrap {
|
||||||
|
display: inline-block;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-wrap span {
|
||||||
|
height: 32px;
|
||||||
|
float: left;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-left {
|
||||||
|
padding-left: 15px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -90px;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-main {
|
||||||
|
font-weight: bold;
|
||||||
|
line-height: 29px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox-x.png');
|
||||||
|
background-position: 0px -40px;
|
||||||
|
color: #FFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-title-right {
|
||||||
|
padding-left: 15px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -55px -90px;
|
||||||
|
background-repeat: no-repeat;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left, #fancybox-right {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0px;
|
||||||
|
height: 100%;
|
||||||
|
width: 35%;
|
||||||
|
cursor: pointer;
|
||||||
|
outline: none;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/blank.gif');
|
||||||
|
z-index: 1102;
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left {
|
||||||
|
left: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-right {
|
||||||
|
right: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left-ico, #fancybox-right-ico {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: -9999px;
|
||||||
|
width: 30px;
|
||||||
|
height: 30px;
|
||||||
|
margin-top: -15px;
|
||||||
|
cursor: pointer;
|
||||||
|
z-index: 1102;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left-ico {
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-right-ico {
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -60px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left:hover, #fancybox-right:hover {
|
||||||
|
visibility: visible; /* IE6 */
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-left:hover span {
|
||||||
|
left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancybox-right:hover span {
|
||||||
|
left: auto;
|
||||||
|
right: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.fancy-bg {
|
||||||
|
position: absolute;
|
||||||
|
padding: 0;
|
||||||
|
margin: 0;
|
||||||
|
border: 0;
|
||||||
|
width: 20px;
|
||||||
|
height: 20px;
|
||||||
|
z-index: 1001;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-n {
|
||||||
|
top: -20px;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox-x.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-ne {
|
||||||
|
top: -20px;
|
||||||
|
right: -20px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -162px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-e {
|
||||||
|
top: 0;
|
||||||
|
right: -20px;
|
||||||
|
height: 100%;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox-y.png');
|
||||||
|
background-position: -20px 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-se {
|
||||||
|
bottom: -20px;
|
||||||
|
right: -20px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -182px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-s {
|
||||||
|
bottom: -20px;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox-x.png');
|
||||||
|
background-position: 0px -20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-sw {
|
||||||
|
bottom: -20px;
|
||||||
|
left: -20px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -142px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-w {
|
||||||
|
top: 0;
|
||||||
|
left: -20px;
|
||||||
|
height: 100%;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox-y.png');
|
||||||
|
}
|
||||||
|
|
||||||
|
#fancy-bg-nw {
|
||||||
|
top: -20px;
|
||||||
|
left: -20px;
|
||||||
|
background-image: url('/images/admin/plugins/fancybox/fancybox.png');
|
||||||
|
background-position: -40px -122px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* IE */
|
||||||
|
|
||||||
|
#fancybox-loading.fancybox-ie div { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_loading.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancybox-close { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_close.png', sizingMethod='scale'); }
|
||||||
|
|
||||||
|
.fancybox-ie #fancybox-title-over { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_title_over.png', sizingMethod='scale'); zoom: 1; }
|
||||||
|
.fancybox-ie #fancybox-title-left { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_title_left.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancybox-title-main { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_title_main.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancybox-title-right { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_title_right.png', sizingMethod='scale'); }
|
||||||
|
|
||||||
|
.fancybox-ie #fancybox-left-ico { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_nav_left.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancybox-right-ico { background: transparent; filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_nav_right.png', sizingMethod='scale'); }
|
||||||
|
|
||||||
|
.fancybox-ie .fancy-bg { background: transparent !important; }
|
||||||
|
|
||||||
|
.fancybox-ie #fancy-bg-n { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_n.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-ne { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_ne.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-e { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_e.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-se { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_se.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-s { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_s.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-sw { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_sw.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-w { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_w.png', sizingMethod='scale'); }
|
||||||
|
.fancybox-ie #fancy-bg-nw { filter: progid:DXImageTransform.Microsoft.AlphaImageLoader(src='fancybox/fancy_shadow_nw.png', sizingMethod='scale'); }
|