- Now Edge supports
outputObject.value — changed indicator.textContent to indicator.value.
- Added a font sizer; made some code improvements.
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1">
<meta name="description" content="Edit your HTML, CSS, and JavaScript code and monitor the instant live preview.">
<title>HTML Editor: online HTML editor with real-time preview</title>
<link rel="icon" href="favicon.ico">
<style>
html,
body {
margin: 0;
padding: 0;
height: 100%;
}
body {
display: flex;
flex-direction: column;
}
header,
footer.shown:not([hidden]) {
display: flex;
flex-wrap: wrap;
align-items: center;
gap: 5px;
padding: 5px;
}
header {
background: linear-gradient(#FFF, #CCC);
row-gap: 5px;
}
label,
#downloader,
select,
#resetter,
#selector,
#fileChooser,
output,
span {
font: bold 11px Arial;
color: #333;
}
[type="checkbox"] {
margin: 0 10px5px 0 5px;0;
}
#resetter,
#resizer[for="fontSizer"] {
margin-left: 0 5px;
}
#selectorselect,
button,
#resizer {
margin: 0;
}
#fileChooser {
margin: 0 auto 0 5px;0;
}
#resizer,
iframe {
padding: 0;
}
output {
margin-right: 10px;5px;
font-family: monospace;
}
#footerToggler {
width: 16px;
height: 16px;
margin: 0;
border: 1px solid #777;#666;
border-bottom-width: 5px;
padding: 0;
background: transparent;
}
#footerToggler.on {
border-color: #333;
background: #FFF;
}
main {
flex: 1;
display: flex;
}
main.horizontal {
flex-direction: column;
}
div {
flex: 0px;
min-basiswidth: 0;
positionmin-height: relative;0;
}
#viewerWrapper {
border-left: 5px solid #CCC;
}
main.horizontal #viewerWrapper {
border-left: 0;
border-top: 5px solid #CCC;
}
div * {
positiondisplay: absolute;block;
width: 100%;
height: 100%;
margin: 0;
border: 0;
background: #FFF;
}
textarea {
box-sizing: border-box;
padding: 5px;
outline: 0;
resize: none;
font-size: 14px;
color: #333;
}
textarea.dark {
background: #333;
color: #FFF;
}
footer.shown {
gap: 5px;
background: linear-gradient(#CCC, #FFF);
}
img {
display: block;
width: 18px;
height: 18px;
}
#copier {
border: 0;
padding: 0;
background: transparent;
cursor: pointer;
}
address {
margin-left: auto;
font-size: 16px;
italic font-family:16px 'Times New Roman';
color: #333;
}
address a {
color: inherit;
}
</style>
</head>
<body>
<header>
<label for="runner">Run</label>
<input type="checkbox" checkedid="runner" id="runner">checked>
<a href="" download="template.html" title="Download the HTML document" id="downloader">Download</a>
<input<label type="button"for="fontSizer">Font value="Reset"size</label>
id="resetter"> <select id="fontSizer">
<input <option>12</option>
<option>13</option>
<option selected>14</option>
<option>15</option>
<option>16</option>
<option>17</option>
<option>18</option>
<option>19</option>
<option>20</option>
</select>
<button type="button" value="Select"id="resetter">Reset</button>
id="selector"> <button type="button" id="selector">Select</button>
<input type="file" accept="text/html" id="fileChooser">
<label for="resizer">Editor size</label>
<input type="range" id="resizer">
<output for="resizer" id="indicator"><id="indicator">0.50</output>
<label for="viewsToggler">Horizontal view</label>
<input type="checkbox" id="viewsToggler">
<label for="themesToggler">Dark theme</label>
<input type="checkbox" id="themesToggler">
<input<button type="button" value="" title="Toggle footer" id="footerToggler">id="footerToggler"></button>
</header>
<main id="main">
<div id="editorWrapper">
<textarea spellcheck="false" id="editor"><!DOCTYPE html>
<html lang="en">
<head>
<title>HTML Document Template</title>
</head><style>
<body> p {
font-family: Arial;
}
</style>
</head>
<body>
<p>Hello, world!</p>
<script>
console.log(document.querySelector('p').textContent);
</script>
</body>
</html></textarea>
</div>
<div id="viewerWrapper">
<iframe id="viewer"></iframe>
</div>
</main>
<footer hiddenid="footer" id="footer">hidden>
<span>Share</span>
<a href="https://www.linkedintwitter.com/sharing/share-offsiteintent/tweet?url=http%3A%2F%2Fhtmleditortext=HTML%20Editor%3A%20online%20HTML%20editor%20with%20real-time%20preview&url=https%3A%2F%2Fhtmleditor.gitlab.io" target="_blank" title="Share on LinkedIn"><imgtarget="_blank"><img src="images/linkedintwitter.svg" alt="LinkedIn"><width="16" height="16" alt="Twitter"></a>
<a href="https://twitterwww.facebook.com/intent/tweetsharer.php?text=HTML%20Editor%3A%20online%20HTML%20editor%20with%20real-time%20preview&url=http%3A%2F%2Fhtmleditoru=https%3A%2F%2Fhtmleditor.gitlab.io" target="_blank" title="Share onio&t=HTML%20Editor%3A%20online%20HTML%20editor%20with%20real-time%20preview" Twitter"><imgtarget="_blank"><img src="images/twitterfacebook.svg" alt="Twitter"><width="16" height="16" alt="Facebook"></a>
<a href="https://www.facebooklinkedin.com/dialogsharing/share-offsite/?app_id=664554287087112&display=page&href=http%3A%2F%2Fhtmleditorurl=https%3A%2F%2Fhtmleditor.gitlab.io" target="_blank"target="_blank"><img title="Sharesrc="images/linkedin.svg" onwidth="16" Facebook"><imgheight="16" alt="LinkedIn"></a>
<a href="mailto:?subject=HTML%20Editor%3A%20online%20HTML%20editor%20with%20real-time%20preview&body=https%3A%2F%2Fhtmleditor.gitlab.io" target="_blank"><img src="images/facebookemail.svg" alt="Facebook"><width="16" height="16" alt="Email"></a>
<button type="button" id="copier"><img src="images/link.svg" width="16" height="16" alt="Link"></button>
<span id="notification" hidden>Copied!</span>
<address><a href="https://codereview.stackexchange.com/questions/56106/html-editor-online-html-editor-with-real-time-preview" title="Code Review Stack Exchange">Feedback</a> | Created by <a href="https://mori.pages.dev" title="Mori" rel="author">Mori</a></address>
</footer>
<script>
var runner = document.getElementById('runner'),
editor = document.getElementById('editor'),
downloader = document.getElementById('downloader'),
fileChooser = document.getElementById('fileChooser'),
resizer = document.getElementById('resizer'),
viewsToggler = document.getElementById('viewsToggler'),
themesToggler = document.getElementById('themesToggler');
function preview() {
if (runner.checked) {
var viewer = document.getElementById('viewer');
try {
var viewerDoc = viewer.contentDocument;
viewerDoc.open();
srcdoc = viewerDoc.write(editor.value);
viewerDoc.close();
} catch (e) { // in case of iframe redirection to a different origin
viewer.src = 'about:blank';
setTimeout(preview, 4); // minimum delay
}value;
}
}
editor.addEventListener('input', preview);
runner.addEventListener('change', preview);
function['click', createURL'contextmenu'].forEach(event => downloader.addEventListener(event, function() {
var blob = new Blob([editor.value], {
type: 'text/html'
});
downloaderthis.href = window.URL.createObjectURL(blob);
}
editor.addEventListener('change', createURL));
functiondocument.getElementById('fontSizer').addEventListener('change', previewAndCreateURLfunction() {
preview();
editor.style.fontSize = this.value + createURL();'px';
});
document.getElementById('resetter').addEventListener('click', function() {
if (!editor.value || editor.value != editor.defaultValue &&function confirmresetFileChooserAndDownload('Your changes will be lost.\nAre you sure you want to reset?')) {
downloader.download = 'template.html';
fileChooser.value = '';
downloader.download = 'template.html';
}
if (!editor.value || editor.value != editor.defaultValue && confirm('Your input will be lost.\nAre you sure you want to reset?')) {
resetFileChooserAndDownload();
editor.value = editor.defaultValue;
previewAndCreateURLpreview();
} else if (editor.value == editor.defaultValue) {
downloader.download = 'template.html';
fileChooser.value = '';resetFileChooserAndDownload();
}
});
document.getElementById('selector').addEventListener('click', function() {
editor.select();
});
fileChooser.addEventListener('change', async function() {
var file = this.files[0],
reader = new FileReader();files[0];
if (file) { // to ensure that there's a file to read so Chrome, for example, doesn't run this function when you cancel choosing a new file
downloader.download = file.name;
reader.readAsText(file);
reader.addEventListener('load', function() {
editor.value = this.result;
await previewAndCreateURLfile.text();
}preview();
}
});
functiondocument.getElementById('resizer').addEventListener('input', resizefunction() {
var resizerVal = resizerthis.value;
document.getElementById('editorWrapper').style.flexGrow = resizerVal;
document.getElementById('viewerWrapper').style.flexGrow = 100 - resizerVal;
document.getElementById('indicator').value = (resizerVal / 100).toFixed(2);
}
resizer.addEventListener('input', resize);
functiondocument.getElementById('viewsToggler').addEventListener('change', toggleViewsfunction() {
var main = document.getElementById('main');
if (viewsToggler.checked) {
main.className = 'horizontal';
} else {
mainclassList.className = '';
}toggle('horizontal');
}
viewsToggler.addEventListener('change', toggleViews);
function toggleThemesdocument.getElementById('themesToggler') {
if.addEventListener('change', function(themesToggler.checked) {
editor.className = 'dark';
} else {
editorclassList.className = '';
}toggle('dark');
}
themesToggler.addEventListener('change', toggleThemes);
document.getElementById('footerToggler').addEventListener('click', function() {
this.classList.toggle('on');
document.getElementById('footer').classListtoggleAttribute('hidden');
});
document.togglegetElementById('shown''copier').addEventListener('click', function() {
navigator.clipboard.writeText('https://htmleditor.gitlab.io');
function toggleNotification() {
document.getElementById('notification').toggleAttribute('hidden');
}
toggleNotification();
setTimeout(toggleNotification, 1500);
});
window.addEventListener('beforeunload', function(event) {
if (editor.value && editor.value != editor.defaultValue) {
event.returnValuepreventDefault();
= 'Your changes may be lost event.';returnValue = '';
}
});
resize();
toggleViews();
toggleThemes();
previewAndCreateURLpreview();
</script>
</body>
</html>