Як додати подію завантаження до елемента div


231

Як додати onloadподію до елемента?

Чи можу я використовувати:

<div onload="oQuickReply.swap();" ></div>

для цього?


кращий bodyкращийdiv
KingRider

2
divелементи не "завантажуються".
amn

1
Зараз є більш відповідна відповідь - можливо, ви можете переглянути прийняття?
mplungjan

Відповіді:


233

Ні, ти не можеш. Найпростіший спосіб змусити його ставити виклик функції безпосередньо після елемента

Приклад:

...
<div id="somid">Some content</div>
<script type="text/javascript">
   oQuickReply.swap('somid');
</script>
...

або - ще краще - прямо перед </body>:

...
<script type="text/javascript">
   oQuickReply.swap('somid');
</script>
</body>

... тому він не блокує завантаження наступного вмісту.


3
Залежно від використання не краще ставити його перед </body>. fe, якщо ви хочете приховати <div>єдине, якщо JavaScript увімкнено, але уникайте "миготіння". Час видимості залежить від того, як довго браузер повинен завантажувати / аналізувати всі вбудовані / зовнішні сценарії.
mgutt

Це анти-шаблон, оскільки краще зберігати всі js у власному файлі? Чи існує спосіб вибіркового запуску js, коли певні елементи завантажуються в документ?
Райан Уолтон

1
Це не анти-шаблон, але ви зазвичай чекаєте, коли DOM буде завантажений, а потім виконайте всі речі JS.
DanMan

1
Передайте тут посилання для всіх, хто стикається з цією відповіддю w3schools.com/tags/ev_onload.asp - Усі елементи HTML, які зараз підтримуютьonload
Брендон Бенефілд

Ця відповідь більше не актуальна. Ось новий
mplungjan

74

onloadПодія може бути використано тільки на document(body)сам, рамки, зображення і скрипти. Іншими словами, він може бути приєднаний лише до органу та / або кожного зовнішнього ресурсу . Div не є зовнішнім ресурсом і завантажується як частина тіла, тому onloadподія там не застосовується.


18
Не тільки в елементі тіла, ви можете використовувати його також, наприклад, із зображенням та рамкою кадрів. w3schools.com/jsref/event_onload.asp
Joe

2
Варто зазначити, що вбудовані скрипти НЕ є зовнішніми ресурсами, і тому onloadне працює жоден <script>без srcатрибута HTML (я намагався використовувати цю пораду в вбудованому сценарії, і виявив, що він не працює)
gog

64

Ви можете запустити деякі js автоматично на елементі IMG за допомогою onerror, а не src.

<img src onerror='alert()'>

4
Блискуче! Можна використовувати і для запуску машинописного тексту з кутового: img src (помилка) = "функція ()" />
Брайан Річардсон,

1
Це фантастичний підхід. Дивіться моє розширення вашого рішення тут: stackoverflow.com/questions/4057236 / ...
Rounin

Канюза? ͏͏͏͏͏͏͏
Pacerier

28

події завантаження він підтримує лише з кількома тегами, як перелічено нижче.

<body>, <frame>, <iframe>, <img>, <input type="image">, <link>, <script>, <style>

Тут посилання на подію onload


17

Спробуйте це! І ніколи не використовуйте тригер двічі на діві!

Ви можете визначити функцію для виклику перед тегом div.

$(function(){
    $('div[onload]').trigger('onload');
});

DEMO: jsfiddle


Нагорі у jsfiddle немає завантажувача jQuery - і він не працює.
Аріф Бурхан

@Arif Burhan Я не розумію. Я завантажую JQuery (край). Ви можете перевірити ще раз? Я бачу "Hello World" навіть за допомогою мобільного.
Куопп

@Kuofp Ця скрипка насправді не обробляє обробників, коли завантажуються цільові елементи. Він просто використовує jquery для пошуку елементів, які мають атрибут (у вашому випадку onload), а потім викликає ці функції. Сценарій працює, навіть якщо ви змінили атрибут на щось інше, onloadнаприклад, наприклад, jsfiddle.net/mrszgbxh
Trindaz

1
@Trindaz Рівно! Інакше кажучи, сценарії діють як подія завантаження. Дякуємо, що залишили коментар!
Kuofp

10

Я просто хочу додати тут, що якщо хтось хоче зателефонувати на функцію завантаження події div & ви не хочете використовувати jQuery (через конфлікт, як у моєму випадку), тоді просто викликайте функцію після всього коду html або будь-якого інший код, який ви написали, включаючи код функції та просто зателефонуйте до функції.

/* All Other Code*/
-----
------
/* ----At the end ---- */
<script type="text/javascript">
   function_name();
</script>

АБО

/* All Other Code*/
-----
------
/* ----At the end ---- */
<script type="text/javascript">
 function my_func(){
   function definition;      

  }

 my_func();
</script>

Чи можете ви додати таким чином кілька функцій? тобто наприкінці html-документа безпосередньо перед тегом </body>? Якщо так, чи має значення порядок, у якому вони написані?
Нео

Так, ви можете додати більше ніж одну функцію, і порядок не має значення для її простого (ванільного) JavaScript.
dev_khan

7

ми можемо використовувати MutationObserver, щоб ефективно вирішити проблему, додавши приклад код нижче

<!DOCTYPE html>
<html>
<head>
    <title></title>
    <style>
        #second{
            position: absolute;
            width: 100px;
            height: 100px;
            background-color: #a1a1a1;
        }
    </style>
</head>
<body>
<div id="first"></div>
<script>
    var callthis = function(element){
           element.setAttribute("tabIndex",0);
        element.focus();
        element.onkeydown = handler;
        function handler(){
                alert("called")
        }
    }


    var observer = new WebKitMutationObserver(function(mutations) {
        mutations.forEach(function(mutation) {
            for (var i = 0; i < mutation.addedNodes.length; i++)
            if(mutation.addedNodes[i].id === "second"){
                callthis(mutation.addedNodes[i]);
            }

        })
    });
    observer.observe(document.getElementById("first"), { childList: true });


    var ele = document.createElement('div');
    ele.id = "second"

    document.getElementById("first").appendChild(ele);

</script>

</body>
</html>

1
Востаннє я перевіряв, як події мутації виконують жахливо.
DanMan

7

У листопаді 2019 року я шукаю спосіб створити (гіпотетичний), onparse EventListenerдля <elements>якого не беруться onload.

(Гіпотетичний) onparse EventListenerповинен вміти слухати, коли елемент аналізується .


Третя спроба (і остаточне рішення)

Я був дуже задоволений Другою спробою нижче, але мене просто вразило, що я можу зробити код коротшим та простішим, створивши індивідуальний захід:

let parseEvent = new Event('parse');

Це найкраще рішення поки.

Приклад нижче:

  1. Створює під замовлення parse Event
  2. Декларує функцію (яку можна запустити в window.onloadбудь-який час), яка:
    • Знаходить будь-які елементи в документі, які включають атрибут data-onparse
    • Прикріплює parse EventListenerдо кожного з цих елементів
    • Відправляє parse Eventкожен із цих елементів для виконанняCallback

Приклад роботи:

// Create (homemade) parse event
let parseEvent = new Event('parse');

// Create Initialising Function which can be run at any time
const initialiseParseableElements = () => {

  // Get all the elements which need to respond to an onparse event
  let elementsWithParseEventListener = document.querySelectorAll('[data-onparse]');
  
  // Attach Event Listeners and Dispatch Events
  elementsWithParseEventListener.forEach((elementWithParseEventListener) => {

    elementWithParseEventListener.addEventListener('parse', updateParseEventTarget, false);
    elementWithParseEventListener.dataset.onparsed = elementWithParseEventListener.dataset.onparse;
    elementWithParseEventListener.removeAttribute('data-onparse');
    elementWithParseEventListener.dispatchEvent(parseEvent);
  });
}

// Callback function for the Parse Event Listener
const updateParseEventTarget = (e) => {
  
  switch (e.target.dataset.onparsed) {

    case ('update-1') : e.target.textContent = 'My First Updated Heading'; break;
    case ('update-2') : e.target.textContent = 'My Second Updated Heading'; break;
    case ('update-3') : e.target.textContent = 'My Third Updated Heading'; break;
    case ('run-oQuickReply.swap()') : e.target.innerHTML = 'This <code>&lt;div&gt;</code> is now loaded and the function <code>oQuickReply.swap()</code> will run...'; break;
  }
}

// Run Initialising Function
initialiseParseableElements();

let dynamicHeading = document.createElement('h3');
dynamicHeading.textContent = 'Heading Text';
dynamicHeading.dataset.onparse = 'update-3';

setTimeout(() => {

  // Add new element to page after time delay
  document.body.appendChild(dynamicHeading);

  // Re-run Initialising Function
  initialiseParseableElements();

}, 3000);
div {
  width: 300px;
  height: 40px;
  padding: 12px;
  border: 1px solid rgb(191, 191, 191);
}

h3 {
position: absolute;
top: 0;
right: 0;
}
<h2 data-onparse="update-1">My Heading</h2>
<h2 data-onparse="update-2">My Heading</h2>
<div data-onparse="run-oQuickReply.swap()">
This div hasn't yet loaded and nothing will happen.
</div>


Друга спроба

Перша спроба нижче ( на основі @JohnWilliams"Блискучий Empty Image Hack ) використовувала зашита <img />і працював.

Я подумав, що має бути можливо повністю видалити жорсткий код <img />і лише динамічно вставити його після виявлення в елементі, необхідному для запуску onparseподії, такого атрибута, як:

data-onparse="run-oQuickReply.swap()"

Виявляється, це справді дуже добре працює.

Приклад нижче:

  1. Знаходить будь-які елементи в документі, які включають атрибут data-onparse
  2. Динамічно генерує <img src />та додає його до документа, відразу після кожного з цих елементів
  3. Запускає, onerror EventListenerколи двигун візуалізації аналізує кожен<img src />
  4. Виконує Callback та видаляє, що динамічно генерується <img src />з документа

Приклад роботи:

// Get all the elements which need to respond to an onparse event
let elementsWithParseEventListener = document.querySelectorAll('[data-onparse]');

// Dynamically create and position an empty <img> after each of those elements 
elementsWithParseEventListener.forEach((elementWithParseEventListener) => {

  let emptyImage = document.createElement('img');
  emptyImage.src = '';
  elementWithParseEventListener.parentNode.insertBefore(emptyImage, elementWithParseEventListener.nextElementSibling);
});

// Get all the empty images
let parseEventTriggers = document.querySelectorAll('img[src=""]');

// Callback function for the EventListener below
const updateParseEventTarget = (e) => {

  let parseEventTarget = e.target.previousElementSibling;
  
  switch (parseEventTarget.dataset.onparse) {

    case ('update-1') : parseEventTarget.textContent = 'My First Updated Heading'; break;
    case ('update-2') : parseEventTarget.textContent = 'My Second Updated Heading'; break;
    case ('run-oQuickReply.swap()') : parseEventTarget.innerHTML = 'This <code>&lt;div&gt;</code> is now loaded and the function <code>oQuickReply.swap()</code> will run...'; break;
  }
  
  // Remove empty image
  e.target.remove();
}

// Add onerror EventListener to all the empty images
parseEventTriggers.forEach((parseEventTrigger) => {
  
  parseEventTrigger.addEventListener('error', updateParseEventTarget, false);
  
});
div {
  width: 300px;
  height: 40px;
  padding: 12px;
  border: 1px solid rgb(191, 191, 191);
}
<h2 data-onparse="update-1">My Heading</h2>
<h2 data-onparse="update-2">My Heading</h2>
<div data-onparse="run-oQuickReply.swap()">
This div hasn't yet loaded and nothing will happen.
</div>


Перша спроба

Я можу побудувати на @JohnWilliams" <img src>хаку" (на цій сторінці, з 2017 року) - це поки що найкращий підхід, який я натрапив.

Приклад нижче:

  1. Запускає, onerror EventListenerколи двигун візуалізації розбирає<img src />
  2. Виконує Callback та видаляє <img src />документ із документа

Приклад роботи:

let myHeadingLoadEventTrigger = document.getElementById('my-heading-load-event-trigger');

const updateHeading = (e) => {

  let myHeading = e.target.previousElementSibling;
  
  if (true) { // <= CONDITION HERE
    
    myHeading.textContent = 'My Updated Heading';
  }
  
  // Modern alternative to document.body.removeChild(e.target);
  e.target.remove();
}

myHeadingLoadEventTrigger.addEventListener('error', updateHeading, false);
<h2>My Heading</h2>
<img id="my-heading-load-event-trigger" src />



@DavidBradshaw - Ви ніколи не повірите, але я думаю , що я б на самому справі зламали. Дивіться другу спробу вгорі відповіді вгорі.
Rounin

1
@DavidBradshaw - подряпини це. Я придумав третю спробу . Це остаточне рішення.
Rounin

Можливо, скоротити відповідь до версії 3
Девід Бредшоу

6

використовуйте iframe та приховайте його, якщо кадр працює як тег тіла

<!DOCTYPE html>
<html>
<body>

<iframe style="display:none" onload="myFunction()" src="http://www.w3schools.com"></iframe>
<p id="demo"></p>

<script>
function myFunction() {
    document.getElementById("demo").innerHTML = "Iframe is loaded.";
}
</script>

</body>
</html>

6

Мені потрібно було запустити деякий код ініціалізації після того, як було вставлено фрагмент html (шаблона), і я, звичайно, не мав доступу до коду, який маніпулює шаблоном і модифікує DOM. Ця ж ідея стосується будь-якої часткової модифікації DOM шляхом вставки html-елемента, як правило, a <div>.

Деякий час тому я зробив хак з події завантаження майже невидимого, що <img>міститься в <div>, але виявив, що масштабний, порожній стиль також буде робити:

<div .... >
<style scoped="scoped" onload="dosomethingto(this.parentElement);" >   </style>
.....
</div>

Оновлення (15 липня 2017 р.) - завантаження <style>не підтримується в останній версії IE. Edge це підтримує, але деякі користувачі бачать це як інший браузер і дотримуються IE. <img>Елемент , здається, краще працювати у всіх браузерах.

<div...>
<img onLoad="dosomthing(this.parentElement);" src="" />
...
</div>

Щоб мінімізувати візуальний вплив та використання ресурсів зображення, використовуйте вбудований src, який зберігає його невеликим та прозорим.

Один коментар, який я думаю, що мені потрібно зробити щодо використання a, <script>- наскільки складніше визначити, який <div>сценарій знаходиться поруч, особливо в шаблонах, коли ви не можете мати ідентичний ідентифікатор у кожному екземплярі, який генерує шаблон. Я думав, що відповідь може бути document.currentScript, але це не підтримується повсюдно. <script>Елемент не може визначити своє власне місце розташування DOM надійно; посилання на "це" вказує на головне вікно і не допомагає.

Я вважаю, що потрібно погодитися з використанням <img>елемента, незважаючи на те, що вони є "goofy". Це може бути дірка в рамках DOM / javascript, яка може використовувати підключення.


3

Оскільки onloadподія підтримується лише на кількох елементах, вам доведеться використовувати альтернативний метод.

Ви можете використовувати MutationObserver для цього:

const trackElement = element => {
  let present = false;
  const checkIfPresent = () => {
    if (document.body.contains(element)) {
      if (!present) {
        console.log('in DOM:', element);
      }
      present = true;
    } else if (present) {
      present = false;
      console.log('Not in DOM');
    }
  };

  const observer = new MutationObserver(checkIfPresent);
  observer.observe(document.body, { childList: true });
  checkIfPresent();

  return observer;
};

const element = document.querySelector('#element');
const add = () => document.body.appendChild(element);
const remove = () => element.remove();

trackElement(element);
<button onclick="add()">Add</button>
<button onclick="remove()">Remove</button>

<div id="element">Element</div>


2

Мені дуже подобається бібліотека YUI3 за подібні речі.

<div id="mydiv"> ... </div>

<script>
YUI().use('node-base', function(Y) {
  Y.on("available", someFunction, '#mydiv')
})

Дивіться: http://developer.yahoo.com/yui/3/event/#onavailable


9
Це досить велика крапка JS, щоб скинути на сторінку заради прив’язки синглу onload.
meagar

1
@meagar - Правильно, хоча мені все рідше здається, що я роблю лише дві-три прості речі JS на сторінці. Турбота про сумісність між веб-переглядачами також зводить мене з душі.
mjhm

0

Я вивчаю javascript та jquery і пережив усю відповідь, я зіткнувся з тією ж проблемою, коли викликав функцію javascript для завантаження елемента element. Я спробував, $('<divid>').ready(function(){alert('test'})і це спрацювало на мене. Я хочу знати, чи це хороший спосіб виконувати виклик завантаження на елемент div так, як я це робив за допомогою селектора jquery.

Дякую


0

Як все сказано, ви не можете використовувати onLoad подію на DIV, але це перед тегом body.

але у випадку, якщо у вас є один колонтитул і включіть його на багато сторінок. краще спочатку перевірити, чи потрібний div на цій сторінці відображається, щоб код не виконувався на сторінках, які не містять цей DIV, щоб швидше завантажувався та економив час для вашої програми.

тож вам потрібно буде вказати ідентифікатору DIV та зробити:

var myElem = document.getElementById('myElementId');
if (myElem !== null){ put your code here}

0

У мене було те саме питання, і я намагався отримати Div для завантаження сценарію прокрутки, використовуючи onload або load. Проблема, яку я знайшов, полягала в тому, що вона завжди працюватиме до того, як Div вдасться відкритись, а не під час і після, тож насправді це не спрацювало.

Тоді я підійшов до цього як до роботи.

<body>

<span onmouseover="window.scrollTo(0, document.body.scrollHeight);" 
onmouseout="window.scrollTo(0, document.body.scrollHeight);">

<div id="">
</div>

<a href="" onclick="window.scrollTo(0, document.body.scrollHeight);">Link to open Div</a>

</span>
</body>

Я розмістив Div всередині Span і дав Span дві події, миші та миші. Потім під цим Div я розмістив посилання, щоб відкрити Div, і дав йому посилання для onclick. Усі події точно такі ж, щоб змусити сторінку прокручуватися донизу сторінки. Тепер, коли натиснути кнопку відкриття Div, сторінка перескочить частину вниз, а Div відкриється над кнопкою, викликаючи події миші та миші, щоб допомогти просунути сценарій прокрутки вниз. Тоді будь-який рух миші в цій точці підштовхне сценарій в останній раз.


0

Ви можете використовувати інтервал, щоб перевірити його, поки він не завантажиться так: https://codepen.io/pager/pen/MBgGGM

let checkonloadDoSomething = setInterval(() => {
  let onloadDoSomething = document.getElementById("onloadDoSomething");
  if (onloadDoSomething) {
    onloadDoSomething.innerHTML="Loaded"
    clearInterval(checkonloadDoSomething);
  } else {`enter code here`
    console.log("Waiting for onloadDoSomething to load");
  }
}, 100);

0

Перш ніж відповісти на ваше запитання: Ні, ви не можете, не прямо так, як хотіли це зробити. Може відповісти трохи пізно, але це моє рішення, без jQuery, чистого JavaScript. Спочатку було написано, щоб застосувати функцію зміни розміру до текстових областей після завантаження DOM та під час клавіатури.

Таким же чином, ви можете використовувати його, щоб зробити щось із (усіма) divs або лише одним, якщо вказано, як-от так:

document.addEventListener("DOMContentLoaded", function() {
    var divs = document.querySelectorAll('div'); // all divs
    var mydiv = document.getElementById('myDiv'); // only div#myDiv
    divs.forEach( div => {
        do_something_with_all_divs(div);
    });
    do_something_with_mydiv(mydiv);
});

Якщо вам дійсно потрібно щось зробити з дівом, завантаженим після завантаження DOM, наприклад, після виклику Ajax, ви можете використовувати дуже корисний хак, який легко зрозуміти, ви знайдете його ... елементи, перед тим , в-будинок-це готовий ... . Він говорить "до того, як DOM буде готовий", але це працює надзвичайно таким же чином, після вставки Ajax або js-appendChild - що б не було з дива. Ось код, із деякими крихітними змінами до моїх потреб.

css

.loaded { // I use only class loaded instead of a nodename
    animation-name: nodeReady;
    animation-duration: 0.001s;
}

@keyframes nodeReady {  
    from { clip: rect(1px, auto, auto, auto); }
    to { clip: rect(0px, auto, auto, auto); }  
}

javascript

document.addEventListener("animationstart", function(event) {
    var e = event || window.event;
    if (e.animationName == "nodeReady") {
        e.target.classList.remove('loaded');
        do_something_else();
    }
}, false);

0

Коли ви завантажуєте якийсь HTML з сервера і вставляєте його в дерево DOM, ви можете використовувати, DOMSubtreeModifiedпроте він застарілий - так що ви можете використовувати MutationObserverабо просто виявляти новий вміст всередині функції loadElement безпосередньо, тому вам не потрібно буде чекати подій DOM


0

Ви можете долучити слухача подій, як показано нижче. Він буде спрацьовувати кожного разу, коли div, що має селектор, #my-idповністю завантажує DOM.

$(document).on('EventName', '#my-id', function() {
 // do something
});

У цьому випадку EventName може бути "завантажити" або "натиснути"

https://api.jquery.com/on/#on-events-selector-data-handler


-1

Використовуйте замість body.onload подію або через attribute ( <body onload="myFn()"> ...), або прив’язуючи подію у Javascript. Це надзвичайно часто в jQuery :

$(document).ready(function() {
    doSomething($('#myDiv'));
});

javascript pure and not jquery, check tag;)
KingRider

-3

Спробуйте це.

document.getElementById("div").onload = alert("This is a div.");
<div id="div">Hello World</div>

Спробуйте і цей. Ви повинні видалити .з oQuickReply.swap()зробити функцію працювати.

document.getElementById("div").onload = oQuickReplyswap();
function oQuickReplyswap() {
alert("Hello World");
}
<div id="div"></div>


Здається, це працює, але це просто, що alert("This is a div.");називається, коли виконується цей рядок і undefinedприсвоюється результат (який є ) div.onload. Що абсолютно безглуздо.
Фредерік

-4

Ви не можете додати події onload в div, але ви можете додати onkeydown та запустити подію onkeydown при завантаженні документа

$(function ()
{
  $(".ccsdvCotentPS").trigger("onkeydown");
});
<script src="https://ajax.googleapis.com/ajax/libs/jquery/1.2.3/jquery.min.js"></script>

<div  onkeydown="setCss( );"> </div>`

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.