Skip to content

Instantly share code, notes, and snippets.

@RightHandedMonkey
Last active October 1, 2017 18:17
Show Gist options
  • Save RightHandedMonkey/03d5a7753e50e22de523f91a75036a8e to your computer and use it in GitHub Desktop.
Save RightHandedMonkey/03d5a7753e50e22de523f91a75036a8e to your computer and use it in GitHub Desktop.
Generates the CSS selector based on the item clicked on in Javascript
(function() {
var CssSelectorGenerator, root,
indexOf = [].indexOf || function(item) {
for (var i = 0, l = this.length; i < l; i++) {
if (i in this && this[i] === item) return i;
}
return -1;
};
CssSelectorGenerator = (function() {
CssSelectorGenerator.prototype.default_options = {
selectors: ['id', 'class', 'tag', 'nthchild']
};
function CssSelectorGenerator(options) {
if (options == null) {
options = {};
}
this.options = {};
this.setOptions(this.default_options);
this.setOptions(options);
}
CssSelectorGenerator.prototype.setOptions = function(options) {
var key, results, val;
if (options == null) {
options = {};
}
results = [];
for (key in options) {
val = options[key];
if (this.default_options.hasOwnProperty(key)) {
results.push(this.options[key] = val);
} else {
results.push(void 0);
}
}
return results;
};
CssSelectorGenerator.prototype.isElement = function(element) {
return !!((element != null ? element.nodeType : void 0) === 1);
};
CssSelectorGenerator.prototype.getParents = function(element) {
var current_element, result;
result = [];
if (this.isElement(element)) {
current_element = element;
while (this.isElement(current_element)) {
result.push(current_element);
current_element = current_element.parentNode;
}
}
return result;
};
CssSelectorGenerator.prototype.getTagSelector = function(element) {
return this.sanitizeItem(element.tagName.toLowerCase());
};
CssSelectorGenerator.prototype.sanitizeItem = function(item) {
var characters;
characters = (item.split('')).map(function(character) {
if (character === ':') {
return "\\" + (':'.charCodeAt(0).toString(16).toUpperCase()) + " ";
} else if (/[ !"#$%&'()*+,.\/;<=>?@\[\\\]^`{|}~]/.test(character)) {
return "\\" + character;
} else {
return escape(character).replace(/\%/g, '\\');
}
});
return characters.join('');
};
CssSelectorGenerator.prototype.getIdSelector = function(element) {
var id, sanitized_id;
id = element.getAttribute('id');
if ((id != null) && (id !== '') && !(/\s/.exec(id)) && !(/^\d/.exec(id))) {
sanitized_id = "#" + (this.sanitizeItem(id));
if (element.ownerDocument.querySelectorAll(sanitized_id).length === 1) {
return sanitized_id;
}
}
return null;
};
CssSelectorGenerator.prototype.getClassSelectors = function(element) {
var class_string, item, result;
result = [];
class_string = element.getAttribute('class');
if (class_string != null) {
class_string = class_string.replace(/\s+/g, ' ');
class_string = class_string.replace(/^\s|\s$/g, '');
if (class_string !== '') {
result = (function() {
var k, len, ref, results;
ref = class_string.split(/\s+/);
results = [];
for (k = 0, len = ref.length; k < len; k++) {
item = ref[k];
results.push("." + (this.sanitizeItem(item)));
}
return results;
}).call(this);
}
}
return result;
};
CssSelectorGenerator.prototype.getAttributeSelectors = function(element) {
var attribute, blacklist, k, len, ref, ref1, result;
result = [];
blacklist = ['id', 'class'];
ref = element.attributes;
for (k = 0, len = ref.length; k < len; k++) {
attribute = ref[k];
if (ref1 = attribute.nodeName, indexOf.call(blacklist, ref1) < 0) {
result.push("[" + attribute.nodeName + "=" + attribute.nodeValue + "]");
}
}
return result;
};
CssSelectorGenerator.prototype.getNthChildSelector = function(element) {
var counter, k, len, parent_element, sibling, siblings;
parent_element = element.parentNode;
if (parent_element != null) {
counter = 0;
siblings = parent_element.childNodes;
for (k = 0, len = siblings.length; k < len; k++) {
sibling = siblings[k];
if (this.isElement(sibling)) {
counter++;
if (sibling === element) {
return ":nth-child(" + counter + ")";
}
}
}
}
return null;
};
CssSelectorGenerator.prototype.testSelector = function(element, selector) {
var is_unique, result;
is_unique = false;
if ((selector != null) && selector !== '') {
result = element.ownerDocument.querySelectorAll(selector);
if (result.length === 1 && result[0] === element) {
is_unique = true;
}
}
return is_unique;
};
CssSelectorGenerator.prototype.getAllSelectors = function(element) {
var result;
result = {
t: null,
i: null,
c: null,
a: null,
n: null
};
if (indexOf.call(this.options.selectors, 'tag') >= 0) {
result.t = this.getTagSelector(element);
}
if (indexOf.call(this.options.selectors, 'id') >= 0) {
result.i = this.getIdSelector(element);
}
if (indexOf.call(this.options.selectors, 'class') >= 0) {
result.c = this.getClassSelectors(element);
}
if (indexOf.call(this.options.selectors, 'attribute') >= 0) {
result.a = this.getAttributeSelectors(element);
}
if (indexOf.call(this.options.selectors, 'nthchild') >= 0) {
result.n = this.getNthChildSelector(element);
}
return result;
};
CssSelectorGenerator.prototype.testUniqueness = function(element, selector) {
var found_elements, parent;
parent = element.parentNode;
found_elements = parent.querySelectorAll(selector);
return found_elements.length === 1 && found_elements[0] === element;
};
CssSelectorGenerator.prototype.testCombinations = function(element, items, tag) {
var item, k, l, len, len1, ref, ref1;
ref = this.getCombinations(items);
for (k = 0, len = ref.length; k < len; k++) {
item = ref[k];
if (this.testUniqueness(element, item)) {
return item;
}
}
if (tag != null) {
ref1 = items.map(function(item) {
return tag + item;
});
for (l = 0, len1 = ref1.length; l < len1; l++) {
item = ref1[l];
if (this.testUniqueness(element, item)) {
return item;
}
}
}
return null;
};
CssSelectorGenerator.prototype.getUniqueSelector = function(element) {
var found_selector, k, len, ref, selector_type, selectors;
selectors = this.getAllSelectors(element);
ref = this.options.selectors;
for (k = 0, len = ref.length; k < len; k++) {
selector_type = ref[k];
switch (selector_type) {
case 'id':
if (selectors.i != null) {
return selectors.i;
}
break;
case 'tag':
if (selectors.t != null) {
if (this.testUniqueness(element, selectors.t)) {
return selectors.t;
}
}
break;
case 'class':
if ((selectors.c != null) && selectors.c.length !== 0) {
found_selector = this.testCombinations(element, selectors.c, selectors.t);
if (found_selector) {
return found_selector;
}
}
break;
case 'attribute':
if ((selectors.a != null) && selectors.a.length !== 0) {
found_selector = this.testCombinations(element, selectors.a, selectors.t);
if (found_selector) {
return found_selector;
}
}
break;
case 'nthchild':
if (selectors.n != null) {
return selectors.n;
}
}
}
return '*';
};
CssSelectorGenerator.prototype.getSelector = function(element) {
var all_selectors, item, k, l, len, len1, parents, result, selector, selectors;
all_selectors = [];
parents = this.getParents(element);
for (k = 0, len = parents.length; k < len; k++) {
item = parents[k];
selector = this.getUniqueSelector(item);
if (selector != null) {
all_selectors.push(selector);
}
}
selectors = [];
for (l = 0, len1 = all_selectors.length; l < len1; l++) {
item = all_selectors[l];
selectors.unshift(item);
result = selectors.join(' > ');
if (this.testSelector(element, result)) {
return result;
}
}
return null;
};
CssSelectorGenerator.prototype.getCombinations = function(items) {
var i, j, k, l, ref, ref1, result;
if (items == null) {
items = [];
}
result = [
[]
];
for (i = k = 0, ref = items.length - 1; 0 <= ref ? k <= ref : k >= ref; i = 0 <= ref ? ++k : --k) {
for (j = l = 0, ref1 = result.length - 1; 0 <= ref1 ? l <= ref1 : l >= ref1; j = 0 <= ref1 ? ++l : --l) {
result.push(result[j].concat(items[i]));
}
}
result.shift();
result = result.sort(function(a, b) {
return a.length - b.length;
});
result = result.map(function(item) {
return item.join('');
});
return result;
};
return CssSelectorGenerator;
})();
if (typeof define !== "undefined" && define !== null ? define.amd : void 0) {
define([], function() {
return CssSelectorGenerator;
});
} else {
root = typeof exports !== "undefined" && exports !== null ? exports : this;
root.CssSelectorGenerator = CssSelectorGenerator;
}
}).call(this);
function getXPathForElement(el, xml) {
var xpath = '';
var pos, tempitem2;
while (el !== xml.documentElement) {
pos = 0;
tempitem2 = el;
while (tempitem2) {
if (tempitem2.nodeType === 1 && tempitem2.nodeName === el.nodeName) { // If it is ELEMENT_NODE of the same name
pos += 1;
}
tempitem2 = tempitem2.previousSibling;
}
xpath = "*[name()='" + el.nodeName + "' and namespace-uri()='" + (el.namespaceURI === null ? '' : el.namespaceURI) + "'][" + pos + ']' + '/' + xpath;
el = el.parentNode;
}
xpath = '/*' + "[name()='" + xml.documentElement.nodeName + "' and namespace-uri()='" + (el.namespaceURI === null ? '' : el.namespaceURI) + "']" + '/' + xpath;
xpath = xpath.replace(/\/$/, '');
return xpath;
}
function context(event) {
event.preventDefault();
console.log(event.target);
var sel_gen = new CssSelectorGenerator();
var sel = sel_gen.getSelector(event.target);
var xpath = getXPathForElement(event.target, document);
alert("sel: " + sel+"\r\nContents: "+event.target.innerHTML);
}
//Capture clicking on each item
document.body.onclick = function(e){
context(e);
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment