Added elastic for text area and mark/reply text feature.
This commit is contained in:
parent
f2f098d496
commit
3c38f222c2
4 changed files with 201 additions and 3 deletions
|
@ -76,6 +76,12 @@ class Index extends App.Controller
|
||||||
form: @formGen( model: { configure_attributes: configure_attributes, className: 'create' } ),
|
form: @formGen( model: { configure_attributes: configure_attributes, className: 'create' } ),
|
||||||
)
|
)
|
||||||
|
|
||||||
|
# add elastic to textarea
|
||||||
|
@el.find('textarea').elastic()
|
||||||
|
|
||||||
|
# update textarea size
|
||||||
|
@el.find('textarea').trigger('change')
|
||||||
|
|
||||||
# start customer info controller
|
# start customer info controller
|
||||||
if defaults['customer_id']
|
if defaults['customer_id']
|
||||||
$('#create_customer_id').val( defaults['customer_id'] )
|
$('#create_customer_id').val( defaults['customer_id'] )
|
||||||
|
|
|
@ -91,14 +91,14 @@ class Index extends App.Controller
|
||||||
|
|
||||||
# preview mode
|
# preview mode
|
||||||
if preview_mode
|
if preview_mode
|
||||||
article['html'] = article['html'].replace /^----SEEMORE----\n/m, (match) =>
|
article['html'] = article['html'].replace /^\n----SEEMORE----\n/m, (match) =>
|
||||||
notify + '<div class="hide">'
|
notify + '<div class="hide">'
|
||||||
article['html'] = article['html'] + '</div>'
|
article['html'] = article['html'] + '</div>'
|
||||||
|
|
||||||
# hide signatures and so on
|
# hide signatures and so on
|
||||||
else
|
else
|
||||||
@article_changed = false
|
@article_changed = false
|
||||||
article['html'] = article['html'].replace /^(--|__)/m, (match) =>
|
article['html'] = article['html'].replace /^\n(--|__)/m, (match) =>
|
||||||
@article_changed = true
|
@article_changed = true
|
||||||
notify + '<div class="hide">' + match
|
notify + '<div class="hide">' + match
|
||||||
if @article_changed
|
if @article_changed
|
||||||
|
@ -173,6 +173,8 @@ class Index extends App.Controller
|
||||||
form_article: form_article,
|
form_article: form_article,
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@el.find('textarea').elastic()
|
||||||
|
|
||||||
@userPopups()
|
@userPopups()
|
||||||
|
|
||||||
# start customer info controller
|
# start customer info controller
|
||||||
|
@ -302,6 +304,18 @@ class Index extends App.Controller
|
||||||
@el.find('[name="to"]').val(article.from)
|
@el.find('[name="to"]').val(article.from)
|
||||||
# @log 'reply ', article, @el.find('[name="to"]')
|
# @log 'reply ', article, @el.find('[name="to"]')
|
||||||
|
|
||||||
|
# add quoted text if needed
|
||||||
|
if window.Session['UISeletion']
|
||||||
|
body = @el.find('[name="body"]').val() || ''
|
||||||
|
selection = window.Session['UISeletion']
|
||||||
|
selection = selection.replace /^(.*)$/mg, (match) =>
|
||||||
|
'> ' + match
|
||||||
|
body = body + selection
|
||||||
|
@el.find('[name="body"]').val(body)
|
||||||
|
|
||||||
|
# update textarea size
|
||||||
|
@el.find('[name="body"]').trigger('change')
|
||||||
|
|
||||||
update: (e) =>
|
update: (e) =>
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
params = @formParam(e.target)
|
params = @formParam(e.target)
|
||||||
|
|
|
@ -18,6 +18,7 @@
|
||||||
#= require ./lib/jquery.noty.js
|
#= require ./lib/jquery.noty.js
|
||||||
#= require ./lib/waypoints.js
|
#= require ./lib/waypoints.js
|
||||||
#= require ./lib/fileuploader.js
|
#= require ./lib/fileuploader.js
|
||||||
|
#= require ./lib/jquery.elastic.source.js
|
||||||
|
|
||||||
#not_used= require_tree ./lib
|
#not_used= require_tree ./lib
|
||||||
#= require_self
|
#= require_self
|
||||||
|
@ -197,6 +198,11 @@ class App.Run extends Spine.Controller
|
||||||
# start content
|
# start content
|
||||||
new App.Content( el: @el.find('#content') );
|
new App.Content( el: @el.find('#content') );
|
||||||
|
|
||||||
|
# bind to fill selected text into
|
||||||
|
$(@el).bind('mouseup', =>
|
||||||
|
window.Session['UISeletion'] = @getSelected() + ''
|
||||||
|
)
|
||||||
|
|
||||||
@ws = new WebSocket("ws://localhost:3001/");
|
@ws = new WebSocket("ws://localhost:3001/");
|
||||||
|
|
||||||
# Set event handlers.
|
# Set event handlers.
|
||||||
|
@ -218,6 +224,16 @@ class App.Run extends Spine.Controller
|
||||||
@ws.onerror = ->
|
@ws.onerror = ->
|
||||||
console.log("onerror")
|
console.log("onerror")
|
||||||
|
|
||||||
|
getSelected: ->
|
||||||
|
text = '';
|
||||||
|
if window.getSelection
|
||||||
|
text = window.getSelection()
|
||||||
|
else if document.getSelection
|
||||||
|
text = document.getSelection()
|
||||||
|
else if document.selection
|
||||||
|
text = document.selection.createRange().text
|
||||||
|
text
|
||||||
|
|
||||||
class App.Content extends Spine.Controller
|
class App.Content extends Spine.Controller
|
||||||
className: 'container'
|
className: 'container'
|
||||||
|
|
||||||
|
|
162
app/assets/javascripts/app/lib/jquery.elastic.source.js
Normal file
162
app/assets/javascripts/app/lib/jquery.elastic.source.js
Normal file
|
@ -0,0 +1,162 @@
|
||||||
|
/**
|
||||||
|
* @name Elastic
|
||||||
|
* @descripton Elastic is jQuery plugin that grow and shrink your textareas automatically
|
||||||
|
* @version 1.6.11
|
||||||
|
* @requires jQuery 1.2.6+
|
||||||
|
*
|
||||||
|
* @author Jan Jarfalk
|
||||||
|
* @author-email jan.jarfalk@unwrongest.com
|
||||||
|
* @author-website http://www.unwrongest.com
|
||||||
|
*
|
||||||
|
* @licence MIT License - http://www.opensource.org/licenses/mit-license.php
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function($){
|
||||||
|
jQuery.fn.extend({
|
||||||
|
elastic: function() {
|
||||||
|
|
||||||
|
// We will create a div clone of the textarea
|
||||||
|
// by copying these attributes from the textarea to the div.
|
||||||
|
var mimics = [
|
||||||
|
'paddingTop',
|
||||||
|
'paddingRight',
|
||||||
|
'paddingBottom',
|
||||||
|
'paddingLeft',
|
||||||
|
'fontSize',
|
||||||
|
'lineHeight',
|
||||||
|
'fontFamily',
|
||||||
|
'width',
|
||||||
|
'fontWeight',
|
||||||
|
'border-top-width',
|
||||||
|
'border-right-width',
|
||||||
|
'border-bottom-width',
|
||||||
|
'border-left-width',
|
||||||
|
'borderTopStyle',
|
||||||
|
'borderTopColor',
|
||||||
|
'borderRightStyle',
|
||||||
|
'borderRightColor',
|
||||||
|
'borderBottomStyle',
|
||||||
|
'borderBottomColor',
|
||||||
|
'borderLeftStyle',
|
||||||
|
'borderLeftColor'
|
||||||
|
];
|
||||||
|
|
||||||
|
return this.each( function() {
|
||||||
|
|
||||||
|
// Elastic only works on textareas
|
||||||
|
if ( this.type !== 'textarea' ) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var $textarea = jQuery(this),
|
||||||
|
$twin = jQuery('<div />').css({
|
||||||
|
'position' : 'absolute',
|
||||||
|
'display' : 'none',
|
||||||
|
'word-wrap' : 'break-word',
|
||||||
|
'white-space' :'pre-wrap'
|
||||||
|
}),
|
||||||
|
lineHeight = parseInt($textarea.css('line-height'),10) || parseInt($textarea.css('font-size'),'10'),
|
||||||
|
minheight = parseInt($textarea.css('height'),10) || lineHeight*3,
|
||||||
|
maxheight = parseInt($textarea.css('max-height'),10) || Number.MAX_VALUE,
|
||||||
|
goalheight = 0;
|
||||||
|
|
||||||
|
// Opera returns max-height of -1 if not set
|
||||||
|
if (maxheight < 0) { maxheight = Number.MAX_VALUE; }
|
||||||
|
|
||||||
|
// Append the twin to the DOM
|
||||||
|
// We are going to meassure the height of this, not the textarea.
|
||||||
|
$twin.appendTo($textarea.parent());
|
||||||
|
|
||||||
|
// Copy the essential styles (mimics) from the textarea to the twin
|
||||||
|
var i = mimics.length;
|
||||||
|
while(i--){
|
||||||
|
$twin.css(mimics[i].toString(),$textarea.css(mimics[i].toString()));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Updates the width of the twin. (solution for textareas with widths in percent)
|
||||||
|
function setTwinWidth(){
|
||||||
|
var curatedWidth = Math.floor(parseInt($textarea.width(),10));
|
||||||
|
if($twin.width() !== curatedWidth){
|
||||||
|
$twin.css({'width': curatedWidth + 'px'});
|
||||||
|
|
||||||
|
// Update height of textarea
|
||||||
|
update(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sets a given height and overflow state on the textarea
|
||||||
|
function setHeightAndOverflow(height, overflow){
|
||||||
|
|
||||||
|
var curratedHeight = Math.floor(parseInt(height,10));
|
||||||
|
if($textarea.height() !== curratedHeight){
|
||||||
|
$textarea.css({'height': curratedHeight + 'px','overflow':overflow});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// This function will update the height of the textarea if necessary
|
||||||
|
function update(forced) {
|
||||||
|
|
||||||
|
// Get curated content from the textarea.
|
||||||
|
var textareaContent = $textarea.val().replace(/&/g,'&').replace(/ {2}/g, ' ').replace(/<|>/g, '>').replace(/\n/g, '<br />');
|
||||||
|
|
||||||
|
// Compare curated content with curated twin.
|
||||||
|
var twinContent = $twin.html().replace(/<br>/ig,'<br />');
|
||||||
|
|
||||||
|
if(forced || textareaContent+' ' !== twinContent){
|
||||||
|
|
||||||
|
// Add an extra white space so new rows are added when you are at the end of a row.
|
||||||
|
$twin.html(textareaContent+' ');
|
||||||
|
|
||||||
|
// Change textarea height if twin plus the height of one line differs more than 3 pixel from textarea height
|
||||||
|
if(Math.abs($twin.height() + lineHeight - $textarea.height()) > 3){
|
||||||
|
|
||||||
|
var goalheight = $twin.height()+lineHeight;
|
||||||
|
if(goalheight >= maxheight) {
|
||||||
|
setHeightAndOverflow(maxheight,'auto');
|
||||||
|
} else if(goalheight <= minheight) {
|
||||||
|
setHeightAndOverflow(minheight,'hidden');
|
||||||
|
} else {
|
||||||
|
setHeightAndOverflow(goalheight,'hidden');
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
// Hide scrollbars
|
||||||
|
$textarea.css({'overflow':'hidden'});
|
||||||
|
|
||||||
|
// Update textarea size on keyup, change, cut and paste
|
||||||
|
$textarea.bind('keyup change cut paste', function(){
|
||||||
|
update();
|
||||||
|
});
|
||||||
|
|
||||||
|
// Update width of twin if browser or textarea is resized (solution for textareas with widths in percent)
|
||||||
|
$(window).bind('resize', setTwinWidth);
|
||||||
|
$textarea.bind('resize', setTwinWidth);
|
||||||
|
$textarea.bind('update', update);
|
||||||
|
|
||||||
|
// Compact textarea on blur
|
||||||
|
$textarea.bind('blur',function(){
|
||||||
|
if($twin.height() < maxheight){
|
||||||
|
if($twin.height() > minheight) {
|
||||||
|
$textarea.height($twin.height());
|
||||||
|
} else {
|
||||||
|
$textarea.height(minheight);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// And this line is to catch the browser paste event
|
||||||
|
$textarea.bind('input paste',function(e){ setTimeout( update, 250); });
|
||||||
|
|
||||||
|
// Run update once when elastic is initialized
|
||||||
|
update();
|
||||||
|
|
||||||
|
});
|
||||||
|
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})(jQuery);
|
Loading…
Reference in a new issue