Commit 45deacbe4770e239447f7f2ed8688c432d72a425

Authored by Victor Costa
2 parents f46898b5 32dfc0b8

Merge branch 'AI3205-comment-paragraph' into stable

Conflicts:
	app/models/comment.rb
Showing 30 changed files with 1204 additions and 1 deletions   Show diff stats
db/migrate/20140505190749_add_paragraph_to_comments.rb 0 → 100644
... ... @@ -0,0 +1,9 @@
  1 +class AddParagraphToComments < ActiveRecord::Migration
  2 + def self.up
  3 + add_column :comments, :paragraph_id, :integer
  4 + end
  5 +
  6 + def self.down
  7 + remove_column :comments, :paragraph_id
  8 + end
  9 +end
... ...
db/schema.rb
... ... @@ -11,7 +11,7 @@
11 11 #
12 12 # It's strongly recommended to check this file into your version control system.
13 13  
14   -ActiveRecord::Schema.define(:version => 20140827191326) do
  14 +ActiveRecord::Schema.define(:version => 20140829153047) do
15 15  
16 16 create_table "abuse_reports", :force => true do |t|
17 17 t.integer "reporter_id"
... ... @@ -414,6 +414,45 @@ ActiveRecord::Schema.define(:version =&gt; 20140827191326) do
414 414 add_index "national_regions", ["name"], :name => "name_index"
415 415 add_index "national_regions", ["national_region_code"], :name => "code_index"
416 416  
  417 + create_table "oauth_access_grants", :force => true do |t|
  418 + t.integer "resource_owner_id", :null => false
  419 + t.integer "application_id", :null => false
  420 + t.string "token", :null => false
  421 + t.integer "expires_in", :null => false
  422 + t.text "redirect_uri", :null => false
  423 + t.datetime "created_at", :null => false
  424 + t.datetime "revoked_at"
  425 + t.string "scopes"
  426 + end
  427 +
  428 + add_index "oauth_access_grants", ["token"], :name => "index_oauth_access_grants_on_token", :unique => true
  429 +
  430 + create_table "oauth_access_tokens", :force => true do |t|
  431 + t.integer "resource_owner_id"
  432 + t.integer "application_id"
  433 + t.string "token", :null => false
  434 + t.string "refresh_token"
  435 + t.integer "expires_in"
  436 + t.datetime "revoked_at"
  437 + t.datetime "created_at", :null => false
  438 + t.string "scopes"
  439 + end
  440 +
  441 + add_index "oauth_access_tokens", ["refresh_token"], :name => "index_oauth_access_tokens_on_refresh_token", :unique => true
  442 + add_index "oauth_access_tokens", ["resource_owner_id"], :name => "index_oauth_access_tokens_on_resource_owner_id"
  443 + add_index "oauth_access_tokens", ["token"], :name => "index_oauth_access_tokens_on_token", :unique => true
  444 +
  445 + create_table "oauth_applications", :force => true do |t|
  446 + t.string "name", :null => false
  447 + t.string "uid", :null => false
  448 + t.string "secret", :null => false
  449 + t.text "redirect_uri", :null => false
  450 + t.datetime "created_at", :null => false
  451 + t.datetime "updated_at", :null => false
  452 + end
  453 +
  454 + add_index "oauth_applications", ["uid"], :name => "index_oauth_applications_on_uid", :unique => true
  455 +
417 456 create_table "price_details", :force => true do |t|
418 457 t.decimal "price", :default => 0.0
419 458 t.integer "product_id"
... ...
plugins/comment_paragraph/README.md 0 → 100644
... ... @@ -0,0 +1,4 @@
  1 +README - Comment Paragraph Plugin
  2 +======================================
  3 +
  4 +This plugins enables all the paragraphs of an article to have comments
... ...
plugins/comment_paragraph/controllers/profile/comment_paragraph_plugin_profile_controller.rb 0 → 100644
... ... @@ -0,0 +1,16 @@
  1 +class CommentParagraphPluginProfileController < ProfileController
  2 + append_view_path File.join(File.dirname(__FILE__) + '/../../views')
  3 +
  4 + def view_comments
  5 + @article_id = params[:article_id]
  6 + @paragraph_id = params[:paragraph_id]
  7 +
  8 + article = profile.articles.find(@article_id)
  9 + @paragraph_comment_page = (params[:paragraph_comment_page] || 1).to_i
  10 +
  11 + @comments = article.comments.without_spam.in_paragraph(@paragraph_id)
  12 + @comments_count = @comments.count
  13 + @comments = @comments.without_reply
  14 + end
  15 +
  16 +end
0 17 \ No newline at end of file
... ...
plugins/comment_paragraph/controllers/public/comment_paragraph_plugin_public_controller.rb 0 → 100644
... ... @@ -0,0 +1,9 @@
  1 +class CommentParagraphPluginPublicController < PublicController
  2 + append_view_path File.join(File.dirname(__FILE__) + '/../views')
  3 +
  4 + def comment_paragraph
  5 + @comment = Comment.find(params[:id])
  6 + render :json => { :paragraph_id => @comment.paragraph_id }
  7 + end
  8 +
  9 +end
... ...
plugins/comment_paragraph/db/migrate/20140715201649_add_paragraph_id_to_comment.rb 0 → 100644
... ... @@ -0,0 +1,9 @@
  1 +class AddParagraphIdToComment < ActiveRecord::Migration
  2 + def self.up
  3 + add_column :comments, :paragraph_id, :integer unless column_exists?(:comments, :paragraph_id)
  4 + end
  5 +
  6 + def self.down
  7 + remove_column :comments, :paragraph_id
  8 + end
  9 +end
... ...
plugins/comment_paragraph/lib/comment_paragraph_plugin.rb 0 → 100644
... ... @@ -0,0 +1,79 @@
  1 +class CommentParagraphPlugin < Noosfero::Plugin
  2 +
  3 + def self.plugin_name
  4 + "Comment Paragraph"
  5 + end
  6 +
  7 + def self.plugin_description
  8 + _("A plugin that display comments divided by paragraphs.")
  9 + end
  10 +
  11 + def unavailable_comments(scope)
  12 + scope.without_paragraph
  13 + end
  14 +
  15 + def comment_form_extra_contents(args)
  16 + comment = args[:comment]
  17 + paragraph_id = comment.paragraph_id || args[:paragraph_id]
  18 + proc {
  19 + arr = []
  20 + arr << hidden_field_tag('comment[id]', comment.id)
  21 + arr << hidden_field_tag('comment[paragraph_id]', paragraph_id) if paragraph_id
  22 + arr << hidden_field_tag('comment[comment_paragraph_selected_area]', comment.comment_paragraph_selected_area) if comment.comment_paragraph_selected_area
  23 + arr
  24 + }
  25 + end
  26 +
  27 + def comment_extra_contents(args)
  28 + comment = args[:comment]
  29 + proc {
  30 + render :file => 'comment/comment_extra', :locals => {:comment => comment}
  31 + }
  32 + end
  33 +
  34 + def js_files
  35 + ['comment_paragraph_macro', 'rangy-core', 'rangy-cssclassapplier', 'rangy-serializer']
  36 + end
  37 +
  38 + def stylesheet?
  39 + true
  40 + end
  41 +
  42 + def cms_controller_filters
  43 + block = proc do
  44 + if params['commit'] == 'Save'
  45 + unless @article.id.blank?
  46 +
  47 + parsed_paragraphs = []
  48 + paragraph_id = 0
  49 +
  50 + doc = Hpricot(@article.body)
  51 + paragraphs = doc.search("/[\r\n]").each do |paragraph|
  52 + parsed_paragraphs << (paragraph.to_html =~ /(.*)paragraph_comment_spacer(.*)|<div(.*)paragraph_comment(.*)/ ? paragraph.to_html : CommentParagraphPlugin.parse_paragraph(paragraph.to_html, paragraph_id))
  53 + paragraph_id += 1
  54 + end
  55 +
  56 + @article.body = parsed_paragraphs.join()
  57 + @article.save
  58 +
  59 + end
  60 + end
  61 + end
  62 +
  63 + { :type => 'after_filter',
  64 + :method_name => 'new',
  65 + :block => block }
  66 + end
  67 +
  68 + private
  69 +
  70 + def self.parse_paragraph( paragraph_content, paragraph_id )
  71 + "<div class='macro article_comments paragraph_comment' " +
  72 + "data-macro='comment_paragraph_plugin/allow_comment' " +
  73 + "data-macro-paragraph_id='#{paragraph_id}'>#{paragraph_content}</div>\r\n" +
  74 + "<p class='paragraph_comment_spacer'></p>\r\n"
  75 + end
  76 +
  77 +end
  78 +
  79 +require_dependency 'comment_paragraph_plugin/macros/allow_comment'
... ...
plugins/comment_paragraph/lib/comment_paragraph_plugin/macros/allow_comment.rb 0 → 100644
... ... @@ -0,0 +1,24 @@
  1 +#FIXME See a better way to generalize this parameter.
  2 +ActionView::Base.sanitized_allowed_attributes += ['data-macro', 'data-macro-paragraph_id']
  3 +
  4 +class CommentParagraphPlugin::AllowComment < Noosfero::Plugin::Macro
  5 + def self.configuration
  6 + { :params => [],
  7 + :skip_dialog => true,
  8 + :generator => 'makeCommentable();',
  9 + :js_files => 'comment_paragraph.js',
  10 + :icon_path => '/plugins/comment_paragraph/images/balloons-comment.png',
  11 + :css_files => 'comment_paragraph.css' }
  12 + end
  13 +
  14 + def parse(params, inner_html, source)
  15 + paragraph_id = params[:paragraph_id].to_i
  16 + article = source
  17 + count = article.paragraph_comments.without_spam.in_paragraph(paragraph_id).count
  18 +
  19 + proc {
  20 + render :partial => 'comment_paragraph_plugin_profile/comment_paragraph',
  21 + :locals => {:paragraph_id => paragraph_id, :article_id => article.id, :inner_html => inner_html, :count => count, :profile_identifier => article.profile.identifier }
  22 + }
  23 + end
  24 +end
... ...
plugins/comment_paragraph/lib/ext/article.rb 0 → 100644
... ... @@ -0,0 +1,17 @@
  1 +require_dependency 'article'
  2 +
  3 +class Article
  4 +
  5 + has_many :paragraph_comments, :class_name => 'Comment', :foreign_key => 'source_id', :dependent => :destroy, :order => 'created_at asc', :conditions => [ 'paragraph_id IS NOT NULL']
  6 +
  7 + validate :body_change_with_comments
  8 +
  9 + def body_change_with_comments
  10 + if body && body_changed? && !self.comments.empty?
  11 + paragraphs_with_comments = self.comments.where("'paragraph_id' IS NOT NULL")
  12 + errors[:base] << (N_('You are unable to change the body of the article when paragraphs are commented')) unless (paragraphs_with_comments).empty?
  13 + end
  14 + end
  15 +
  16 +end
  17 +
... ...
plugins/comment_paragraph/lib/ext/comment.rb 0 → 100644
... ... @@ -0,0 +1,16 @@
  1 +require_dependency 'comment'
  2 +
  3 +class Comment
  4 +
  5 + scope :without_paragraph, :conditions => {:paragraph_id => nil }
  6 +
  7 + settings_items :comment_paragraph_selected_area, :type => :string
  8 +
  9 + scope :in_paragraph, proc { |paragraph_id| {
  10 + :conditions => ['paragraph_id = ?', paragraph_id]
  11 + }
  12 + }
  13 +
  14 + attr_accessible :paragraph_id, :comment_paragraph_selected_area, :id
  15 +
  16 +end
... ...
plugins/comment_paragraph/public/comment_paragraph.css 0 → 100644
... ... @@ -0,0 +1,6 @@
  1 +div.article_comments {
  2 + background-color:#dddddd;
  3 + border-style: solid;
  4 + border-color:#bbbbbb;
  5 + border-width:2px;
  6 +}
0 7 \ No newline at end of file
... ...
plugins/comment_paragraph/public/comment_paragraph.js 0 → 100644
... ... @@ -0,0 +1,57 @@
  1 +String.prototype.startsWith = function(needle){
  2 + return(this.indexOf(needle) == 0);
  3 +};
  4 +
  5 +function makeCommentable() {
  6 + var paragraphsTxt="";
  7 + var selectedTextCount=0;
  8 + var notSelectedTextCount=0;
  9 + var text;
  10 +
  11 + //Search for text that is not selected in the middle of selected text, in this case unselect everything
  12 + jQuery('#article_body_ifr').contents().find('body').children().each(function( index ) {
  13 + //Check if there are other texts not selected
  14 + var element=jQuery(this).prop('outerHTML');
  15 + if(element.startsWith('<div')){
  16 + selectedTextCount++;
  17 + }else{
  18 + if(! element.startsWith('<p><br></p>') && ! element.startsWith('<p>&nbsp;</p>') ){
  19 + notSelectedTextCount++;
  20 + }
  21 + }
  22 + });
  23 +
  24 + if(selectedTextCount > 0 && notSelectedTextCount>0){
  25 + jQuery('#article_body_ifr').contents().find('body').children('.paragraph_comment').contents().unwrap();
  26 + //Workaround necessary to post the body of the article
  27 + tinymce.activeEditor.execCommand('mceInsertContent', false, " ");
  28 + return;
  29 + }
  30 +
  31 + //Add p tag, when opening the editor. For some season it appear at the end without p tag
  32 + foundCommentableParagraph = false;
  33 + jQuery('#article_body_ifr').contents().find('body').children('div.article_comments').each(function( index ) {
  34 + if(jQuery(this).html()!="" && jQuery(this).html()!=" " && jQuery(this).html()!="<br>"){
  35 + paragraphsTxt+="<p>" + jQuery(this).html() + "</p>";
  36 + }
  37 + foundCommentableParagraph = true;
  38 + });
  39 +
  40 + //undo the paragraph comment tags
  41 + if(foundCommentableParagraph === true){
  42 + tinyMCE.activeEditor.setContent(paragraphsTxt);
  43 + return;
  44 + }
  45 +
  46 + //Wraps the paragraph using the chosen class
  47 + jQuery('#article_body_ifr').contents().find('body').children('p,table,img').each(function( index ) {
  48 + text=jQuery(this).prop('outerHTML');
  49 + if(text!="" && text!=" " && text!="<br>"){
  50 + paragraphsTxt+='<div class="macro article_comments paragraph_comment" data-macro="comment_paragraph_plugin/allow_comment" data-macro-paragraph_id="' + index + '">' + text + '</div><br>'
  51 + }
  52 + });
  53 + tinyMCE.activeEditor.setContent(paragraphsTxt);
  54 +
  55 + //Workaround necessary to post the body of the article
  56 + tinymce.activeEditor.execCommand('mceInsertContent', false, " ");
  57 +}
0 58 \ No newline at end of file
... ...
plugins/comment_paragraph/public/comment_paragraph_macro.js 0 → 100644
... ... @@ -0,0 +1,260 @@
  1 +var comment_paragraph_anchor;
  2 +var lastParagraph = [];
  3 +var lastSelectedArea = [];
  4 +
  5 +function setPlusIfZeroComments($){
  6 + $('.comment-count').each(function(){
  7 + var commentCount = $(this).text().trim();
  8 + if(commentCount=='0')
  9 + $(this).text("+");
  10 + });
  11 +}
  12 +
  13 +jQuery(document).ready(function($) {
  14 + //Quit if does not detect a comment for that plugin
  15 + if($('.comment_paragraph').size() < 1)
  16 + return;
  17 +
  18 + $(document).keyup(function(e) {
  19 + // on press ESC key...
  20 + if (e.which == 27) {
  21 + // closing side comment box
  22 + $("div.side-comment").hide();
  23 + }
  24 + });
  25 +
  26 + setPlusIfZeroComments($);
  27 + $('.display-comment-form').unbind();
  28 + $('.display-comment-form').click(function(){
  29 + var $button = $(this);
  30 + showBox($button.parents('.post_comment_box'));
  31 + $($button).hide();
  32 + $button.closest('.page-comment-form').find('input').first().focus();
  33 + return false;
  34 + });
  35 +
  36 + $('#cancel-comment').die();
  37 + $(document.body).on("click", '#cancel-comment', function(){
  38 + $("div.side-comment").hide();
  39 + });
  40 +
  41 + function showBox(div){
  42 + if(div.hasClass('closed')) {
  43 + div.removeClass('closed');
  44 + div.addClass('opened');
  45 + }
  46 + }
  47 +
  48 + //Hides old style ballons
  49 + $("img[alt|=Comments]").hide();
  50 + rangy.init();
  51 + cssApplier = rangy.createCssClassApplier("commented-area", {normalize: false});
  52 + //Add marked text bubble
  53 + $("body").append('\
  54 + <a id="comment-bubble" style="width:90px;text-decoration: none">\
  55 + <div align="center" class="triangle-right" >Comentar</div>\
  56 + </a>');
  57 +
  58 +
  59 + $('.side-comments-counter').click(function(){
  60 + var paragraphId = $(this).data('paragraph');
  61 + hideAllCommentsExcept(paragraphId);
  62 + hideAllSelectedAreasExcept(paragraphId);
  63 + $('#comment-bubble').hide();
  64 + $('#side_comment_' + paragraphId).toggle();
  65 + $('#side_comment_' + paragraphId).find().toggle();
  66 + //Loads the comments
  67 + var url = $('#link_to_ajax_comments_' + paragraphId).data('url');
  68 + $.ajax({
  69 + dataType: "script",
  70 + url: url
  71 + }).done(function() {
  72 + var button = $('#page-comment-form-' + paragraphId + ' a').get(0);
  73 + button.click();
  74 + });
  75 + });
  76 +
  77 +
  78 + $('#comment-bubble').click(function(event){
  79 + $(this).hide();
  80 + $("#comment-bubble").css({top: 0, left: 0, position:'absolute'});
  81 + var url = $("#comment-bubble").data('url');
  82 + var paragraphId = $("#comment-bubble").data("paragraphId");
  83 + hideAllCommentsExcept(paragraphId);
  84 + $('#side_comment_' + paragraphId).show();
  85 + $.ajax({
  86 + dataType: "script",
  87 + url: url
  88 + }).done(function() {
  89 + var button = $('#page-comment-form-' + paragraphId + ' a').get(0);
  90 + button.click();
  91 + });
  92 + });
  93 +
  94 + function hideAllCommentsExcept(clickedParagraph){
  95 + $(".side-comment").each(function(){
  96 + paragraph = $(this).data('paragraph');
  97 + if(paragraph != clickedParagraph){
  98 + $(this).hide();
  99 + $(this).find().hide();
  100 + }
  101 + });
  102 + }
  103 +
  104 + function hideAllSelectedAreasExcept(clickedParagraph){
  105 + $(".comment_paragraph").each(function(){
  106 + paragraph = $(this).data('paragraph');
  107 + if(paragraph != clickedParagraph){
  108 + $(this).find(".commented-area").contents().unwrap();
  109 + }
  110 + });
  111 + }
  112 +
  113 + $("#comment-bubble").hide();
  114 + //Undo previous highlight from the paragraph
  115 + $('.comment_paragraph').mousedown(function(){
  116 + $("#comment-bubble").hide();
  117 + var paragraphId = $(this).data('paragraph');
  118 + $(this).find('.commented-area').replaceWith(function() {
  119 + return $(this).html();
  120 + });
  121 + var rootElement = $(this).get(0);
  122 + if(lastParagraph[paragraphId]){
  123 + rootElement.innerHTML = lastParagraph[paragraphId];
  124 + }
  125 + });
  126 +
  127 + function getSelectionText() {
  128 + var text = "";
  129 + if (window.getSelection) {
  130 + text = window.getSelection().toString();
  131 + } else if (document.selection && document.selection.type != "Control") {
  132 + text = document.selection.createRange().text;
  133 + }
  134 + return text;
  135 + }
  136 +
  137 + //highlight area from the paragraph
  138 + $('.comment_paragraph').mouseup(function(event){
  139 + deselectAll();
  140 + //Don't do anything if there is no selected text
  141 + if(getSelectionText().length == 0)
  142 + return;
  143 + var paragraphId = $(this).data('paragraph');
  144 + var currentMousePos = { x: -1, y: -1 };
  145 + currentMousePos.x = event.pageX;
  146 + currentMousePos.y = event.pageY;
  147 + $("#comment-bubble").css({top: event.pageY-70, left: event.pageX-70, position:'absolute'});
  148 + //Relates a bubble to the mouse up paragraph
  149 + $("#comment-bubble").data("paragraphId", paragraphId)
  150 + //Prepare to open the div
  151 + var url = $('#link_to_ajax_comments_' + paragraphId).data('url');
  152 + $("#comment-bubble").data("url", url)
  153 + $("#comment-bubble").show();
  154 + var rootElement = $(this).get(0);
  155 + //Stores the HTML content of the lastParagraph
  156 + lastParagraph[paragraphId] = rootElement.innerHTML;
  157 + //Maybe it is needed to handle exceptions here
  158 + try{
  159 + var selObj = rangy.getSelection();
  160 + var selected_area = rangy.serializeSelection(selObj, true,rootElement);
  161 + cssApplier.toggleSelection();
  162 + }catch(e){
  163 + //Translate this mesage
  164 + deselectAll();
  165 + rangy.init();
  166 + cssApplier = rangy.createCssClassApplier("commented-area", {normalize: false});
  167 + return;
  168 + }
  169 + //Register the area the has been selected at input.selected_area
  170 + lastSelectedArea[paragraphId] = selected_area;
  171 + form = $('#page-comment-form-' + paragraphId).find('form');
  172 + if (form.find('input.selected_area').length === 0){
  173 + $('<input>').attr({
  174 + class: 'selected_area',
  175 + type: 'hidden',
  176 + name: 'comment[comment_paragraph_selected_area]',
  177 + value: selected_area
  178 + }).appendTo(form)
  179 + }else{
  180 + form.find('input.selected_area').val(selected_area)
  181 + }
  182 + rootElement.focus();
  183 + });
  184 +
  185 + function deselectAll(){
  186 + $(".commented-area").contents().unwrap();
  187 + }
  188 +
  189 + function processAnchor(){
  190 + var anchor = window.location.hash;
  191 + if(anchor.length==0) return;
  192 + var val = anchor.split('-'); //anchor format = #comment-\d+
  193 + if(val.length!=2 || val[0]!='#comment') return;
  194 + if($('div[data-macro=comment_paragraph_plugin\\/allow_comment]').length==0) return; //comment_paragraph_plugin/allow_comment div must exists
  195 + var comment_id = val[1];
  196 + if(!/^\d+$/.test(comment_id)) return; //test for integer
  197 + comment_paragraph_anchor = anchor;
  198 + var url = '/plugin/comment_paragraph/public/comment_paragraph/'+comment_id;
  199 + $.ajax({
  200 + dataType: "script",
  201 + url: url
  202 + }).done(function() {
  203 + var button = $('#page-comment-form-' + comment_id + ' a').get(0)
  204 + button.click();
  205 + });
  206 + }
  207 +
  208 + processAnchor();
  209 +
  210 + $(document).on('mouseover', 'li.article-comment', function(){
  211 + var selected_area = $(this).find('input.paragraph_comment_area').val();
  212 + var paragraph_id = $(this).find('input.paragraph_id').val();
  213 + var rootElement = $('#comment_paragraph_'+ paragraph_id).get(0);
  214 +
  215 + if(lastParagraph[paragraph_id] == null || lastParagraph[paragraph_id] == 'undefined'){
  216 + lastParagraph[paragraph_id] = rootElement.innerHTML;
  217 + }
  218 + else {
  219 + rootElement.innerHTML = lastParagraph[paragraph_id] ;
  220 + }
  221 + if(selected_area != ""){
  222 + rangy.deserializeSelection(selected_area, rootElement);
  223 + cssApplier.toggleSelection();
  224 + }
  225 + });
  226 +
  227 + $(document).on('mouseout', 'li.article-comment', function(){
  228 + deselectAll();
  229 + var paragraph_id = $(this).find('input.paragraph_id').val();
  230 + var rootElement = $('#comment_paragraph_'+ paragraph_id).get(0);
  231 +
  232 + if(lastSelectedArea[paragraph_id] != null && lastSelectedArea[paragraph_id] != 'undefined' ){
  233 + rootElement = $('#comment_paragraph_'+ paragraph_id).get(0);
  234 + rootElement.innerHTML = lastParagraph[paragraph_id];
  235 + rangy.deserializeSelection(lastSelectedArea[paragraph_id], rootElement);
  236 + cssApplier.toggleSelection();
  237 + } else {
  238 + cssApplier.toggleSelection();
  239 + var sel = rangy.getSelection();
  240 + sel.removeAllRanges();
  241 + }
  242 + });
  243 +
  244 + function toggleParagraph(paragraph) {
  245 + var div = $('div.comments_list_toggle_paragraph_'+paragraph);
  246 + var visible = div.is(':visible');
  247 + if(!visible)
  248 + $('div.comment-paragraph-loading-' + paragraph).addClass('comment-button-loading');
  249 + div.toggle('fast');
  250 + return visible;
  251 + }
  252 +
  253 + function loadCompleted(paragraph) {
  254 + $('div.comment-paragraph-loading-'+paragraph).removeClass('comment-button-loading')
  255 + if(comment_paragraph_anchor) {
  256 + $.scrollTo($(comment_paragraph_anchor));
  257 + comment_paragraph_anchor = null;
  258 + }
  259 + }
  260 +});
... ...
plugins/comment_paragraph/public/images/balloons-comment.png 0 → 100644

671 Bytes

plugins/comment_paragraph/public/images/comments.gif 0 → 100644

1.66 KB

plugins/comment_paragraph/public/rangy-core.js 0 → 100644
... ... @@ -0,0 +1,94 @@
  1 +/*
  2 + Rangy, a cross-browser JavaScript range and selection library
  3 + http://code.google.com/p/rangy/
  4 +
  5 + Copyright 2012, Tim Down
  6 + Licensed under the MIT license.
  7 + Version: 1.2.3
  8 + Build date: 26 February 2012
  9 +*/
  10 +window.rangy=function(){function l(p,u){var w=typeof p[u];return w=="function"||!!(w=="object"&&p[u])||w=="unknown"}function K(p,u){return!!(typeof p[u]=="object"&&p[u])}function H(p,u){return typeof p[u]!="undefined"}function I(p){return function(u,w){for(var B=w.length;B--;)if(!p(u,w[B]))return false;return true}}function z(p){return p&&A(p,x)&&v(p,t)}function C(p){window.alert("Rangy not supported in your browser. Reason: "+p);c.initialized=true;c.supported=false}function N(){if(!c.initialized){var p,
  11 +u=false,w=false;if(l(document,"createRange")){p=document.createRange();if(A(p,n)&&v(p,i))u=true;p.detach()}if((p=K(document,"body")?document.body:document.getElementsByTagName("body")[0])&&l(p,"createTextRange")){p=p.createTextRange();if(z(p))w=true}!u&&!w&&C("Neither Range nor TextRange are implemented");c.initialized=true;c.features={implementsDomRange:u,implementsTextRange:w};u=k.concat(f);w=0;for(p=u.length;w<p;++w)try{u[w](c)}catch(B){K(window,"console")&&l(window.console,"log")&&window.console.log("Init listener threw an exception. Continuing.",
  12 +B)}}}function O(p){this.name=p;this.supported=this.initialized=false}var i=["startContainer","startOffset","endContainer","endOffset","collapsed","commonAncestorContainer","START_TO_START","START_TO_END","END_TO_START","END_TO_END"],n=["setStart","setStartBefore","setStartAfter","setEnd","setEndBefore","setEndAfter","collapse","selectNode","selectNodeContents","compareBoundaryPoints","deleteContents","extractContents","cloneContents","insertNode","surroundContents","cloneRange","toString","detach"],
  13 +t=["boundingHeight","boundingLeft","boundingTop","boundingWidth","htmlText","text"],x=["collapse","compareEndPoints","duplicate","getBookmark","moveToBookmark","moveToElementText","parentElement","pasteHTML","select","setEndPoint","getBoundingClientRect"],A=I(l),q=I(K),v=I(H),c={version:"1.2.3",initialized:false,supported:true,util:{isHostMethod:l,isHostObject:K,isHostProperty:H,areHostMethods:A,areHostObjects:q,areHostProperties:v,isTextRange:z},features:{},modules:{},config:{alertOnWarn:false,preferTextRange:false}};
  14 +c.fail=C;c.warn=function(p){p="Rangy warning: "+p;if(c.config.alertOnWarn)window.alert(p);else typeof window.console!="undefined"&&typeof window.console.log!="undefined"&&window.console.log(p)};if({}.hasOwnProperty)c.util.extend=function(p,u){for(var w in u)if(u.hasOwnProperty(w))p[w]=u[w]};else C("hasOwnProperty not supported");var f=[],k=[];c.init=N;c.addInitListener=function(p){c.initialized?p(c):f.push(p)};var r=[];c.addCreateMissingNativeApiListener=function(p){r.push(p)};c.createMissingNativeApi=
  15 +function(p){p=p||window;N();for(var u=0,w=r.length;u<w;++u)r[u](p)};O.prototype.fail=function(p){this.initialized=true;this.supported=false;throw Error("Module '"+this.name+"' failed to load: "+p);};O.prototype.warn=function(p){c.warn("Module "+this.name+": "+p)};O.prototype.createError=function(p){return Error("Error in Rangy "+this.name+" module: "+p)};c.createModule=function(p,u){var w=new O(p);c.modules[p]=w;k.push(function(B){u(B,w);w.initialized=true;w.supported=true})};c.requireModules=function(p){for(var u=
  16 +0,w=p.length,B,V;u<w;++u){V=p[u];B=c.modules[V];if(!B||!(B instanceof O))throw Error("Module '"+V+"' not found");if(!B.supported)throw Error("Module '"+V+"' not supported");}};var L=false;q=function(){if(!L){L=true;c.initialized||N()}};if(typeof window=="undefined")C("No window found");else if(typeof document=="undefined")C("No document found");else{l(document,"addEventListener")&&document.addEventListener("DOMContentLoaded",q,false);if(l(window,"addEventListener"))window.addEventListener("load",
  17 +q,false);else l(window,"attachEvent")?window.attachEvent("onload",q):C("Window does not have required addEventListener or attachEvent method");return c}}();
  18 +rangy.createModule("DomUtil",function(l,K){function H(c){for(var f=0;c=c.previousSibling;)f++;return f}function I(c,f){var k=[],r;for(r=c;r;r=r.parentNode)k.push(r);for(r=f;r;r=r.parentNode)if(v(k,r))return r;return null}function z(c,f,k){for(k=k?c:c.parentNode;k;){c=k.parentNode;if(c===f)return k;k=c}return null}function C(c){c=c.nodeType;return c==3||c==4||c==8}function N(c,f){var k=f.nextSibling,r=f.parentNode;k?r.insertBefore(c,k):r.appendChild(c);return c}function O(c){if(c.nodeType==9)return c;
  19 +else if(typeof c.ownerDocument!="undefined")return c.ownerDocument;else if(typeof c.document!="undefined")return c.document;else if(c.parentNode)return O(c.parentNode);else throw Error("getDocument: no document found for node");}function i(c){if(!c)return"[No node]";return C(c)?'"'+c.data+'"':c.nodeType==1?"<"+c.nodeName+(c.id?' id="'+c.id+'"':"")+">["+c.childNodes.length+"]":c.nodeName}function n(c){this._next=this.root=c}function t(c,f){this.node=c;this.offset=f}function x(c){this.code=this[c];
  20 +this.codeName=c;this.message="DOMException: "+this.codeName}var A=l.util;A.areHostMethods(document,["createDocumentFragment","createElement","createTextNode"])||K.fail("document missing a Node creation method");A.isHostMethod(document,"getElementsByTagName")||K.fail("document missing getElementsByTagName method");var q=document.createElement("div");A.areHostMethods(q,["insertBefore","appendChild","cloneNode"])||K.fail("Incomplete Element implementation");A.isHostProperty(q,"innerHTML")||K.fail("Element is missing innerHTML property");
  21 +q=document.createTextNode("test");A.areHostMethods(q,["splitText","deleteData","insertData","appendData","cloneNode"])||K.fail("Incomplete Text Node implementation");var v=function(c,f){for(var k=c.length;k--;)if(c[k]===f)return true;return false};n.prototype={_current:null,hasNext:function(){return!!this._next},next:function(){var c=this._current=this._next,f;if(this._current)if(f=c.firstChild)this._next=f;else{for(f=null;c!==this.root&&!(f=c.nextSibling);)c=c.parentNode;this._next=f}return this._current},
  22 +detach:function(){this._current=this._next=this.root=null}};t.prototype={equals:function(c){return this.node===c.node&this.offset==c.offset},inspect:function(){return"[DomPosition("+i(this.node)+":"+this.offset+")]"}};x.prototype={INDEX_SIZE_ERR:1,HIERARCHY_REQUEST_ERR:3,WRONG_DOCUMENT_ERR:4,NO_MODIFICATION_ALLOWED_ERR:7,NOT_FOUND_ERR:8,NOT_SUPPORTED_ERR:9,INVALID_STATE_ERR:11};x.prototype.toString=function(){return this.message};l.dom={arrayContains:v,isHtmlNamespace:function(c){var f;return typeof c.namespaceURI==
  23 +"undefined"||(f=c.namespaceURI)===null||f=="http://www.w3.org/1999/xhtml"},parentElement:function(c){c=c.parentNode;return c.nodeType==1?c:null},getNodeIndex:H,getNodeLength:function(c){var f;return C(c)?c.length:(f=c.childNodes)?f.length:0},getCommonAncestor:I,isAncestorOf:function(c,f,k){for(f=k?f:f.parentNode;f;)if(f===c)return true;else f=f.parentNode;return false},getClosestAncestorIn:z,isCharacterDataNode:C,insertAfter:N,splitDataNode:function(c,f){var k=c.cloneNode(false);k.deleteData(0,f);
  24 +c.deleteData(f,c.length-f);N(k,c);return k},getDocument:O,getWindow:function(c){c=O(c);if(typeof c.defaultView!="undefined")return c.defaultView;else if(typeof c.parentWindow!="undefined")return c.parentWindow;else throw Error("Cannot get a window object for node");},getIframeWindow:function(c){if(typeof c.contentWindow!="undefined")return c.contentWindow;else if(typeof c.contentDocument!="undefined")return c.contentDocument.defaultView;else throw Error("getIframeWindow: No Window object found for iframe element");
  25 +},getIframeDocument:function(c){if(typeof c.contentDocument!="undefined")return c.contentDocument;else if(typeof c.contentWindow!="undefined")return c.contentWindow.document;else throw Error("getIframeWindow: No Document object found for iframe element");},getBody:function(c){return A.isHostObject(c,"body")?c.body:c.getElementsByTagName("body")[0]},getRootContainer:function(c){for(var f;f=c.parentNode;)c=f;return c},comparePoints:function(c,f,k,r){var L;if(c==k)return f===r?0:f<r?-1:1;else if(L=z(k,
  26 +c,true))return f<=H(L)?-1:1;else if(L=z(c,k,true))return H(L)<r?-1:1;else{f=I(c,k);c=c===f?f:z(c,f,true);k=k===f?f:z(k,f,true);if(c===k)throw Error("comparePoints got to case 4 and childA and childB are the same!");else{for(f=f.firstChild;f;){if(f===c)return-1;else if(f===k)return 1;f=f.nextSibling}throw Error("Should not be here!");}}},inspectNode:i,fragmentFromNodeChildren:function(c){for(var f=O(c).createDocumentFragment(),k;k=c.firstChild;)f.appendChild(k);return f},createIterator:function(c){return new n(c)},
  27 +DomPosition:t};l.DOMException=x});
  28 +rangy.createModule("DomRange",function(l){function K(a,e){return a.nodeType!=3&&(g.isAncestorOf(a,e.startContainer,true)||g.isAncestorOf(a,e.endContainer,true))}function H(a){return g.getDocument(a.startContainer)}function I(a,e,j){if(e=a._listeners[e])for(var o=0,E=e.length;o<E;++o)e[o].call(a,{target:a,args:j})}function z(a){return new Z(a.parentNode,g.getNodeIndex(a))}function C(a){return new Z(a.parentNode,g.getNodeIndex(a)+1)}function N(a,e,j){var o=a.nodeType==11?a.firstChild:a;if(g.isCharacterDataNode(e))j==
  29 +e.length?g.insertAfter(a,e):e.parentNode.insertBefore(a,j==0?e:g.splitDataNode(e,j));else j>=e.childNodes.length?e.appendChild(a):e.insertBefore(a,e.childNodes[j]);return o}function O(a){for(var e,j,o=H(a.range).createDocumentFragment();j=a.next();){e=a.isPartiallySelectedSubtree();j=j.cloneNode(!e);if(e){e=a.getSubtreeIterator();j.appendChild(O(e));e.detach(true)}if(j.nodeType==10)throw new S("HIERARCHY_REQUEST_ERR");o.appendChild(j)}return o}function i(a,e,j){var o,E;for(j=j||{stop:false};o=a.next();)if(a.isPartiallySelectedSubtree())if(e(o)===
  30 +false){j.stop=true;return}else{o=a.getSubtreeIterator();i(o,e,j);o.detach(true);if(j.stop)return}else for(o=g.createIterator(o);E=o.next();)if(e(E)===false){j.stop=true;return}}function n(a){for(var e;a.next();)if(a.isPartiallySelectedSubtree()){e=a.getSubtreeIterator();n(e);e.detach(true)}else a.remove()}function t(a){for(var e,j=H(a.range).createDocumentFragment(),o;e=a.next();){if(a.isPartiallySelectedSubtree()){e=e.cloneNode(false);o=a.getSubtreeIterator();e.appendChild(t(o));o.detach(true)}else a.remove();
  31 +if(e.nodeType==10)throw new S("HIERARCHY_REQUEST_ERR");j.appendChild(e)}return j}function x(a,e,j){var o=!!(e&&e.length),E,T=!!j;if(o)E=RegExp("^("+e.join("|")+")$");var m=[];i(new q(a,false),function(s){if((!o||E.test(s.nodeType))&&(!T||j(s)))m.push(s)});return m}function A(a){return"["+(typeof a.getName=="undefined"?"Range":a.getName())+"("+g.inspectNode(a.startContainer)+":"+a.startOffset+", "+g.inspectNode(a.endContainer)+":"+a.endOffset+")]"}function q(a,e){this.range=a;this.clonePartiallySelectedTextNodes=
  32 +e;if(!a.collapsed){this.sc=a.startContainer;this.so=a.startOffset;this.ec=a.endContainer;this.eo=a.endOffset;var j=a.commonAncestorContainer;if(this.sc===this.ec&&g.isCharacterDataNode(this.sc)){this.isSingleCharacterDataNode=true;this._first=this._last=this._next=this.sc}else{this._first=this._next=this.sc===j&&!g.isCharacterDataNode(this.sc)?this.sc.childNodes[this.so]:g.getClosestAncestorIn(this.sc,j,true);this._last=this.ec===j&&!g.isCharacterDataNode(this.ec)?this.ec.childNodes[this.eo-1]:g.getClosestAncestorIn(this.ec,
  33 +j,true)}}}function v(a){this.code=this[a];this.codeName=a;this.message="RangeException: "+this.codeName}function c(a,e,j){this.nodes=x(a,e,j);this._next=this.nodes[0];this._position=0}function f(a){return function(e,j){for(var o,E=j?e:e.parentNode;E;){o=E.nodeType;if(g.arrayContains(a,o))return E;E=E.parentNode}return null}}function k(a,e){if(G(a,e))throw new v("INVALID_NODE_TYPE_ERR");}function r(a){if(!a.startContainer)throw new S("INVALID_STATE_ERR");}function L(a,e){if(!g.arrayContains(e,a.nodeType))throw new v("INVALID_NODE_TYPE_ERR");
  34 +}function p(a,e){if(e<0||e>(g.isCharacterDataNode(a)?a.length:a.childNodes.length))throw new S("INDEX_SIZE_ERR");}function u(a,e){if(h(a,true)!==h(e,true))throw new S("WRONG_DOCUMENT_ERR");}function w(a){if(D(a,true))throw new S("NO_MODIFICATION_ALLOWED_ERR");}function B(a,e){if(!a)throw new S(e);}function V(a){return!!a.startContainer&&!!a.endContainer&&!(!g.arrayContains(ba,a.startContainer.nodeType)&&!h(a.startContainer,true))&&!(!g.arrayContains(ba,a.endContainer.nodeType)&&!h(a.endContainer,
  35 +true))&&a.startOffset<=(g.isCharacterDataNode(a.startContainer)?a.startContainer.length:a.startContainer.childNodes.length)&&a.endOffset<=(g.isCharacterDataNode(a.endContainer)?a.endContainer.length:a.endContainer.childNodes.length)}function J(a){r(a);if(!V(a))throw Error("Range error: Range is no longer valid after DOM mutation ("+a.inspect()+")");}function ca(){}function Y(a){a.START_TO_START=ia;a.START_TO_END=la;a.END_TO_END=ra;a.END_TO_START=ma;a.NODE_BEFORE=na;a.NODE_AFTER=oa;a.NODE_BEFORE_AND_AFTER=
  36 +pa;a.NODE_INSIDE=ja}function W(a){Y(a);Y(a.prototype)}function da(a,e){return function(){J(this);var j=this.startContainer,o=this.startOffset,E=this.commonAncestorContainer,T=new q(this,true);if(j!==E){j=g.getClosestAncestorIn(j,E,true);o=C(j);j=o.node;o=o.offset}i(T,w);T.reset();E=a(T);T.detach();e(this,j,o,j,o);return E}}function fa(a,e,j){function o(m,s){return function(y){r(this);L(y,$);L(d(y),ba);y=(m?z:C)(y);(s?E:T)(this,y.node,y.offset)}}function E(m,s,y){var F=m.endContainer,Q=m.endOffset;
  37 +if(s!==m.startContainer||y!==m.startOffset){if(d(s)!=d(F)||g.comparePoints(s,y,F,Q)==1){F=s;Q=y}e(m,s,y,F,Q)}}function T(m,s,y){var F=m.startContainer,Q=m.startOffset;if(s!==m.endContainer||y!==m.endOffset){if(d(s)!=d(F)||g.comparePoints(s,y,F,Q)==-1){F=s;Q=y}e(m,F,Q,s,y)}}a.prototype=new ca;l.util.extend(a.prototype,{setStart:function(m,s){r(this);k(m,true);p(m,s);E(this,m,s)},setEnd:function(m,s){r(this);k(m,true);p(m,s);T(this,m,s)},setStartBefore:o(true,true),setStartAfter:o(false,true),setEndBefore:o(true,
  38 +false),setEndAfter:o(false,false),collapse:function(m){J(this);m?e(this,this.startContainer,this.startOffset,this.startContainer,this.startOffset):e(this,this.endContainer,this.endOffset,this.endContainer,this.endOffset)},selectNodeContents:function(m){r(this);k(m,true);e(this,m,0,m,g.getNodeLength(m))},selectNode:function(m){r(this);k(m,false);L(m,$);var s=z(m);m=C(m);e(this,s.node,s.offset,m.node,m.offset)},extractContents:da(t,e),deleteContents:da(n,e),canSurroundContents:function(){J(this);w(this.startContainer);
  39 +w(this.endContainer);var m=new q(this,true),s=m._first&&K(m._first,this)||m._last&&K(m._last,this);m.detach();return!s},detach:function(){j(this)},splitBoundaries:function(){J(this);var m=this.startContainer,s=this.startOffset,y=this.endContainer,F=this.endOffset,Q=m===y;g.isCharacterDataNode(y)&&F>0&&F<y.length&&g.splitDataNode(y,F);if(g.isCharacterDataNode(m)&&s>0&&s<m.length){m=g.splitDataNode(m,s);if(Q){F-=s;y=m}else y==m.parentNode&&F>=g.getNodeIndex(m)&&F++;s=0}e(this,m,s,y,F)},normalizeBoundaries:function(){J(this);
  40 +var m=this.startContainer,s=this.startOffset,y=this.endContainer,F=this.endOffset,Q=function(U){var R=U.nextSibling;if(R&&R.nodeType==U.nodeType){y=U;F=U.length;U.appendData(R.data);R.parentNode.removeChild(R)}},qa=function(U){var R=U.previousSibling;if(R&&R.nodeType==U.nodeType){m=U;var sa=U.length;s=R.length;U.insertData(0,R.data);R.parentNode.removeChild(R);if(m==y){F+=s;y=m}else if(y==U.parentNode){R=g.getNodeIndex(U);if(F==R){y=U;F=sa}else F>R&&F--}}},ga=true;if(g.isCharacterDataNode(y))y.length==
  41 +F&&Q(y);else{if(F>0)(ga=y.childNodes[F-1])&&g.isCharacterDataNode(ga)&&Q(ga);ga=!this.collapsed}if(ga)if(g.isCharacterDataNode(m))s==0&&qa(m);else{if(s<m.childNodes.length)(Q=m.childNodes[s])&&g.isCharacterDataNode(Q)&&qa(Q)}else{m=y;s=F}e(this,m,s,y,F)},collapseToPoint:function(m,s){r(this);k(m,true);p(m,s);if(m!==this.startContainer||s!==this.startOffset||m!==this.endContainer||s!==this.endOffset)e(this,m,s,m,s)}});W(a)}function ea(a){a.collapsed=a.startContainer===a.endContainer&&a.startOffset===
  42 +a.endOffset;a.commonAncestorContainer=a.collapsed?a.startContainer:g.getCommonAncestor(a.startContainer,a.endContainer)}function ha(a,e,j,o,E){var T=a.startContainer!==e||a.startOffset!==j,m=a.endContainer!==o||a.endOffset!==E;a.startContainer=e;a.startOffset=j;a.endContainer=o;a.endOffset=E;ea(a);I(a,"boundarychange",{startMoved:T,endMoved:m})}function M(a){this.startContainer=a;this.startOffset=0;this.endContainer=a;this.endOffset=0;this._listeners={boundarychange:[],detach:[]};ea(this)}l.requireModules(["DomUtil"]);
  43 +var g=l.dom,Z=g.DomPosition,S=l.DOMException;q.prototype={_current:null,_next:null,_first:null,_last:null,isSingleCharacterDataNode:false,reset:function(){this._current=null;this._next=this._first},hasNext:function(){return!!this._next},next:function(){var a=this._current=this._next;if(a){this._next=a!==this._last?a.nextSibling:null;if(g.isCharacterDataNode(a)&&this.clonePartiallySelectedTextNodes){if(a===this.ec)(a=a.cloneNode(true)).deleteData(this.eo,a.length-this.eo);if(this._current===this.sc)(a=
  44 +a.cloneNode(true)).deleteData(0,this.so)}}return a},remove:function(){var a=this._current,e,j;if(g.isCharacterDataNode(a)&&(a===this.sc||a===this.ec)){e=a===this.sc?this.so:0;j=a===this.ec?this.eo:a.length;e!=j&&a.deleteData(e,j-e)}else a.parentNode&&a.parentNode.removeChild(a)},isPartiallySelectedSubtree:function(){return K(this._current,this.range)},getSubtreeIterator:function(){var a;if(this.isSingleCharacterDataNode){a=this.range.cloneRange();a.collapse()}else{a=new M(H(this.range));var e=this._current,
  45 +j=e,o=0,E=e,T=g.getNodeLength(e);if(g.isAncestorOf(e,this.sc,true)){j=this.sc;o=this.so}if(g.isAncestorOf(e,this.ec,true)){E=this.ec;T=this.eo}ha(a,j,o,E,T)}return new q(a,this.clonePartiallySelectedTextNodes)},detach:function(a){a&&this.range.detach();this.range=this._current=this._next=this._first=this._last=this.sc=this.so=this.ec=this.eo=null}};v.prototype={BAD_BOUNDARYPOINTS_ERR:1,INVALID_NODE_TYPE_ERR:2};v.prototype.toString=function(){return this.message};c.prototype={_current:null,hasNext:function(){return!!this._next},
  46 +next:function(){this._current=this._next;this._next=this.nodes[++this._position];return this._current},detach:function(){this._current=this._next=this.nodes=null}};var $=[1,3,4,5,7,8,10],ba=[2,9,11],aa=[1,3,4,5,7,8,10,11],b=[1,3,4,5,7,8],d=g.getRootContainer,h=f([9,11]),D=f([5,6,10,12]),G=f([6,10,12]),P=document.createElement("style"),X=false;try{P.innerHTML="<b>x</b>";X=P.firstChild.nodeType==3}catch(ta){}l.features.htmlParsingConforms=X;var ka=["startContainer","startOffset","endContainer","endOffset",
  47 +"collapsed","commonAncestorContainer"],ia=0,la=1,ra=2,ma=3,na=0,oa=1,pa=2,ja=3;ca.prototype={attachListener:function(a,e){this._listeners[a].push(e)},compareBoundaryPoints:function(a,e){J(this);u(this.startContainer,e.startContainer);var j=a==ma||a==ia?"start":"end",o=a==la||a==ia?"start":"end";return g.comparePoints(this[j+"Container"],this[j+"Offset"],e[o+"Container"],e[o+"Offset"])},insertNode:function(a){J(this);L(a,aa);w(this.startContainer);if(g.isAncestorOf(a,this.startContainer,true))throw new S("HIERARCHY_REQUEST_ERR");
  48 +this.setStartBefore(N(a,this.startContainer,this.startOffset))},cloneContents:function(){J(this);var a,e;if(this.collapsed)return H(this).createDocumentFragment();else{if(this.startContainer===this.endContainer&&g.isCharacterDataNode(this.startContainer)){a=this.startContainer.cloneNode(true);a.data=a.data.slice(this.startOffset,this.endOffset);e=H(this).createDocumentFragment();e.appendChild(a);return e}else{e=new q(this,true);a=O(e);e.detach()}return a}},canSurroundContents:function(){J(this);w(this.startContainer);
  49 +w(this.endContainer);var a=new q(this,true),e=a._first&&K(a._first,this)||a._last&&K(a._last,this);a.detach();return!e},surroundContents:function(a){L(a,b);if(!this.canSurroundContents())throw new v("BAD_BOUNDARYPOINTS_ERR");var e=this.extractContents();if(a.hasChildNodes())for(;a.lastChild;)a.removeChild(a.lastChild);N(a,this.startContainer,this.startOffset);a.appendChild(e);this.selectNode(a)},cloneRange:function(){J(this);for(var a=new M(H(this)),e=ka.length,j;e--;){j=ka[e];a[j]=this[j]}return a},
  50 +toString:function(){J(this);var a=this.startContainer;if(a===this.endContainer&&g.isCharacterDataNode(a))return a.nodeType==3||a.nodeType==4?a.data.slice(this.startOffset,this.endOffset):"";else{var e=[];a=new q(this,true);i(a,function(j){if(j.nodeType==3||j.nodeType==4)e.push(j.data)});a.detach();return e.join("")}},compareNode:function(a){J(this);var e=a.parentNode,j=g.getNodeIndex(a);if(!e)throw new S("NOT_FOUND_ERR");a=this.comparePoint(e,j);e=this.comparePoint(e,j+1);return a<0?e>0?pa:na:e>0?
  51 +oa:ja},comparePoint:function(a,e){J(this);B(a,"HIERARCHY_REQUEST_ERR");u(a,this.startContainer);if(g.comparePoints(a,e,this.startContainer,this.startOffset)<0)return-1;else if(g.comparePoints(a,e,this.endContainer,this.endOffset)>0)return 1;return 0},createContextualFragment:X?function(a){var e=this.startContainer,j=g.getDocument(e);if(!e)throw new S("INVALID_STATE_ERR");var o=null;if(e.nodeType==1)o=e;else if(g.isCharacterDataNode(e))o=g.parentElement(e);o=o===null||o.nodeName=="HTML"&&g.isHtmlNamespace(g.getDocument(o).documentElement)&&
  52 +g.isHtmlNamespace(o)?j.createElement("body"):o.cloneNode(false);o.innerHTML=a;return g.fragmentFromNodeChildren(o)}:function(a){r(this);var e=H(this).createElement("body");e.innerHTML=a;return g.fragmentFromNodeChildren(e)},toHtml:function(){J(this);var a=H(this).createElement("div");a.appendChild(this.cloneContents());return a.innerHTML},intersectsNode:function(a,e){J(this);B(a,"NOT_FOUND_ERR");if(g.getDocument(a)!==H(this))return false;var j=a.parentNode,o=g.getNodeIndex(a);B(j,"NOT_FOUND_ERR");
  53 +var E=g.comparePoints(j,o,this.endContainer,this.endOffset);j=g.comparePoints(j,o+1,this.startContainer,this.startOffset);return e?E<=0&&j>=0:E<0&&j>0},isPointInRange:function(a,e){J(this);B(a,"HIERARCHY_REQUEST_ERR");u(a,this.startContainer);return g.comparePoints(a,e,this.startContainer,this.startOffset)>=0&&g.comparePoints(a,e,this.endContainer,this.endOffset)<=0},intersectsRange:function(a,e){J(this);if(H(a)!=H(this))throw new S("WRONG_DOCUMENT_ERR");var j=g.comparePoints(this.startContainer,
  54 +this.startOffset,a.endContainer,a.endOffset),o=g.comparePoints(this.endContainer,this.endOffset,a.startContainer,a.startOffset);return e?j<=0&&o>=0:j<0&&o>0},intersection:function(a){if(this.intersectsRange(a)){var e=g.comparePoints(this.startContainer,this.startOffset,a.startContainer,a.startOffset),j=g.comparePoints(this.endContainer,this.endOffset,a.endContainer,a.endOffset),o=this.cloneRange();e==-1&&o.setStart(a.startContainer,a.startOffset);j==1&&o.setEnd(a.endContainer,a.endOffset);return o}return null},
  55 +union:function(a){if(this.intersectsRange(a,true)){var e=this.cloneRange();g.comparePoints(a.startContainer,a.startOffset,this.startContainer,this.startOffset)==-1&&e.setStart(a.startContainer,a.startOffset);g.comparePoints(a.endContainer,a.endOffset,this.endContainer,this.endOffset)==1&&e.setEnd(a.endContainer,a.endOffset);return e}else throw new v("Ranges do not intersect");},containsNode:function(a,e){return e?this.intersectsNode(a,false):this.compareNode(a)==ja},containsNodeContents:function(a){return this.comparePoint(a,
  56 +0)>=0&&this.comparePoint(a,g.getNodeLength(a))<=0},containsRange:function(a){return this.intersection(a).equals(a)},containsNodeText:function(a){var e=this.cloneRange();e.selectNode(a);var j=e.getNodes([3]);if(j.length>0){e.setStart(j[0],0);a=j.pop();e.setEnd(a,a.length);a=this.containsRange(e);e.detach();return a}else return this.containsNodeContents(a)},createNodeIterator:function(a,e){J(this);return new c(this,a,e)},getNodes:function(a,e){J(this);return x(this,a,e)},getDocument:function(){return H(this)},
  57 +collapseBefore:function(a){r(this);this.setEndBefore(a);this.collapse(false)},collapseAfter:function(a){r(this);this.setStartAfter(a);this.collapse(true)},getName:function(){return"DomRange"},equals:function(a){return M.rangesEqual(this,a)},isValid:function(){return V(this)},inspect:function(){return A(this)}};fa(M,ha,function(a){r(a);a.startContainer=a.startOffset=a.endContainer=a.endOffset=null;a.collapsed=a.commonAncestorContainer=null;I(a,"detach",null);a._listeners=null});l.rangePrototype=ca.prototype;
  58 +M.rangeProperties=ka;M.RangeIterator=q;M.copyComparisonConstants=W;M.createPrototypeRange=fa;M.inspect=A;M.getRangeDocument=H;M.rangesEqual=function(a,e){return a.startContainer===e.startContainer&&a.startOffset===e.startOffset&&a.endContainer===e.endContainer&&a.endOffset===e.endOffset};l.DomRange=M;l.RangeException=v});
  59 +rangy.createModule("WrappedRange",function(l){function K(i,n,t,x){var A=i.duplicate();A.collapse(t);var q=A.parentElement();z.isAncestorOf(n,q,true)||(q=n);if(!q.canHaveHTML)return new C(q.parentNode,z.getNodeIndex(q));n=z.getDocument(q).createElement("span");var v,c=t?"StartToStart":"StartToEnd";do{q.insertBefore(n,n.previousSibling);A.moveToElementText(n)}while((v=A.compareEndPoints(c,i))>0&&n.previousSibling);c=n.nextSibling;if(v==-1&&c&&z.isCharacterDataNode(c)){A.setEndPoint(t?"EndToStart":"EndToEnd",
  60 +i);if(/[\r\n]/.test(c.data)){q=A.duplicate();t=q.text.replace(/\r\n/g,"\r").length;for(t=q.moveStart("character",t);q.compareEndPoints("StartToEnd",q)==-1;){t++;q.moveStart("character",1)}}else t=A.text.length;q=new C(c,t)}else{c=(x||!t)&&n.previousSibling;q=(t=(x||t)&&n.nextSibling)&&z.isCharacterDataNode(t)?new C(t,0):c&&z.isCharacterDataNode(c)?new C(c,c.length):new C(q,z.getNodeIndex(n))}n.parentNode.removeChild(n);return q}function H(i,n){var t,x,A=i.offset,q=z.getDocument(i.node),v=q.body.createTextRange(),
  61 +c=z.isCharacterDataNode(i.node);if(c){t=i.node;x=t.parentNode}else{t=i.node.childNodes;t=A<t.length?t[A]:null;x=i.node}q=q.createElement("span");q.innerHTML="&#feff;";t?x.insertBefore(q,t):x.appendChild(q);v.moveToElementText(q);v.collapse(!n);x.removeChild(q);if(c)v[n?"moveStart":"moveEnd"]("character",A);return v}l.requireModules(["DomUtil","DomRange"]);var I,z=l.dom,C=z.DomPosition,N=l.DomRange;if(l.features.implementsDomRange&&(!l.features.implementsTextRange||!l.config.preferTextRange)){(function(){function i(f){for(var k=
  62 +t.length,r;k--;){r=t[k];f[r]=f.nativeRange[r]}}var n,t=N.rangeProperties,x,A;I=function(f){if(!f)throw Error("Range must be specified");this.nativeRange=f;i(this)};N.createPrototypeRange(I,function(f,k,r,L,p){var u=f.endContainer!==L||f.endOffset!=p;if(f.startContainer!==k||f.startOffset!=r||u){f.setEnd(L,p);f.setStart(k,r)}},function(f){f.nativeRange.detach();f.detached=true;for(var k=t.length,r;k--;){r=t[k];f[r]=null}});n=I.prototype;n.selectNode=function(f){this.nativeRange.selectNode(f);i(this)};
  63 +n.deleteContents=function(){this.nativeRange.deleteContents();i(this)};n.extractContents=function(){var f=this.nativeRange.extractContents();i(this);return f};n.cloneContents=function(){return this.nativeRange.cloneContents()};n.surroundContents=function(f){this.nativeRange.surroundContents(f);i(this)};n.collapse=function(f){this.nativeRange.collapse(f);i(this)};n.cloneRange=function(){return new I(this.nativeRange.cloneRange())};n.refresh=function(){i(this)};n.toString=function(){return this.nativeRange.toString()};
  64 +var q=document.createTextNode("test");z.getBody(document).appendChild(q);var v=document.createRange();v.setStart(q,0);v.setEnd(q,0);try{v.setStart(q,1);x=true;n.setStart=function(f,k){this.nativeRange.setStart(f,k);i(this)};n.setEnd=function(f,k){this.nativeRange.setEnd(f,k);i(this)};A=function(f){return function(k){this.nativeRange[f](k);i(this)}}}catch(c){x=false;n.setStart=function(f,k){try{this.nativeRange.setStart(f,k)}catch(r){this.nativeRange.setEnd(f,k);this.nativeRange.setStart(f,k)}i(this)};
  65 +n.setEnd=function(f,k){try{this.nativeRange.setEnd(f,k)}catch(r){this.nativeRange.setStart(f,k);this.nativeRange.setEnd(f,k)}i(this)};A=function(f,k){return function(r){try{this.nativeRange[f](r)}catch(L){this.nativeRange[k](r);this.nativeRange[f](r)}i(this)}}}n.setStartBefore=A("setStartBefore","setEndBefore");n.setStartAfter=A("setStartAfter","setEndAfter");n.setEndBefore=A("setEndBefore","setStartBefore");n.setEndAfter=A("setEndAfter","setStartAfter");v.selectNodeContents(q);n.selectNodeContents=
  66 +v.startContainer==q&&v.endContainer==q&&v.startOffset==0&&v.endOffset==q.length?function(f){this.nativeRange.selectNodeContents(f);i(this)}:function(f){this.setStart(f,0);this.setEnd(f,N.getEndOffset(f))};v.selectNodeContents(q);v.setEnd(q,3);x=document.createRange();x.selectNodeContents(q);x.setEnd(q,4);x.setStart(q,2);n.compareBoundaryPoints=v.compareBoundaryPoints(v.START_TO_END,x)==-1&v.compareBoundaryPoints(v.END_TO_START,x)==1?function(f,k){k=k.nativeRange||k;if(f==k.START_TO_END)f=k.END_TO_START;
  67 +else if(f==k.END_TO_START)f=k.START_TO_END;return this.nativeRange.compareBoundaryPoints(f,k)}:function(f,k){return this.nativeRange.compareBoundaryPoints(f,k.nativeRange||k)};if(l.util.isHostMethod(v,"createContextualFragment"))n.createContextualFragment=function(f){return this.nativeRange.createContextualFragment(f)};z.getBody(document).removeChild(q);v.detach();x.detach()})();l.createNativeRange=function(i){i=i||document;return i.createRange()}}else if(l.features.implementsTextRange){I=function(i){this.textRange=
  68 +i;this.refresh()};I.prototype=new N(document);I.prototype.refresh=function(){var i,n,t=this.textRange;i=t.parentElement();var x=t.duplicate();x.collapse(true);n=x.parentElement();x=t.duplicate();x.collapse(false);t=x.parentElement();n=n==t?n:z.getCommonAncestor(n,t);n=n==i?n:z.getCommonAncestor(i,n);if(this.textRange.compareEndPoints("StartToEnd",this.textRange)==0)n=i=K(this.textRange,n,true,true);else{i=K(this.textRange,n,true,false);n=K(this.textRange,n,false,false)}this.setStart(i.node,i.offset);
  69 +this.setEnd(n.node,n.offset)};N.copyComparisonConstants(I);var O=function(){return this}();if(typeof O.Range=="undefined")O.Range=I;l.createNativeRange=function(i){i=i||document;return i.body.createTextRange()}}if(l.features.implementsTextRange)I.rangeToTextRange=function(i){if(i.collapsed)return H(new C(i.startContainer,i.startOffset),true);else{var n=H(new C(i.startContainer,i.startOffset),true),t=H(new C(i.endContainer,i.endOffset),false);i=z.getDocument(i.startContainer).body.createTextRange();
  70 +i.setEndPoint("StartToStart",n);i.setEndPoint("EndToEnd",t);return i}};I.prototype.getName=function(){return"WrappedRange"};l.WrappedRange=I;l.createRange=function(i){i=i||document;return new I(l.createNativeRange(i))};l.createRangyRange=function(i){i=i||document;return new N(i)};l.createIframeRange=function(i){return l.createRange(z.getIframeDocument(i))};l.createIframeRangyRange=function(i){return l.createRangyRange(z.getIframeDocument(i))};l.addCreateMissingNativeApiListener(function(i){i=i.document;
  71 +if(typeof i.createRange=="undefined")i.createRange=function(){return l.createRange(this)};i=i=null})});
  72 +rangy.createModule("WrappedSelection",function(l,K){function H(b){return(b||window).getSelection()}function I(b){return(b||window).document.selection}function z(b,d,h){var D=h?"end":"start";h=h?"start":"end";b.anchorNode=d[D+"Container"];b.anchorOffset=d[D+"Offset"];b.focusNode=d[h+"Container"];b.focusOffset=d[h+"Offset"]}function C(b){b.anchorNode=b.focusNode=null;b.anchorOffset=b.focusOffset=0;b.rangeCount=0;b.isCollapsed=true;b._ranges.length=0}function N(b){var d;if(b instanceof k){d=b._selectionNativeRange;
  73 +if(!d){d=l.createNativeRange(c.getDocument(b.startContainer));d.setEnd(b.endContainer,b.endOffset);d.setStart(b.startContainer,b.startOffset);b._selectionNativeRange=d;b.attachListener("detach",function(){this._selectionNativeRange=null})}}else if(b instanceof r)d=b.nativeRange;else if(l.features.implementsDomRange&&b instanceof c.getWindow(b.startContainer).Range)d=b;return d}function O(b){var d=b.getNodes(),h;a:if(!d.length||d[0].nodeType!=1)h=false;else{h=1;for(var D=d.length;h<D;++h)if(!c.isAncestorOf(d[0],
  74 +d[h])){h=false;break a}h=true}if(!h)throw Error("getSingleElementFromRange: range "+b.inspect()+" did not consist of a single element");return d[0]}function i(b,d){var h=new r(d);b._ranges=[h];z(b,h,false);b.rangeCount=1;b.isCollapsed=h.collapsed}function n(b){b._ranges.length=0;if(b.docSelection.type=="None")C(b);else{var d=b.docSelection.createRange();if(d&&typeof d.text!="undefined")i(b,d);else{b.rangeCount=d.length;for(var h,D=c.getDocument(d.item(0)),G=0;G<b.rangeCount;++G){h=l.createRange(D);
  75 +h.selectNode(d.item(G));b._ranges.push(h)}b.isCollapsed=b.rangeCount==1&&b._ranges[0].collapsed;z(b,b._ranges[b.rangeCount-1],false)}}}function t(b,d){var h=b.docSelection.createRange(),D=O(d),G=c.getDocument(h.item(0));G=c.getBody(G).createControlRange();for(var P=0,X=h.length;P<X;++P)G.add(h.item(P));try{G.add(D)}catch(ta){throw Error("addRange(): Element within the specified Range could not be added to control selection (does it have layout?)");}G.select();n(b)}function x(b,d,h){this.nativeSelection=
  76 +b;this.docSelection=d;this._ranges=[];this.win=h;this.refresh()}function A(b,d){var h=c.getDocument(d[0].startContainer);h=c.getBody(h).createControlRange();for(var D=0,G;D<rangeCount;++D){G=O(d[D]);try{h.add(G)}catch(P){throw Error("setRanges(): Element within the one of the specified Ranges could not be added to control selection (does it have layout?)");}}h.select();n(b)}function q(b,d){if(b.anchorNode&&c.getDocument(b.anchorNode)!==c.getDocument(d))throw new L("WRONG_DOCUMENT_ERR");}function v(b){var d=
  77 +[],h=new p(b.anchorNode,b.anchorOffset),D=new p(b.focusNode,b.focusOffset),G=typeof b.getName=="function"?b.getName():"Selection";if(typeof b.rangeCount!="undefined")for(var P=0,X=b.rangeCount;P<X;++P)d[P]=k.inspect(b.getRangeAt(P));return"["+G+"(Ranges: "+d.join(", ")+")(anchor: "+h.inspect()+", focus: "+D.inspect()+"]"}l.requireModules(["DomUtil","DomRange","WrappedRange"]);l.config.checkSelectionRanges=true;var c=l.dom,f=l.util,k=l.DomRange,r=l.WrappedRange,L=l.DOMException,p=c.DomPosition,u,w,
  78 +B=l.util.isHostMethod(window,"getSelection"),V=l.util.isHostObject(document,"selection"),J=V&&(!B||l.config.preferTextRange);if(J){u=I;l.isSelectionValid=function(b){b=(b||window).document;var d=b.selection;return d.type!="None"||c.getDocument(d.createRange().parentElement())==b}}else if(B){u=H;l.isSelectionValid=function(){return true}}else K.fail("Neither document.selection or window.getSelection() detected.");l.getNativeSelection=u;B=u();var ca=l.createNativeRange(document),Y=c.getBody(document),
  79 +W=f.areHostObjects(B,f.areHostProperties(B,["anchorOffset","focusOffset"]));l.features.selectionHasAnchorAndFocus=W;var da=f.isHostMethod(B,"extend");l.features.selectionHasExtend=da;var fa=typeof B.rangeCount=="number";l.features.selectionHasRangeCount=fa;var ea=false,ha=true;f.areHostMethods(B,["addRange","getRangeAt","removeAllRanges"])&&typeof B.rangeCount=="number"&&l.features.implementsDomRange&&function(){var b=document.createElement("iframe");b.frameBorder=0;b.style.position="absolute";b.style.left=
  80 +"-10000px";Y.appendChild(b);var d=c.getIframeDocument(b);d.open();d.write("<html><head></head><body>12</body></html>");d.close();var h=c.getIframeWindow(b).getSelection(),D=d.documentElement.lastChild.firstChild;d=d.createRange();d.setStart(D,1);d.collapse(true);h.addRange(d);ha=h.rangeCount==1;h.removeAllRanges();var G=d.cloneRange();d.setStart(D,0);G.setEnd(D,2);h.addRange(d);h.addRange(G);ea=h.rangeCount==2;d.detach();G.detach();Y.removeChild(b)}();l.features.selectionSupportsMultipleRanges=ea;
  81 +l.features.collapsedNonEditableSelectionsSupported=ha;var M=false,g;if(Y&&f.isHostMethod(Y,"createControlRange")){g=Y.createControlRange();if(f.areHostProperties(g,["item","add"]))M=true}l.features.implementsControlRange=M;w=W?function(b){return b.anchorNode===b.focusNode&&b.anchorOffset===b.focusOffset}:function(b){return b.rangeCount?b.getRangeAt(b.rangeCount-1).collapsed:false};var Z;if(f.isHostMethod(B,"getRangeAt"))Z=function(b,d){try{return b.getRangeAt(d)}catch(h){return null}};else if(W)Z=
  82 +function(b){var d=c.getDocument(b.anchorNode);d=l.createRange(d);d.setStart(b.anchorNode,b.anchorOffset);d.setEnd(b.focusNode,b.focusOffset);if(d.collapsed!==this.isCollapsed){d.setStart(b.focusNode,b.focusOffset);d.setEnd(b.anchorNode,b.anchorOffset)}return d};l.getSelection=function(b){b=b||window;var d=b._rangySelection,h=u(b),D=V?I(b):null;if(d){d.nativeSelection=h;d.docSelection=D;d.refresh(b)}else{d=new x(h,D,b);b._rangySelection=d}return d};l.getIframeSelection=function(b){return l.getSelection(c.getIframeWindow(b))};
  83 +g=x.prototype;if(!J&&W&&f.areHostMethods(B,["removeAllRanges","addRange"])){g.removeAllRanges=function(){this.nativeSelection.removeAllRanges();C(this)};var S=function(b,d){var h=k.getRangeDocument(d);h=l.createRange(h);h.collapseToPoint(d.endContainer,d.endOffset);b.nativeSelection.addRange(N(h));b.nativeSelection.extend(d.startContainer,d.startOffset);b.refresh()};g.addRange=fa?function(b,d){if(M&&V&&this.docSelection.type=="Control")t(this,b);else if(d&&da)S(this,b);else{var h;if(ea)h=this.rangeCount;
  84 +else{this.removeAllRanges();h=0}this.nativeSelection.addRange(N(b));this.rangeCount=this.nativeSelection.rangeCount;if(this.rangeCount==h+1){if(l.config.checkSelectionRanges)if((h=Z(this.nativeSelection,this.rangeCount-1))&&!k.rangesEqual(h,b))b=new r(h);this._ranges[this.rangeCount-1]=b;z(this,b,aa(this.nativeSelection));this.isCollapsed=w(this)}else this.refresh()}}:function(b,d){if(d&&da)S(this,b);else{this.nativeSelection.addRange(N(b));this.refresh()}};g.setRanges=function(b){if(M&&b.length>
  85 +1)A(this,b);else{this.removeAllRanges();for(var d=0,h=b.length;d<h;++d)this.addRange(b[d])}}}else if(f.isHostMethod(B,"empty")&&f.isHostMethod(ca,"select")&&M&&J){g.removeAllRanges=function(){try{this.docSelection.empty();if(this.docSelection.type!="None"){var b;if(this.anchorNode)b=c.getDocument(this.anchorNode);else if(this.docSelection.type=="Control"){var d=this.docSelection.createRange();if(d.length)b=c.getDocument(d.item(0)).body.createTextRange()}if(b){b.body.createTextRange().select();this.docSelection.empty()}}}catch(h){}C(this)};
  86 +g.addRange=function(b){if(this.docSelection.type=="Control")t(this,b);else{r.rangeToTextRange(b).select();this._ranges[0]=b;this.rangeCount=1;this.isCollapsed=this._ranges[0].collapsed;z(this,b,false)}};g.setRanges=function(b){this.removeAllRanges();var d=b.length;if(d>1)A(this,b);else d&&this.addRange(b[0])}}else{K.fail("No means of selecting a Range or TextRange was found");return false}g.getRangeAt=function(b){if(b<0||b>=this.rangeCount)throw new L("INDEX_SIZE_ERR");else return this._ranges[b]};
  87 +var $;if(J)$=function(b){var d;if(l.isSelectionValid(b.win))d=b.docSelection.createRange();else{d=c.getBody(b.win.document).createTextRange();d.collapse(true)}if(b.docSelection.type=="Control")n(b);else d&&typeof d.text!="undefined"?i(b,d):C(b)};else if(f.isHostMethod(B,"getRangeAt")&&typeof B.rangeCount=="number")$=function(b){if(M&&V&&b.docSelection.type=="Control")n(b);else{b._ranges.length=b.rangeCount=b.nativeSelection.rangeCount;if(b.rangeCount){for(var d=0,h=b.rangeCount;d<h;++d)b._ranges[d]=
  88 +new l.WrappedRange(b.nativeSelection.getRangeAt(d));z(b,b._ranges[b.rangeCount-1],aa(b.nativeSelection));b.isCollapsed=w(b)}else C(b)}};else if(W&&typeof B.isCollapsed=="boolean"&&typeof ca.collapsed=="boolean"&&l.features.implementsDomRange)$=function(b){var d;d=b.nativeSelection;if(d.anchorNode){d=Z(d,0);b._ranges=[d];b.rangeCount=1;d=b.nativeSelection;b.anchorNode=d.anchorNode;b.anchorOffset=d.anchorOffset;b.focusNode=d.focusNode;b.focusOffset=d.focusOffset;b.isCollapsed=w(b)}else C(b)};else{K.fail("No means of obtaining a Range or TextRange from the user's selection was found");
  89 +return false}g.refresh=function(b){var d=b?this._ranges.slice(0):null;$(this);if(b){b=d.length;if(b!=this._ranges.length)return false;for(;b--;)if(!k.rangesEqual(d[b],this._ranges[b]))return false;return true}};var ba=function(b,d){var h=b.getAllRanges(),D=false;b.removeAllRanges();for(var G=0,P=h.length;G<P;++G)if(D||d!==h[G])b.addRange(h[G]);else D=true;b.rangeCount||C(b)};g.removeRange=M?function(b){if(this.docSelection.type=="Control"){var d=this.docSelection.createRange();b=O(b);var h=c.getDocument(d.item(0));
  90 +h=c.getBody(h).createControlRange();for(var D,G=false,P=0,X=d.length;P<X;++P){D=d.item(P);if(D!==b||G)h.add(d.item(P));else G=true}h.select();n(this)}else ba(this,b)}:function(b){ba(this,b)};var aa;if(!J&&W&&l.features.implementsDomRange){aa=function(b){var d=false;if(b.anchorNode)d=c.comparePoints(b.anchorNode,b.anchorOffset,b.focusNode,b.focusOffset)==1;return d};g.isBackwards=function(){return aa(this)}}else aa=g.isBackwards=function(){return false};g.toString=function(){for(var b=[],d=0,h=this.rangeCount;d<
  91 +h;++d)b[d]=""+this._ranges[d];return b.join("")};g.collapse=function(b,d){q(this,b);var h=l.createRange(c.getDocument(b));h.collapseToPoint(b,d);this.removeAllRanges();this.addRange(h);this.isCollapsed=true};g.collapseToStart=function(){if(this.rangeCount){var b=this._ranges[0];this.collapse(b.startContainer,b.startOffset)}else throw new L("INVALID_STATE_ERR");};g.collapseToEnd=function(){if(this.rangeCount){var b=this._ranges[this.rangeCount-1];this.collapse(b.endContainer,b.endOffset)}else throw new L("INVALID_STATE_ERR");
  92 +};g.selectAllChildren=function(b){q(this,b);var d=l.createRange(c.getDocument(b));d.selectNodeContents(b);this.removeAllRanges();this.addRange(d)};g.deleteFromDocument=function(){if(M&&V&&this.docSelection.type=="Control"){for(var b=this.docSelection.createRange(),d;b.length;){d=b.item(0);b.remove(d);d.parentNode.removeChild(d)}this.refresh()}else if(this.rangeCount){b=this.getAllRanges();this.removeAllRanges();d=0;for(var h=b.length;d<h;++d)b[d].deleteContents();this.addRange(b[h-1])}};g.getAllRanges=
  93 +function(){return this._ranges.slice(0)};g.setSingleRange=function(b){this.setRanges([b])};g.containsNode=function(b,d){for(var h=0,D=this._ranges.length;h<D;++h)if(this._ranges[h].containsNode(b,d))return true;return false};g.toHtml=function(){var b="";if(this.rangeCount){b=k.getRangeDocument(this._ranges[0]).createElement("div");for(var d=0,h=this._ranges.length;d<h;++d)b.appendChild(this._ranges[d].cloneContents());b=b.innerHTML}return b};g.getName=function(){return"WrappedSelection"};g.inspect=
  94 +function(){return v(this)};g.detach=function(){this.win=this.anchorNode=this.focusNode=this.win._rangySelection=null};x.inspect=v;l.Selection=x;l.selectionPrototype=g;l.addCreateMissingNativeApiListener(function(b){if(typeof b.getSelection=="undefined")b.getSelection=function(){return l.getSelection(this)};b=null})});
0 95 \ No newline at end of file
... ...
plugins/comment_paragraph/public/rangy-cssclassapplier.js 0 → 100644
... ... @@ -0,0 +1,32 @@
  1 +/*
  2 + CSS Class Applier module for Rangy.
  3 + Adds, removes and toggles CSS classes on Ranges and Selections
  4 +
  5 + Part of Rangy, a cross-browser JavaScript range and selection library
  6 + http://code.google.com/p/rangy/
  7 +
  8 + Depends on Rangy core.
  9 +
  10 + Copyright 2012, Tim Down
  11 + Licensed under the MIT license.
  12 + Version: 1.2.3
  13 + Build date: 26 February 2012
  14 +*/
  15 +rangy.createModule("CssClassApplier",function(i,v){function r(a,b){return a.className&&RegExp("(?:^|\\s)"+b+"(?:\\s|$)").test(a.className)}function s(a,b){if(a.className)r(a,b)||(a.className+=" "+b);else a.className=b}function o(a){return a.split(/\s+/).sort().join(" ")}function w(a,b){return o(a.className)==o(b.className)}function x(a){for(var b=a.parentNode;a.hasChildNodes();)b.insertBefore(a.firstChild,a);b.removeChild(a)}function y(a,b){var c=a.cloneRange();c.selectNodeContents(b);var d=c.intersection(a);
  16 +d=d?d.toString():"";c.detach();return d!=""}function z(a){return a.getNodes([3],function(b){return y(a,b)})}function A(a,b){if(a.attributes.length!=b.attributes.length)return false;for(var c=0,d=a.attributes.length,e,f;c<d;++c){e=a.attributes[c];f=e.name;if(f!="class"){f=b.attributes.getNamedItem(f);if(e.specified!=f.specified)return false;if(e.specified&&e.nodeValue!==f.nodeValue)return false}}return true}function B(a,b){for(var c=0,d=a.attributes.length,e;c<d;++c){e=a.attributes[c].name;if(!(b&&
  17 +h.arrayContains(b,e))&&a.attributes[c].specified&&e!="class")return true}return false}function C(a){var b;return a&&a.nodeType==1&&((b=a.parentNode)&&b.nodeType==9&&b.designMode=="on"||k(a)&&!k(a.parentNode))}function D(a){return(k(a)||a.nodeType!=1&&k(a.parentNode))&&!C(a)}function E(a){return a&&a.nodeType==1&&!M.test(p(a,"display"))}function N(a){if(a.data.length==0)return true;if(O.test(a.data))return false;switch(p(a.parentNode,"whiteSpace")){case "pre":case "pre-wrap":case "-moz-pre-wrap":return false;
  18 +case "pre-line":if(/[\r\n]/.test(a.data))return false}return E(a.previousSibling)||E(a.nextSibling)}function m(a,b,c,d){var e,f=c==0;if(h.isAncestorOf(b,a))return a;if(h.isCharacterDataNode(b))if(c==0){c=h.getNodeIndex(b);b=b.parentNode}else if(c==b.length){c=h.getNodeIndex(b)+1;b=b.parentNode}else throw v.createError("splitNodeAt should not be called with offset in the middle of a data node ("+c+" in "+b.data);var g;g=b;var j=c;g=h.isCharacterDataNode(g)?j==0?!!g.previousSibling:j==g.length?!!g.nextSibling:
  19 +true:j>0&&j<g.childNodes.length;if(g){if(!e){e=b.cloneNode(false);for(e.id&&e.removeAttribute("id");f=b.childNodes[c];)e.appendChild(f);h.insertAfter(e,b)}return b==a?e:m(a,e.parentNode,h.getNodeIndex(e),d)}else if(a!=b){e=b.parentNode;b=h.getNodeIndex(b);f||b++;return m(a,e,b,d)}return a}function F(a){var b=a?"nextSibling":"previousSibling";return function(c,d){var e=c.parentNode,f=c[b];if(f){if(f&&f.nodeType==3)return f}else if(d)if((f=e[b])&&f.nodeType==1&&e.tagName==f.tagName&&w(e,f)&&A(e,f))return f[a?
  20 +"firstChild":"lastChild"];return null}}function t(a){this.firstTextNode=(this.isElementMerge=a.nodeType==1)?a.lastChild:a;this.textNodes=[this.firstTextNode]}function q(a,b,c){this.cssClass=a;var d,e,f=null;if(typeof b=="object"&&b!==null){c=b.tagNames;f=b.elementProperties;for(d=0;e=P[d++];)if(b.hasOwnProperty(e))this[e]=b[e];d=b.normalize}else d=b;this.normalize=typeof d=="undefined"?true:d;this.attrExceptions=[];d=document.createElement(this.elementTagName);this.elementProperties={};for(var g in f)if(f.hasOwnProperty(g)){if(G.hasOwnProperty(g))g=
  21 +G[g];d[g]=f[g];this.elementProperties[g]=d[g];this.attrExceptions.push(g)}this.elementSortedClassName=this.elementProperties.hasOwnProperty("className")?o(this.elementProperties.className+" "+a):a;this.applyToAnyTagName=false;a=typeof c;if(a=="string")if(c=="*")this.applyToAnyTagName=true;else this.tagNames=c.toLowerCase().replace(/^\s\s*/,"").replace(/\s\s*$/,"").split(/\s*,\s*/);else if(a=="object"&&typeof c.length=="number"){this.tagNames=[];d=0;for(a=c.length;d<a;++d)if(c[d]=="*")this.applyToAnyTagName=
  22 +true;else this.tagNames.push(c[d].toLowerCase())}else this.tagNames=[this.elementTagName]}i.requireModules(["WrappedSelection","WrappedRange"]);var h=i.dom,H=function(){function a(b,c,d){return c&&d?" ":""}return function(b,c){if(b.className)b.className=b.className.replace(RegExp("(?:^|\\s)"+c+"(?:\\s|$)"),a)}}(),p;if(typeof window.getComputedStyle!="undefined")p=function(a,b){return h.getWindow(a).getComputedStyle(a,null)[b]};else if(typeof document.documentElement.currentStyle!="undefined")p=function(a,
  23 +b){return a.currentStyle[b]};else v.fail("No means of obtaining computed style properties found");var k;(function(){k=typeof document.createElement("div").isContentEditable=="boolean"?function(a){return a&&a.nodeType==1&&a.isContentEditable}:function(a){if(!a||a.nodeType!=1||a.contentEditable=="false")return false;return a.contentEditable=="true"||k(a.parentNode)}})();var M=/^inline(-block|-table)?$/i,O=/[^\r\n\t\f \u200B]/,Q=F(false),R=F(true);t.prototype={doMerge:function(){for(var a=[],b,c,d=0,
  24 +e=this.textNodes.length;d<e;++d){b=this.textNodes[d];c=b.parentNode;a[d]=b.data;if(d){c.removeChild(b);c.hasChildNodes()||c.parentNode.removeChild(c)}}return this.firstTextNode.data=a=a.join("")},getLength:function(){for(var a=this.textNodes.length,b=0;a--;)b+=this.textNodes[a].length;return b},toString:function(){for(var a=[],b=0,c=this.textNodes.length;b<c;++b)a[b]="'"+this.textNodes[b].data+"'";return"[Merge("+a.join(",")+")]"}};var P=["elementTagName","ignoreWhiteSpace","applyToEditableOnly"],
  25 +G={"class":"className"};q.prototype={elementTagName:"span",elementProperties:{},ignoreWhiteSpace:true,applyToEditableOnly:false,hasClass:function(a){return a.nodeType==1&&h.arrayContains(this.tagNames,a.tagName.toLowerCase())&&r(a,this.cssClass)},getSelfOrAncestorWithClass:function(a){for(;a;){if(this.hasClass(a,this.cssClass))return a;a=a.parentNode}return null},isModifiable:function(a){return!this.applyToEditableOnly||D(a)},isIgnorableWhiteSpaceNode:function(a){return this.ignoreWhiteSpace&&a&&
  26 +a.nodeType==3&&N(a)},postApply:function(a,b,c){for(var d=a[0],e=a[a.length-1],f=[],g,j=d,I=e,J=0,K=e.length,n,L,l=0,u=a.length;l<u;++l){n=a[l];if(L=Q(n,!c)){if(!g){g=new t(L);f.push(g)}g.textNodes.push(n);if(n===d){j=g.firstTextNode;J=j.length}if(n===e){I=g.firstTextNode;K=g.getLength()}}else g=null}if(a=R(e,!c)){if(!g){g=new t(e);f.push(g)}g.textNodes.push(a)}if(f.length){l=0;for(u=f.length;l<u;++l)f[l].doMerge();b.setStart(j,J);b.setEnd(I,K)}},createContainer:function(a){a=a.createElement(this.elementTagName);
  27 +i.util.extend(a,this.elementProperties);s(a,this.cssClass);return a},applyToTextNode:function(a){var b=a.parentNode;if(b.childNodes.length==1&&h.arrayContains(this.tagNames,b.tagName.toLowerCase()))s(b,this.cssClass);else{b=this.createContainer(h.getDocument(a));a.parentNode.insertBefore(b,a);b.appendChild(a)}},isRemovable:function(a){var b;if(b=a.tagName.toLowerCase()==this.elementTagName){if(b=o(a.className)==this.elementSortedClassName){var c;a:{b=this.elementProperties;for(c in b)if(b.hasOwnProperty(c)&&
  28 +a[c]!==b[c]){c=false;break a}c=true}b=c&&!B(a,this.attrExceptions)&&this.isModifiable(a)}b=b}return b},undoToTextNode:function(a,b,c){if(!b.containsNode(c)){a=b.cloneRange();a.selectNode(c);if(a.isPointInRange(b.endContainer,b.endOffset)){m(c,b.endContainer,b.endOffset,[b]);b.setEndAfter(c)}if(a.isPointInRange(b.startContainer,b.startOffset))c=m(c,b.startContainer,b.startOffset,[b])}this.isRemovable(c)?x(c):H(c,this.cssClass)},applyToRange:function(a){a.splitBoundaries();var b=z(a);if(b.length){for(var c,
  29 +d=0,e=b.length;d<e;++d){c=b[d];!this.isIgnorableWhiteSpaceNode(c)&&!this.getSelfOrAncestorWithClass(c)&&this.isModifiable(c)&&this.applyToTextNode(c)}a.setStart(b[0],0);c=b[b.length-1];a.setEnd(c,c.length);this.normalize&&this.postApply(b,a,false)}},applyToSelection:function(a){a=a||window;a=i.getSelection(a);var b,c=a.getAllRanges();a.removeAllRanges();for(var d=c.length;d--;){b=c[d];this.applyToRange(b);a.addRange(b)}},undoToRange:function(a){a.splitBoundaries();var b=z(a),c,d,e=b[b.length-1];if(b.length){for(var f=
  30 +0,g=b.length;f<g;++f){c=b[f];(d=this.getSelfOrAncestorWithClass(c))&&this.isModifiable(c)&&this.undoToTextNode(c,a,d);a.setStart(b[0],0);a.setEnd(e,e.length)}this.normalize&&this.postApply(b,a,true)}},undoToSelection:function(a){a=a||window;a=i.getSelection(a);var b=a.getAllRanges(),c;a.removeAllRanges();for(var d=0,e=b.length;d<e;++d){c=b[d];this.undoToRange(c);a.addRange(c)}},getTextSelectedByRange:function(a,b){var c=b.cloneRange();c.selectNodeContents(a);var d=c.intersection(b);d=d?d.toString():
  31 +"";c.detach();return d},isAppliedToRange:function(a){if(a.collapsed)return!!this.getSelfOrAncestorWithClass(a.commonAncestorContainer);else{for(var b=a.getNodes([3]),c=0,d;d=b[c++];)if(!this.isIgnorableWhiteSpaceNode(d)&&y(a,d)&&this.isModifiable(d)&&!this.getSelfOrAncestorWithClass(d))return false;return true}},isAppliedToSelection:function(a){a=a||window;a=i.getSelection(a).getAllRanges();for(var b=a.length;b--;)if(!this.isAppliedToRange(a[b]))return false;return true},toggleRange:function(a){this.isAppliedToRange(a)?
  32 +this.undoToRange(a):this.applyToRange(a)},toggleSelection:function(a){this.isAppliedToSelection(a)?this.undoToSelection(a):this.applyToSelection(a)},detach:function(){}};q.util={hasClass:r,addClass:s,removeClass:H,hasSameClasses:w,replaceWithOwnChildren:x,elementsHaveSameNonClassAttributes:A,elementHasNonClassAttributes:B,splitNodeAt:m,isEditableElement:k,isEditingHost:C,isEditable:D};i.CssClassApplier=q;i.createCssClassApplier=function(a,b,c){return new q(a,b,c)}});
0 33 \ No newline at end of file
... ...
plugins/comment_paragraph/public/rangy-serializer.js 0 → 100644
... ... @@ -0,0 +1,23 @@
  1 +/*
  2 + Serializer module for Rangy.
  3 + Serializes Ranges and Selections. An example use would be to store a user's selection on a particular page in a
  4 + cookie or local storage and restore it on the user's next visit to the same page.
  5 +
  6 + Part of Rangy, a cross-browser JavaScript range and selection library
  7 + http://code.google.com/p/rangy/
  8 +
  9 + Depends on Rangy core.
  10 +
  11 + Copyright 2012, Tim Down
  12 + Licensed under the MIT license.
  13 + Version: 1.2.3
  14 + Build date: 26 February 2012
  15 +*/
  16 +rangy.createModule("Serializer",function(g,n){function o(c,a){a=a||[];var b=c.nodeType,e=c.childNodes,d=e.length,f=[b,c.nodeName,d].join(":"),h="",k="";switch(b){case 3:h=c.nodeValue.replace(/</g,"&lt;").replace(/>/g,"&gt;");break;case 8:h="<!--"+c.nodeValue.replace(/</g,"&lt;").replace(/>/g,"&gt;")+"--\>";break;default:h="<"+f+">";k="</>";break}h&&a.push(h);for(b=0;b<d;++b)o(e[b],a);k&&a.push(k);return a}function j(c){c=o(c).join("");return u(c).toString(16)}function l(c,a,b){var e=[],d=c;for(b=
  17 +b||i.getDocument(c).documentElement;d&&d!=b;){e.push(i.getNodeIndex(d,true));d=d.parentNode}return e.join("/")+":"+a}function m(c,a,b){if(a)b||i.getDocument(a);else{b=b||document;a=b.documentElement}c=c.split(":");a=a;b=c[0]?c[0].split("/"):[];for(var e=b.length,d;e--;){d=parseInt(b[e],10);if(d<a.childNodes.length)a=a.childNodes[parseInt(b[e],10)];else throw n.createError("deserializePosition failed: node "+i.inspectNode(a)+" has no child with index "+d+", "+e);}return new i.DomPosition(a,parseInt(c[1],
  18 +10))}function p(c,a,b){b=b||g.DomRange.getRangeDocument(c).documentElement;if(!i.isAncestorOf(b,c.commonAncestorContainer,true))throw Error("serializeRange: range is not wholly contained within specified root node");c=l(c.startContainer,c.startOffset,b)+","+l(c.endContainer,c.endOffset,b);a||(c+="{"+j(b)+"}");return c}function q(c,a,b){if(a)b=b||i.getDocument(a);else{b=b||document;a=b.documentElement}c=/^([^,]+),([^,\{]+)({([^}]+)})?$/.exec(c);var e=c[4],d=j(a);if(e&&e!==j(a))throw Error("deserializeRange: checksums of serialized range root node ("+
  19 +e+") and target root node ("+d+") do not match");e=m(c[1],a,b);a=m(c[2],a,b);b=g.createRange(b);b.setStart(e.node,e.offset);b.setEnd(a.node,a.offset);return b}function r(c,a,b){if(a)b||i.getDocument(a);else{b=b||document;a=b.documentElement}c=/^([^,]+),([^,]+)({([^}]+)})?$/.exec(c)[3];return!c||c===j(a)}function s(c,a,b){c=c||g.getSelection();c=c.getAllRanges();for(var e=[],d=0,f=c.length;d<f;++d)e[d]=p(c[d],a,b);return e.join("|")}function t(c,a,b){if(a)b=b||i.getWindow(a);else{b=b||window;a=b.document.documentElement}c=
  20 +c.split("|");for(var e=g.getSelection(b),d=[],f=0,h=c.length;f<h;++f)d[f]=q(c[f],a,b.document);e.setRanges(d);return e}g.requireModules(["WrappedSelection","WrappedRange"]);if(typeof encodeURIComponent=="undefined"||typeof decodeURIComponent=="undefined")n.fail("Global object is missing encodeURIComponent and/or decodeURIComponent method");var u=function(){var c=null;return function(a){for(var b=[],e=0,d=a.length,f;e<d;++e){f=a.charCodeAt(e);if(f<128)b.push(f);else f<2048?b.push(f>>6|192,f&63|128):
  21 +b.push(f>>12|224,f>>6&63|128,f&63|128)}a=-1;if(!c){e=[];d=0;for(var h;d<256;++d){h=d;for(f=8;f--;)if((h&1)==1)h=h>>>1^3988292384;else h>>>=1;e[d]=h>>>0}c=e}e=c;d=0;for(f=b.length;d<f;++d){h=(a^b[d])&255;a=a>>>8^e[h]}return(a^-1)>>>0}}(),i=g.dom;g.serializePosition=l;g.deserializePosition=m;g.serializeRange=p;g.deserializeRange=q;g.canDeserializeRange=r;g.serializeSelection=s;g.deserializeSelection=t;g.canDeserializeSelection=function(c,a,b){var e;if(a)e=b?b.document:i.getDocument(a);else{b=b||window;
  22 +a=b.document.documentElement}c=c.split("|");b=0;for(var d=c.length;b<d;++b)if(!r(c[b],a,e))return false;return true};g.restoreSelectionFromCookie=function(c){c=c||window;var a;a:{a=c.document.cookie.split(/[;,]/);for(var b=0,e=a.length,d;b<e;++b){d=a[b].split("=");if(d[0].replace(/^\s+/,"")=="rangySerializedSelection")if(d=d[1]){a=decodeURIComponent(d.replace(/\s+$/,""));break a}}a=null}a&&t(a,c.doc)};g.saveSelectionCookie=function(c,a){c=c||window;a=typeof a=="object"?a:{};var b=a.expires?";expires="+
  23 +a.expires.toUTCString():"",e=a.path?";path="+a.path:"",d=a.domain?";domain="+a.domain:"",f=a.secure?";secure":"",h=s(g.getSelection(c));c.document.cookie=encodeURIComponent("rangySerializedSelection")+"="+encodeURIComponent(h)+b+e+d+f};g.getElementChecksum=j});
0 24 \ No newline at end of file
... ...
plugins/comment_paragraph/public/style.css 0 → 100644
... ... @@ -0,0 +1,205 @@
  1 +div.article-comments-list-more{
  2 + width: 100%;
  3 + height: 30px;
  4 + text-align: center;
  5 + font-size: 20px;
  6 + margin-bottom: 5px;
  7 +}
  8 +
  9 +.popBox_comment_paragraph {
  10 + position: absolute;
  11 + z-index: 2;
  12 + background: #cccccc;
  13 + width: 60px;
  14 + padding: 0.3em;
  15 + position: absolute;border: 1px solid gray;
  16 +}
  17 +
  18 +.span_comment_paragraph {
  19 + color: red;
  20 + font-weight: bold;
  21 +}
  22 +
  23 +.commented-area {
  24 + background-color: lightseagreen;
  25 +}
  26 +
  27 +.comment_paragraph ::selection {
  28 + background: lightseagreen; /* WebKit/Blink Browsers */
  29 +}
  30 +
  31 +.comment_paragraph ::-moz-selection {
  32 + background: lightseagreen; /* Gecko Browsers */
  33 +}
  34 +
  35 +.comment_paragraph{
  36 + padding: 0px;
  37 +}
  38 +
  39 +.triangle-right {
  40 + position:relative;
  41 + padding:15px;
  42 + margin:1em 0 3em;
  43 + color:#fff;
  44 + background:#075698; /* default background for browsers without gradient support */
  45 + /* css3 */
  46 + background:-webkit-gradient(linear, 0 0, 0 100%, from(#2e88c4), to(#075698));
  47 + background:-moz-linear-gradient(#2e88c4, #075698);
  48 + background:-o-linear-gradient(#2e88c4, #075698);
  49 + background:linear-gradient(#2e88c4, #075698);
  50 + -webkit-border-radius:10px;
  51 + -moz-border-radius:10px;
  52 + border-radius:10px;
  53 +}
  54 +
  55 +.triangle-right:after {
  56 + content: "";
  57 + position: absolute;
  58 + bottom: -20px;
  59 + left: 50px;
  60 + border-width: 20px 0 0 20px;
  61 + border-style: solid;
  62 + border-color: #075698 transparent;
  63 + display: block;
  64 + width: 0;
  65 +}
  66 +
  67 +.triangle-border {
  68 + position: relative;
  69 + padding: 15px;
  70 + margin: 1em 0 3em;
  71 + border: 5px solid #5a8f00;
  72 + color: #333;
  73 + background: #fff;
  74 + -webkit-border-radius: 10px;
  75 + -moz-border-radius: 10px;
  76 + border-radius: 10px;
  77 +}
  78 +
  79 +.side-comment .comment-created-at{display: none;}
  80 +.side-comment #comment_title{display: none;}
  81 +.side-comment .comment_title{display: none;}
  82 +.side-comment label[for="comment_title"] {display: none;}
  83 +
  84 +
  85 +
  86 +div[class^='comments_list_toggle_paragraph_'] {
  87 + border-style: solid;
  88 + border-width: 1px;
  89 + border-color: #ccc;
  90 + padding: 5px;
  91 + background-color: whitesmoke;
  92 + width: 250px;
  93 +}
  94 +
  95 +div[class^='comment-paragraph-loading-'] {
  96 + background-color: whitesmoke;
  97 + z-index: 99;
  98 + right: -230px;
  99 +}
  100 +
  101 +#content .side-comment .comment-balloon div[class^='comment-wrapper-']{
  102 + background: none;
  103 +}
  104 +
  105 +.side-comment{
  106 + z-index: 199;
  107 +}
  108 +
  109 +.side-comments-counter {
  110 + position: relative;
  111 + width: 20px;
  112 + height: 19px;
  113 + padding: 0px;
  114 + background: #b3b2d4;
  115 + -webkit-border-radius: 5px;
  116 + -moz-border-radius: 5px;
  117 + border-radius: 5px;
  118 + cursor: pointer;
  119 + z-index: 1;
  120 +}
  121 +
  122 +.side-comments-counter:hover{
  123 + color: white;
  124 +}
  125 +
  126 +.side-comments-counter:after {
  127 + content: "";
  128 + position: absolute;
  129 + bottom: -9px;
  130 + left: 8px;
  131 + border-style: solid;
  132 + border-width: 12px 5px 0;
  133 + border-color: #b3b2d4 transparent;
  134 + display: block;
  135 + width: 0;
  136 +}
  137 +
  138 +.comment-count{
  139 + position: relative;
  140 + top: 3px;
  141 +}
  142 +
  143 +.article-comment-inner {border-bottom: 1px solid #ddd;}
  144 +
  145 +#article .side-comment .comment-replies .article-comment{background: white; border: 0px; border-top: 1px solid #ddd;}
  146 +
  147 +.comment-replies .comment-from-owner.comment-content {background: none;}
  148 +
  149 +
  150 +
  151 +
  152 +.side-comment .comment-balloon-content {
  153 + margin: 0;
  154 +}
  155 +
  156 +#article .side-comment .article-comments-list {
  157 + margin: 0;
  158 +}
  159 +#article .side-comment .article-comments-list .article-comment {
  160 + margin: 0;
  161 +}
  162 +.side-comment .comment-details {
  163 + margin: 0;
  164 + padding: 0;
  165 +}
  166 +.side-comment .comment-text {
  167 + padding: 0;
  168 + width: 100%;
  169 + text-align: left;
  170 +}
  171 +.side-comment .comment-text p {
  172 + margin: 0;
  173 +}
  174 +#content #article .article-body .side-comment img {
  175 + max-width: 30px;
  176 +}
  177 +#content #article .article-body .side-comment span.comment-info {
  178 + position: relative;
  179 + top: -10px;
  180 + left: 36px;
  181 + line-height: 0;
  182 + font-weight: bold;
  183 +}
  184 +.side-comment .comment-wrapper-1 {
  185 + margin-left: 36px;
  186 +}
  187 +#article .side-comment .article-comments-list .comment-replies {
  188 + padding-left: 25px;
  189 +}
  190 +#article .side-comment .comment-picture {
  191 + width: 75%;
  192 + height: auto;
  193 +}
  194 +#article .side-comment .article-comment .comment-details > h4 {
  195 + display: none;
  196 +}
  197 +.side-comment .formlabel[for='comment_body'] {
  198 + display: none;
  199 +}
  200 +.side-comment > div > div {
  201 + background: #FFFFFF;
  202 +}
  203 +.side-comment .comment_form p {
  204 + display: none;
  205 +}
... ...
plugins/comment_paragraph/test/functional/comment_paragraph_plugin_profile_controller_test.rb 0 → 100644
... ... @@ -0,0 +1,72 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +require File.dirname(__FILE__) + '/../../controllers/profile/comment_paragraph_plugin_profile_controller'
  3 +
  4 +# Re-raise errors caught by the controller.
  5 +class CommentParagraphPluginProfileController; def rescue_action(e) raise e end; end
  6 +
  7 +class CommentParagraphPluginProfileControllerTest < ActionController::TestCase
  8 +
  9 + def setup
  10 + @controller = CommentParagraphPluginProfileController.new
  11 + @request = ActionController::TestRequest.new
  12 + @response = ActionController::TestResponse.new
  13 +
  14 + @profile = create_user('testuser').person
  15 + @article = profile.articles.build(:name => 'test')
  16 + @article.save!
  17 + end
  18 + attr_reader :article
  19 + attr_reader :profile
  20 +
  21 + should 'be able to show paragraph comments' do
  22 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  23 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  24 + assert_template 'comment_paragraph_plugin_profile/view_comments'
  25 + assert_match /comments_list_paragraph_0/, @response.body
  26 + assert_match /\"comment-count-0\", \"1\"/, @response.body
  27 + end
  28 +
  29 + should 'do not show global comments' do
  30 + fast_create(Comment, :source_id => article, :author_id => profile, :title => 'global comment', :body => 'global', :paragraph_id => nil)
  31 + fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  32 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  33 + assert_template 'comment_paragraph_plugin_profile/view_comments'
  34 + assert_match /comments_list_paragraph_0/, @response.body
  35 + assert_match /\"comment-count-0\", \"1\"/, @response.body
  36 + end
  37 +
  38 + should 'show first page comments only' do
  39 + comment1 = fast_create(Comment, :created_at => Time.now - 1.days, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'secondpage', :paragraph_id => 0)
  40 + comment2 = fast_create(Comment, :created_at => Time.now - 2.days, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'firstpage 1', :paragraph_id => 0)
  41 + comment3 = fast_create(Comment, :created_at => Time.now - 3.days, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'firstpage 2', :paragraph_id => 0)
  42 + comment4 = fast_create(Comment, :created_at => Time.now - 4.days, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'firstpage 3', :paragraph_id => 0)
  43 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  44 + assert_match /firstpage 1/, @response.body
  45 + assert_match /firstpage 2/, @response.body
  46 + assert_match /firstpage 3/, @response.body
  47 + assert_no_match /secondpage/, @response.body
  48 + end
  49 +
  50 + should 'show link to display more comments' do
  51 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  52 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  53 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  54 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'secondpage', :body => 'secondpage', :paragraph_id => 0)
  55 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  56 + assert_match /paragraph_comment_page=2/, @response.body
  57 + end
  58 +
  59 + should 'do not show link to display more comments if do not have more pages' do
  60 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  61 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  62 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  63 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  64 + assert_no_match /paragraph_comment_page/, @response.body
  65 + end
  66 +
  67 + should 'do not show link to display more comments if do not have any comments' do
  68 + xhr :get, :view_comments, :profile => @profile.identifier, :article_id => article.id, :paragraph_id => 0
  69 + assert_no_match /paragraph_comment_page/, @response.body
  70 + end
  71 +
  72 +end
... ...
plugins/comment_paragraph/test/functional/comment_paragraph_plugin_public_controller_test.rb 0 → 100644
... ... @@ -0,0 +1,37 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +require File.dirname(__FILE__) + '/../../controllers/public/comment_paragraph_plugin_public_controller'
  3 +
  4 +
  5 +# Re-raise errors caught by the controller.
  6 +class CommentParagraphPluginPublicController; def rescue_action(e) raise e end; end
  7 +
  8 +class CommentParagraphPluginPublicControllerTest < ActionController::TestCase
  9 +
  10 + def setup
  11 + @controller = CommentParagraphPluginPublicController.new
  12 + @request = ActionController::TestRequest.new
  13 + @response = ActionController::TestResponse.new
  14 +
  15 + @profile = create_user('testuser').person
  16 + @article = profile.articles.build(:name => 'test')
  17 + @article.save!
  18 + end
  19 + attr_reader :article
  20 + attr_reader :profile
  21 +
  22 +
  23 + should 'be able to return paragraph_id for a comment' do
  24 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala', :paragraph_id => 0)
  25 + cid = comment.id
  26 + xhr :get, :comment_paragraph, :id => cid
  27 + assert_match /\{\"paragraph_id\":0\}/, @response.body
  28 + end
  29 +
  30 + should 'return paragraph_id=null for a global comment' do
  31 + comment = fast_create(Comment, :source_id => article, :author_id => profile, :title => 'a comment', :body => 'lalala' )
  32 + xhr :get, :comment_paragraph, :id => comment.id
  33 + assert_match /\{\"paragraph_id\":null\}/, @response.body
  34 + end
  35 +
  36 +
  37 +end
... ...
plugins/comment_paragraph/test/functional/content_viewer_controller_test.rb 0 → 100644
... ... @@ -0,0 +1,27 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +
  3 +class ContentViewerController
  4 + append_view_path File.join(File.dirname(__FILE__) + '/../../views')
  5 + def rescue_action(e)
  6 + raise e
  7 + end
  8 +end
  9 +
  10 +class ContentViewerControllerTest < ActionController::TestCase
  11 +
  12 + def setup
  13 + @profile = fast_create(Community)
  14 + @page = fast_create(Article, :profile_id => @profile.id, :body => "<div class=\"macro\" data-macro-paragraph_id=\"0\" data-macro='comment_paragraph_plugin/allow_comment' ></div>")
  15 + @environment = Environment.default
  16 + @environment.enable_plugin(CommentParagraphPlugin)
  17 + end
  18 +
  19 + attr_reader :page
  20 +
  21 + should 'parse article body and render comment paragraph view' do
  22 + comment1 = fast_create(Comment, :paragraph_id => 0, :source_id => page.id)
  23 + get :view_page, @page.url
  24 + assert_tag 'div', :attributes => {:class => 'comment_paragraph'}
  25 + end
  26 +
  27 +end
... ...
plugins/comment_paragraph/test/test_helper.rb 0 → 100644
... ... @@ -0,0 +1 @@
  1 +require File.dirname(__FILE__) + '/../../../test/test_helper'
... ...
plugins/comment_paragraph/test/unit/allow_comment_test.rb 0 → 100644
... ... @@ -0,0 +1,26 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +
  3 +class AllowCommentTest < ActiveSupport::TestCase
  4 +
  5 + def setup
  6 + @macro = CommentParagraphPlugin::AllowComment.new
  7 + end
  8 +
  9 + attr_reader :macro
  10 +
  11 + should 'have a configuration' do
  12 + assert CommentParagraphPlugin::AllowComment.configuration
  13 + end
  14 +
  15 + should 'parse contents to include comment paragraph view' do
  16 + profile = fast_create(Community)
  17 + article = fast_create(Article, :profile_id => profile.id)
  18 + comment = fast_create(Comment, :paragraph_id => 1, :source_id => article.id)
  19 + inner_html = 'inner'
  20 + content = macro.parse({:paragraph_id => comment.paragraph_id}, inner_html, article)
  21 +
  22 + expects(:render).with({:partial => 'comment_paragraph_plugin_profile/comment_paragraph', :locals => {:paragraph_id => comment.paragraph_id, :article_id => article.id, :inner_html => inner_html, :count => 1, :profile_identifier => profile.identifier} })
  23 + instance_eval(&content)
  24 + end
  25 +
  26 +end
... ...
plugins/comment_paragraph/test/unit/article_test.rb 0 → 100644
... ... @@ -0,0 +1,26 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +require 'benchmark'
  3 +
  4 +class ArticleTest < ActiveSupport::TestCase
  5 +
  6 + def setup
  7 + profile = fast_create(Community)
  8 + @article = fast_create(Article, :profile_id => profile.id)
  9 + end
  10 +
  11 + attr_reader :article
  12 +
  13 + should 'return paragraph comments from article' do
  14 + comment1 = fast_create(Comment, :paragraph_id => 1, :source_id => article.id)
  15 + comment2 = fast_create(Comment, :paragraph_id => nil, :source_id => article.id)
  16 + assert_equal [comment1], article.paragraph_comments
  17 + end
  18 +
  19 + should 'allow save if comment paragraph macro is not removed for paragraph with comments' do
  20 + article.body = "<div class=\"macro\" data-macro-paragraph_id=0></div>"
  21 + comment1 = fast_create(Comment, :paragraph_id => 0, :source_id => article.id)
  22 + assert article.save
  23 + end
  24 +
  25 +
  26 +end
... ...
plugins/comment_paragraph/test/unit/comment_paragraph_plugin_test.rb 0 → 100644
... ... @@ -0,0 +1,28 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +
  3 +class CommentParagraphPluginTest < ActiveSupport::TestCase
  4 +
  5 + def setup
  6 + @environment = Environment.default
  7 + @plugin = CommentParagraphPlugin.new
  8 + end
  9 +
  10 + attr_reader :environment, :plugin
  11 +
  12 + should 'have a name' do
  13 + assert_not_equal Noosfero::Plugin.plugin_name, CommentParagraphPlugin::plugin_name
  14 + end
  15 +
  16 + should 'describe yourself' do
  17 + assert_not_equal Noosfero::Plugin.plugin_description, CommentParagraphPlugin::plugin_description
  18 + end
  19 +
  20 + should 'have a js file' do
  21 + assert !plugin.js_files.blank?
  22 + end
  23 +
  24 + should 'have stylesheet' do
  25 + assert plugin.stylesheet?
  26 + end
  27 +
  28 +end
... ...
plugins/comment_paragraph/test/unit/comment_test.rb 0 → 100644
... ... @@ -0,0 +1,26 @@
  1 +require File.dirname(__FILE__) + '/../test_helper'
  2 +
  3 +class CommentTest < ActiveSupport::TestCase
  4 +
  5 + def setup
  6 + profile = fast_create(Community)
  7 + @article = fast_create(Article, :profile_id => profile.id)
  8 + end
  9 +
  10 + attr_reader :article
  11 +
  12 + should 'return comments that belongs to a specified paragraph' do
  13 + comment1 = fast_create(Comment, :paragraph_id => 1, :source_id => article.id)
  14 + comment2 = fast_create(Comment, :paragraph_id => nil, :source_id => article.id)
  15 + comment3 = fast_create(Comment, :paragraph_id => 2, :source_id => article.id)
  16 + assert_equal [comment1], article.comments.in_paragraph(1)
  17 + end
  18 +
  19 + should 'return comments that do not belongs to any paragraph' do
  20 + comment1 = fast_create(Comment, :paragraph_id => 1, :source_id => article.id)
  21 + comment2 = fast_create(Comment, :paragraph_id => nil, :source_id => article.id)
  22 + comment3 = fast_create(Comment, :paragraph_id => 2, :source_id => article.id)
  23 + assert_equal [comment2], article.comments.without_paragraph
  24 + end
  25 +
  26 +end
... ...
plugins/comment_paragraph/views/comment/comment_extra.html.erb 0 → 100644
... ... @@ -0,0 +1,2 @@
  1 +<input type="hidden" value="<%= comment.comment_paragraph_selected_area%>" class="paragraph_comment_area" />
  2 +<input type="hidden" value="<%= comment.paragraph_id%>" class="paragraph_id" />
... ...
plugins/comment_paragraph/views/comment_paragraph_plugin_profile/_comment_paragraph.html.erb 0 → 100644
... ... @@ -0,0 +1,47 @@
  1 +<div class="comments">
  2 + <table border="0">
  3 + <tr>
  4 + <td width="90%">
  5 + <div class="comment_paragraph" id="comment_paragraph_<%= paragraph_id %>" data-paragraph="<%= paragraph_id %>">
  6 + <%= inner_html %>
  7 + </div>
  8 + </td>
  9 + <td>
  10 + <div align="center" class="side-comments-counter" id="side_comments_counter_<%= paragraph_id %>" data-paragraph="<%= paragraph_id %>" style="vertical-align: middle; padding-left: 3px; padding-right: 5px;">
  11 + <span id="comment-count-<%= paragraph_id %>" class='comment-count'>
  12 + <%= count %>
  13 + </span>
  14 + </div>
  15 + </td>
  16 + <td valign="top">
  17 + <div align="center">
  18 + <%=
  19 + url = { :profile => profile_identifier, :controller => 'comment_paragraph_plugin_profile', :action => 'view_comments', :paragraph_id => paragraph_id, :article_id => article_id}
  20 + link_to_remote(
  21 + image_tag("/plugins/comment_paragraph/images/comments.gif"),
  22 + {
  23 + :url => url,
  24 + :method => :post,
  25 + :condition => "!toggleParagraph(#{paragraph_id})",
  26 + :complete => "loadCompleted(#{paragraph_id})"
  27 + },
  28 + {
  29 + :id => "link_to_ajax_comments_#{paragraph_id}",
  30 + :'data-url' => url_for(url)
  31 + }
  32 + )%>
  33 + </div>
  34 + <div class="side-comment" id="side_comment_<%= paragraph_id %>" data-paragraph="<%= paragraph_id %>" style="display:none">
  35 + <div class="comment-paragraph-loading-<%= paragraph_id %>" style="position: absolute;">
  36 + <div class="comments_list_toggle_paragraph_<%= paragraph_id %>" >
  37 + <div class="article-comments-list" id="comments_list_paragraph_<%= paragraph_id %>"></div>
  38 + <div class ="article-comments-list-more" id="comments_list_paragraph_<%= paragraph_id %>_more"></div>
  39 + <div id="page-comment-form-<%= paragraph_id %>" class='post_comment_box closed'>
  40 + <%= render :partial => 'comment/comment_form', :locals => {:comment => Comment.new, :display_link => true, :cancel_triggers_hide => true, :paragraph_id => paragraph_id}%>
  41 + </div>
  42 + </div>
  43 + </div>
  44 + </div>
  45 + </td>
  46 + </tr>
  47 + </table>
... ...
plugins/comment_paragraph/views/comment_paragraph_plugin_profile/view_comments.rjs 0 → 100644
... ... @@ -0,0 +1,12 @@
  1 +if @paragraph_comment_page == 1
  2 + page.replace_html "comments_list_paragraph_#{@paragraph_id}", :partial => 'comment/comment.html.erb', :collection => @comments
  3 +else
  4 + page.insert_html :bottom, "comments_list_paragraph_#{@paragraph_id}", :partial => 'comment/comment.html.erb', :collection => @comments
  5 +end
  6 +page.replace_html "comment-count-#{@paragraph_id}", @comments_count
  7 +
  8 +if @no_more_pages
  9 + page.replace_html "comments_list_paragraph_#{@paragraph_id}_more", ""
  10 +else
  11 + page.replace_html "comments_list_paragraph_#{@paragraph_id}_more", link_to_remote(_('More'), :url => { :profile => profile.identifier, :controller => 'comment_paragraph_plugin_profile', :action => 'view_comments', :paragraph_id => @paragraph_id, :article_id => @article_id, :paragraph_comment_page => @paragraph_comment_page + 1}, :loaded => visual_effect(:highlight, "comments_list_paragraph_#{@paragraph_id}"), :method => :post, :complete => "loadCompleted(#{@paragraph_id})")
  12 +end
... ...