Skip to content

Instantly share code, notes, and snippets.

@NerdyDeedsLLC
Last active December 23, 2018 05:11
Show Gist options
  • Save NerdyDeedsLLC/65255115424e87472ffa5dccd2f602be to your computer and use it in GitHub Desktop.
Save NerdyDeedsLLC/65255115424e87472ffa5dccd2f602be to your computer and use it in GitHub Desktop.
CSS/JS Coordinate-Targeting Reticle (with visible pixel coordinates) - works by passing X/Y coordinates, specifying target element, or onClick anywhere within the window

Reticle.js

Displays a crosshair reticle (along with its corresponding coordinate pair) on screen. Useful for verifying object positioning, cursor position, locating X/Y coordinates of an object for callouts, etc.

Sample Red Reticle

Implementation/Usage Modes

Reticle currently has three modes:

1. Direct Placement Mode (Red Reticle)

A red reticle will be placed such that it's center lies precisely at the on-screen coordiantes specified. That is to say, it will entirely ignore your current scroll position. This is due to the reticle being absolutely-positioned. A fixed position reticle will respect the true coordinates, but will bollocks onClick mode. Thus, if you're so inclined to extend this out, keep that in mind.

Code Usage

To enable the reticle at the specified coordinates, simply call its place(x,y) method, where x and y are equal to the desired on-screen coordinates.

  xCoordinate = 123;
  yCoordinate = 456;
  reticle.place(xCoordinate, yCoordinate);

2. Screen-Click Placement Mode (Green Reticle)

A green reticle will be placed such that it's center lies precisely at the on-screen coordiantes where the user last clicked. It will not inhibit functionality of whatever was clicked (i.e. links will still link, buttons will still button, snozzberries... well, you get it). It will also update with every click, so you may want to jot down whatever you're after before clicking again. Extending it out such that the values were push()'d into an array with each click would be a matter of about 3 lines of code, if it's important to you.

Code Usage

  reticle.enableOnClick(); // That's it. Seriously.

3. Object Target Mode (Red Reticle)

A red reticle will appear at the top-left of the object specified.

Code Usage

Objects may be specified either:

A. With a QuerySelector string (in which the first matching object will be presupposed to be the target):

  reticle.target('#MyObjectsID.AssumingItHasThisClass');

B. With an HTMLElement object itself:

  let myObj = document.getElementbyId('MyObjectsID');
  reticle.target(myObj);

API

Public Methods

(Really, considering I just slapped the whole damn thing into a JSON object, they're ALL public, but if I WERE to go the extra couple steps, these would the ones I intended for humans to work with):

reticle.place(x, y)

Places the center of the reticle at the specified coordinates

Parameters:
Parameter Type Purpose Required Default Value
x Integer X-Coordinate (on-screen; agnostic of scroll position) to place reticle at Yes 0
y Integer Y-Coordinate (on-screen; agnostic of scroll position) to place reticle at Yes 0
Example Usage
  reticle.place(123,456); // Places center of reticle at coordinates X: 123, Y: 456)

reticle.target(objQS)

Places the center of the reticle at the top-left-most position of the first object that matches the provided QuerySelector (or HTMLElement, if passed instead).

Parameters:
Parameter Type Purpose Required Default Value
objQS String The QuerySelector that matches the target object, OR Yes -
Object An HTMLElement object (functionally just bypasses the targeting method to directly focus on the object
Example Usage
  reticle.target('a.nav-link'); // Places reticle at top-left of first located anchor with class of 'nav-link'

  // -OR-

  let footerObj = document.getElemenstByTagName("footer")[0]; // Targets the first <footer> tag on the page
  reticle.target(footerObj);                                  // Places reticle at top-left of targeted <footer>

reticle.enableOnClick() and reticle.disableOnClick()

Switches reticle to and from window onClick mode (enableOnClick and disableOnClick, respectively): reticle will switch to green, and reticle's center will appear at any location clicked by the user. Note that it will NOT inhibit functionality of the clicked point, it will simply drop the reticle there. This is worth noting, as, should the object clicked be a link, the browser will still follow the href, and thus the reticle will be visible for but a fraction of a second.

Parameters:

None

Example Usage
  reticle.enableOnClick();  // Green reticle will be centered on every location clicked by the user
  reticle.disableOnClick(); // Removes enableOnClick's functionality, reticle color returns to red, and is hidden until next used.

reticle.color(clr)

Changes the color of the reticle and its accompanying text, by altering its CSS --reticle-color property. Color formats accepted are:

  • Hex (#FF0000)
  • Short Hex (#F00)
  • RGB (rgb(0, 255, 255)
  • RGBA (rgba(0, 255, 255, 0.5)
  • Color Name ('red')
Parameters:
Parameter Type Purpose Required Default Value
clr String The color you wish to change the reticle and text to, in one of the formats listed above Yes 'red'
Example Usage
  reticle.color('red'); // Changes reticle and text color to red (#F00/#FF0000)
  reticle.color('rgba(0, 255, 0, 0.5)'); changes reticle color to 50%-opaque green (#0F0/#00FF00)
  reticle.color('#0000FF'); // Changes reticle and text color to blue (#00F)

reticle.show() and reticle.hide()

Simply toggles the reticle's css display property, rendering it visible (show()) or hidden (hide()), respectively.

Parameters:

None

Example Usage
  reticle.show(); // Reticle becomes visible at last location and color (default: 0,0 and red)
  reticle.hide(); // Reticle becomes hidden (though is not destroyed; e.g. can still be manipulated)
.reticle {
position:absolute;
border:2px solid red;
border-bottom:0 none;
border-right:0 none;
width: 10px;
height: 10px;
z-index: 1000000;
font-size:10px;
font-weight:bold;
color:red;
line-height:10px;
padding:5px;
top:10rem;
left:10rem;
}
.reticle::before {
content:'';
position:absolute;
border:2px solid red;
border-top:0 none;
border-left:0 none;
width: 10px;
height: 10px;
z-index: 1000000;
transform: translate(calc(-150% + 1px), calc(-150% + 1px));
}
.reticle::after {
content:'';
position:absolute;
border:1px solid red;
border-radius: 50%;
width: 10px;
height: 10px;
z-index: 1000000;
transform: translate(-100%, calc(-10px + -100%));
}
var reticle = {
retObj: null, // VAR: retObj Storage object for the reticle HTML object
retColor:'red', // VAR: retColor Default color of the reticle and its text
ret: function() { // FN: ret() Creates the reticle HTML object (if not present), and
this.retObj = document.querySelector(".reticle") || null; // returns it (be it newly-created or already-extant)
if(this.retObj !== null){ // params: NONE
return(this.retObj);
}else{
let clrVar = `style="--reticle-color:${this.retColor}"`; // TODO: Modify the reticle so it's position is dictated by CSS variables
document.body.innerHTML += `<span class="reticle" ${clrVar}>(0,0)</span>`;
this.retObj = document.querySelector(".reticle");
}
return(this.retObj);
},
place: function(x,y) { // FN: place(x,y) Drops the reticle at the X/Y coordinates specified.
let rO = this.ret(); // Note these are ON-SCREEN coordinates, and don't account for
rO.style.top = y + 'px'; // page scrolling (due to absolutely-positioning the reticle)
rO.style.left = x + 'px'; // params: x - the X-coordinate we want to place the reticle center at
rO.innerHTML = `(${x + ',' + y})`; // y - the Y-coordinate we want to place the reticle center at
this.show();
},
target: function(objQS) { // FN: target(objQS) Drops the reticle at TOP LEFT of the FIRST ELEMENT
let obj = (typeof(objQS) === 'string') ? document.querySelector(objQS) : objQS; // returned that matches the QuerySelector provided
if(objQS == null) return false; // params: objQS - The QuerySelector that will be matched to target the
obj = obj.getBoundingClientRect(); // element we wish to target
this.place(obj.x, obj.y); // Also will accept an htmlElement object.
this.show();
},
clickShowFn: function(e){ // FN: clickShowFn(e) Internal method compartmentalizing the place() function
reticle.place(e.pageX, e.pageY); // params: e - Window's event object used to determine click X/Y
},
enableOnClick: function(){ // FN: enableOnClick() Enables the reticle's "place wherever user just clicked" mode
this.hide(); // params: NONE
this.color('#070');
window.addEventListener('mouseup', this.clickShowFn);
},
disableOnClick: function(){ // FN: disableOnClick() Disables the reticle's "place wherever user just clicked" mode
this.hide(); // params: NONE
window.removeEventListener('mouseup', this.clickShowFn);
this.color('#F00');
},
color: function(clr='red') { // FN: color(clr) Sets the reticle's and its text's color to the specified string
this.retColor = clr; // params: clr - Color string to set the reticle and text to (supports hex,
this.ret().style.setProperty('--reticle-color', clr); // rgb, rgba, and color names)
},
show: function() { this.ret().style.display = 'block'; }, // FN: show() Helper function to make the reticle visible
hide: function() { this.ret().style.display = 'none'; } // FN: hide() Helper function to hide the reticle
}
.reticle{position:absolute;border:2px solid red;border-bottom:0 none;border-right:0 none;width:10px;height:10px;z-index:1000000;font-size:10px;font-weight:bold;color:red;line-height:10px;padding:5px;top:10rem;left:10rem}.reticle::before{content:'';position:absolute;border:2px solid red;border-top:0 none;border-left:0 none;width:10px;height:10px;z-index:1000000;transform:translate(calc(-150% + 1px),calc(-150% + 1px))}.reticle::after{content:'';position:absolute;border:1px solid red;border-radius:50%;width:10px;height:10px;z-index:1000000;transform:translate(-100%,calc(-10px + -100%))}
var reticle={retObj:null,retColor:"red",ret:function(){if(this.retObj=document.querySelector(".reticle")||null,null!==this.retObj)return this.retObj;{let e=`style="--reticle-color:${this.retColor}"`;document.body.innerHTML+=`<span class="reticle" ${e}>(0,0)</span>`,this.retObj=document.querySelector(".reticle")}return this.retObj},place:function(e,t){let i=this.ret();i.style.top=t+"px",i.style.left=e+"px",i.innerHTML=`(${e+","+t})`,this.show()},target:function(e){let t="string"==typeof e?document.querySelector(e):e;if(null==e)return!1;t=t.getBoundingClientRect(),this.place(t.x,t.y),this.show()},clickShowFn:function(e){reticle.place(e.pageX,e.pageY)},enableOnClick:function(){this.hide(),this.color("#070"),window.addEventListener("mouseup",this.clickShowFn)},disableOnClick:function(){this.hide(),window.removeEventListener("mouseup",this.clickShowFn),this.color("#F00")},color:function(e="red"){this.retColor=e,this.ret().style.setProperty("--reticle-color",e)},show:function(){this.ret().style.display="block"},hide:function(){this.ret().style.display="none"}};
.reticle {
position:absolute;
border:2px solid red;
border-bottom:0 none;
border-right:0 none;
width: 10px;
height: 10px;
z-index: 1000000;
font-size:10px;
font-weight:bold;
color:red;
line-height:10px;
padding:5px;
top:10rem;
left:10rem;
&::before {
content:'';
position:absolute;
border:2px solid red;
border-top:0 none;
border-left:0 none;
width: 10px;
height: 10px;
z-index: 1000000;
transform: translate(calc(-150% + 1px), calc(-150% + 1px));
}
&::after {
content:'';
position:absolute;
border:1px solid red;
border-radius: 50%;
width: 10px;
height: 10px;
z-index: 1000000;
transform: translate(-100%, calc(-10px + -100%));
}
}
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment