Javascript-форум (https://javascript.ru/forum/)
-   Events/DOM/Window (https://javascript.ru/forum/events/)
-   -   Как сделать цитирование? (https://javascript.ru/forum/events/4217-kak-sdelat-citirovanie.html)

`p r o x y 09.07.2009 06:03

окей, а как вот такой вариант:

// When a generated page (in PHP, Perl or something else)
// need add to each post (DIV, TABLE, TR, TD): onmouseover="qq.Nick='user nick'"
// Exsample for PHP: echo '<tr class="bugnote" id="'.$postId.'" onmouseover="qq.Nick=\''.$userNick.'\'">';

var qq = {

	// QuickQuote Settings
	LinkShowTime	: 3000,
	LinkId  		: 'quickQuote',
	LinkStyle		: 'z-index:1000; cursor:pointer; position:absolute; visibility:hidden',
	LinkText		: '<b>Цитировать</b>',
	TextAreaId   	: 'bugnote_text', // id or name
	// <--------------------
	
	Link : '', TextArea : '', Text : '', Nick : '', TimerId : '',
	
	Install		: function(){
		document.write(	'<div onmousedown="qq.InsertText()" onmouseout="qq.TimerStart()" onmouseover="qq.TimerStop()" \
						class="button" id="'+qq.LinkId+'" style="'+qq.LinkStyle+'">'+qq.LinkText+'</div>');
		
		qq.Link = document.getElementById(qq.LinkId);
		
		document.onclick   = qq.GetSelText;
		document.onmouseup = qq.LinkShow;
	},
	
	TimerStart	: function(){ qq.TimerStop(); qq.TimerId = setTimeout(qq.LinkHide, qq.LinkShowTime); },
	TimerStop	: function(){ clearTimeout(qq.TimerId); },
	LinkHide	: function(){ qq.Link.style.visibility = 'hidden'; },

	LinkShow 	: function(event){
		qq.GetSelText();
		if (qq.Text == '') return;
		
		var mc = event || window.event;
		if (mc.pageX || mc.pageY){
			mc.X = mc.pageX;
			mc.Y = mc.pageY;
		}else if (mc.clientX || mc.clientY){
			mc.X = mc.clientX + document.body.scrollLeft + document.documentElement.scrollLeft;
			mc.Y = mc.clientY + document.body.scrollTop  + document.documentElement.scrollTop;
		}
		qq.Link.style.left = (mc.X-35)+'px';
		qq.Link.style.top  = (mc.Y+11)+'px';
		qq.Link.style.visibility = 'visible';
		
		qq.TimerStart();
	},
	
	GetSelText	: function(){
		qq.Text = getSelText();
		qq.Text.replace(/(\r?\n\s*){2,}/gi,'\r\n').replace(/^\s+|\s+$/gi,'').replace(/(\ |\t)+/gi,' ');
		if (qq.Text == '') qq.LinkHide();
	},
	
	InsertText	: function(){
		qq.LinkHide();
		if (qq.TextArea == '' || qq.TextArea == null){
			qq.TextArea = document.getElementById(qq.TextAreaId);
			if (qq.TextArea == null) qq.TextArea = document.getElementsByName(qq.TextAreaId)[0];
		}
		if (qq.TextArea == null) return;
		
		if (navigator.appName == 'Opera') qq.Text = '[q]'+qq.Nick+': ' + qq.Text + '[/q]\r\n';
		else qq.Text = '[q]'+qq.Nick+': ' + qq.Text + '[/q]\n';
		
		insertText(qq.TextArea, qq.Text);
	}
};

// Common functions
function saveCaretPos(event){
    event = event || window.event;
    var obj = event.target || event.srcElement;
    if (typeof(document.selection) != 'undefined' && typeof(document.selection.createRange) != 'undefined')
    obj.caretPos = document.selection.createRange().duplicate();
}
function getSelText(){
	if (window.getSelection && !window.opera) 	var selText = window.getSelection(); // ff
	else if (document.getSelection) 			var selText = document.getSelection(); // opera
	else if (document.selection) 				var selText = document.selection.createRange().text; // ie
	if (!selText) selText = '';
	return selText.toString();
}
function insertText(obj, text){

	var scrollTop, scrollLeft;
	if (obj.type == 'TEXTAREA' && obj.scrollTop){
		scrollTop = obj.scrollTop;
		scrollLeft = obj.scrollLeft;
	}

	if (window.getSelection || document.getSelection) var caretPos = obj.selectionStart + text.length;
	
	if (obj && obj.caretPos) 
		obj.caretPos.text = text;
	else if (obj && obj.selectionStart+1 && obj.selectionEnd+1)
		obj.value = obj.value.substring(0, obj.selectionStart) + text + obj.value.substring(obj.selectionEnd, obj.value.length);
	else if (obj)
		obj.value += text;
	
	if (window.getSelection || document.getSelection) obj.setSelectionRange(caretPos, caretPos);
	
	if (typeof scrollTop != 'undefined'){
		textObj.scrollTop = scrollTop;
		textObj.scrollLeft = scrollLeft;
	}
}
// <--------------------

// Install QuickQuote
qq.Install();

document.body.onload = function(){
	qq.TextArea = document.getElementById(qq.TextAreaId);
	if (qq.TextArea == null) qq.TextArea = document.getElementsByName(qq.TextAreaId)[0];
	if (qq.TextArea == null) return;
	
	qq.TextArea.onselect = saveCaretPos;
	qq.TextArea.onclick  = saveCaretPos;
	qq.TextArea.onkeyup  = saveCaretPos;
}
// <--------------------



Осталось только разобраться с прокруткой в textarea.
Demo страница

x-yuri 09.07.2009 06:25

можно и так. Минус: рассчитано на одну textarea (хотя мне не приходит в голову ситуация, когда нужно больше). Минус является следствием того, что ты по сути спрятал все в один namespace. Если бы это было реализовано в виде "класса"... но можно и так

`p r o x y 09.07.2009 07:12

Цитата:

Минус: рассчитано на одну textarea
эту разницу я понимаю, именно на это и рассчитываю.

А вот следующий вариант будет на "классе"....

Вопрос: а как проверить, назначен ли объекту обработчик событий: onselect / onclick / onkeyup ?

B~Vladi 09.07.2009 10:21

obj.onclick

Возвратит то, что было назначено объекту через свойство:
obj.onclick=function(){
   alert('');
}

или null.

Что добавленно через addEventListener или attachEvent узнать нельзя, разве что только самому сохранять.

`p r o x y 09.07.2009 12:10

ни как не могу победить - установку позиции корректки в textarea для ie.
Пока только через:

qq.TextArea.onselect = saveCaretPos;
qq.TextArea.onclick  = saveCaretPos;
qq.TextArea.onkeyup  = saveCaretPos;

function saveCaretPos(event){
    event = event || window.event;
    var obj = event.target || event.srcElement;
    if (typeof(document.selection) != 'undefined' && typeof(document.selection.createRange) != 'undefined')
    obj.caretPos = document.selection.createRange().duplicate();
}


Хотя для Opera и FF отлично работает:
obj.setSelectionRange(caretPos, caretPos);


Может для ie есть вариант подобный setSelectionRange?

B~Vladi 09.07.2009 14:19

Для ИЕ есть метод window.getSelection().
Возвращает выделенный текст. Пока только знаю метод removeAllRanges(), который снимает выделение. По поводу других пользуемся конструкцией var in и гуглом.

Octane 09.07.2009 14:25

Можно создать новый TextRange и сдвинуть его границы в нужную позицию

`p r o x y 09.07.2009 16:17

B~Vladi,
дело в том, что getSelection() и removeAllRanges() это только в Opera, FF и т.д. - в IE свое направление на эту тему: selection ............(

Octane,
а можешь привести строку кода для примера, не получается сообразить как выполнить...делаю так:

var obj = document.getElementById('textarea_id');
obj.selection.createRange();

а какой метод дальше? move ? ...ни как не разберусь - все в ошибках постоянно (читаю статью, что ты выше постил).
Кстати, если устанавливать позицию корретки, то как для IE определить в какой позиции она уже стоит?

Octane 09.07.2009 16:21

Как-то так:
var obj = document.getElementById('textarea_id');
var range = obj.selection.createRange();
range.moveStart('character', selStart);
range.moveEnd('character', selEnd);
range.select();


Подробнее здесь: Range, TextRange и Selection

B~Vladi 09.07.2009 17:00

Цитата:

Сообщение от `p r o x y
getSelection() и removeAllRanges() это только в Opera, FF и т.д.

Аха, блин, перепутал немного...


Часовой пояс GMT +3, время: 02:21.