I am creating an interactive website where the user is able to click anywhere on the page and each paragraph will show up one by one. At the moment the paragraphs are displaying to the right of the click but i want the paragraph to be centred on the mouse position. Is there a way to do this?
Is there also a way to restrict the paragraphs from going off the screen as this creates a landscape scroll which i don't want, Any help would be appreciated.
const texts = [
"Paragraph: 1",
"Paragraph: 2",
"Paragraph: 3",
"Paragraph: 4",
"Paragraph: 5",
"Paragraph: 6",
"Paragraph: 7",
"Paragraph: 8",
"Paragraph: 9",
"Paragraph: 10"
];
const classes = [
"red gray-bg font-1",
"blue font-2",
"red",
"blue",
"red",
"blue",
"red",
"blue",
"red",
"blue",
];
// Keeps Track of Current Text
let currentParaIdx = 0;
document.addEventListener("click", (e) => {
// Stop once All Texts are Displayed
if (currentParaIdx === texts.length) return;
const { clientX, clientY } = e; //get the click position
//create the div
const div = document.createElement("div");
//set its text
div.innerText = texts[currentParaIdx];
//set its position and position style
div.classList=classes[currentParaIdx];
div.style.position = "absolute";
div.style.left = clientX + "px";
div.style.top = clientY + "px";
currentParaIdx++;
document.body.append(div); //add div to the page
});
This is the code I have so far
Notes
While there isn't anything wrong with your code, there are a few things that could be changed that should make things a little more optimized and easier to manage your code going forward.
The first is your two arrays for the text and classes. Creating one array of objects would allow you to only have to work with one array, and you can pull whichever key/property you need depending on what you are doing.
Another thing is using template literals instead of string concatenation when setting some of your values. This is a small thing, but typically a good idea to do.
Solution
Essentially, if you want to center an element on your mouse/cursor position, you need to move it by half of the elements
width
andheight
. In this situation, we are working with an element that is being dynamically created and so it initially will not have awidth
orheight
. We can add it to the document first, pull those values and then adjust thetop
andleft
properties accordingly. And because JavaScript is processed synchronously, this won't be noticeable to the user.Additional Notes
I opted to use
getComputedStyle()
here instead of something likegetBoundingClientRect()
when getting things like thewidth
andheight
because things likepadding
on an element can make these values inaccurate. Not knowing the full context of how this will be used, I felt it was a safer choice.Edit
To answer a comment, as I need to use backticks and Stack Overflow code formatting will interfere, I'm adding this here:
Basically you wrap the entire value in backticks, And any JavaScript variables need to be wrapped in
${}
. All other text can be placed inside normally, but this way you don't need to use things like the+
operator to concatenate multiple strings.