Not sure if the title is well chosen...
I am trying to simulate text-selection in HTML/JS/CSS to get rid of the action bubble on mobile device when truly selecting t
A few thoughts:
I can't see how wrapping everything in a <span>
is a viable solution. What happens when you have something like this?
<p>The <b>important terms</b> will be bolded in this text</p>
When you select The important
, you'll have some tag mayhem where the <span>
tag's body wants to overlap the <b>
tag's body.
I think a better solution would be to have multiple selection <span>
tags with the same class. Each time an HTML tag is encountered, you skip over the tag and create a new selection <span>
. Then, when you reposition your carets, you simply do this:
$(".selection:first").prepend(startCaret);
$(".selection:last").append(endCaret);
I played around with this idea and it was working fine for me. If you want to get really fancy, you could try and merge selection <span>
s when you find a closing tag after you've encountered its corresponding start tag, but that would be a lot of work.
Your carets aren't moving correctly because you specify a style of absolute positioning. Change that to relative and give each caret a body of
. You'll have to play around with a few other positioning CSS settings to get it to look just right, but it moves as you'd expect with those changes (example here).
As I was playing with some ideas for this, I tried doing absolute positioning on the carets so they didn't have to be moved around inside the selection <span>
. That was a mistake. I ran into problems with paragraphs that overflowed to a new line without any markup. Stay with your idea of having the carets inside of your selection <span>
.
Lastly, a few thought questions:
<p>
tag, don't allow selection to spill outside of that tag to the next <p>
or whatever follows)<table>
, <script>
, <select>
, etc)I think this idea is pretty cool, but it could be a pretty massive project. I'll post some of my prototypes that I tried when I get them a bit more polished.
To cover Rusty's issue:
Split the tags out first:
var textBlock = textContainer.html(),
taglessArray = textBlock.split(/<[^>]*>/),
arrayOfTags = textBlock.match(/<[^>]*>/g),
tagsFirst = (textBlock[0] === '<') ? true : false, //parens for legibility only
//tagsFirst is a switch that tells us which to start splicing back in later
Now let's wrap all of our non-whitespace in span tags we can use
var i = taglessArray.length;
while(i--){
taglessArray[i].replace(/([^\s]*)/g,'<span class="selectableWord">$1</span>');
}
//not 100% sure I got that right - haven't tested - but the idea is , wrap all non
//whitespace/word boundary blocks in 'selectableWord' classed span tags
Now stick 'em back together. Original text formatting tags should retain position. Obviously 'selectableWord' classed spans should avoid layout/formatting impact
var addToArray, adderArray;
if(tagsFirst){ addToArray = arrayOfTags; adderArray = taglessArray; }
else { addToArray = taglessArray; adderArray = arrayOfTags; }
var oLen, //used to retain 'original length'
i2 = oLen = (tagsFirst.length + arrayOfTags.length),
rejoinArray = [];
while(i2--){ //not 100% sure I got this right - hope you get the general idea
var currentKey = oLen - 1 - i2;
//if first or alternating from first 0,2,4,etc...
if(currentKey === 0 || ( currentKey % 2 === 0) {
rejoinArray[currentKey] = addToArray.shift(); //remove first element and return
}
//should be keys 1,3,5,etc...
else {
rejoinArray[currentKey] = adderArray.shift();
}
}
All of the above of course is hugely untested and probably buggy/wrong but I think the basic idea is pretty cool. Rip the tags out. Wrap the array of untagged words in spans you can use. Then splice all the silly tags back in. They should still wrap all the same words. This assumes basic HTML integrity. Your text blocks only contain inline-display tags that would go around words or sets of words. Improper nesting might not be an issue unless there are improperly nested spans.