<!DOCTYPE html>
<html lang="th">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>เครื่องตั้งสายกีตาร์แบบ Chromatic</title>
<style>
* {
box-sizing: border-box;
margin: 0;
padding: 0;
font-family: 'Segoe UI', Tahoma, Geneva, Verdana, sans-serif;
}
body {
background: linear-gradient(135deg, #1a2a6c, #b21f1f, #fdbb2d);
color: #fff;
min-height: 100vh;
padding: 20px;
display: flex;
flex-direction: column;
align-items: center;
}
.container {
max-width: 800px;
width: 100%;
background: rgba(0, 0, 0, 0.7);
border-radius: 15px;
padding: 20px;
box-shadow: 0 10px 30px rgba(0, 0, 0, 0.5);
margin-top: 20px;
}
header {
text-align: center;
margin-bottom: 30px;
}
h1 {
font-size: 2.5rem;
margin-bottom: 10px;
color: #fdbb2d;
text-shadow: 2px 2px 4px rgba(0, 0, 0, 0.5);
}
.description {
font-size: 1.1rem;
margin-bottom: 20px;
line-height: 1.6;
text-align: center;
}
.string-container {
display: flex;
align-items: center;
margin-bottom: 15px;
padding: 15px;
background: rgba(255, 255, 255, 0.1);
border-radius: 10px;
transition: all 0.3s ease;
}
.string-container:hover {
background: rgba(255, 255, 255, 0.2);
transform: translateY(-2px);
}
.string-label {
width: 80px;
font-weight: bold;
font-size: 1.2rem;
}
.string-name {
color: #fdbb2d;
}
.string-frequency {
color: #4ecdc4;
margin-left: 5px;
}
.tuner {
flex: 1;
display: flex;
align-items: center;
}
.needle {
width: 40%;
height: 4px;
background: #ff6b6b;
position: relative;
margin: 0 15px;
}
.needle::after {
content: '';
position: absolute;
top: -8px;
right: -5px;
width: 0;
height: 0;
border-left: 10px solid #ff6b6b;
border-top: 10px solid transparent;
border-bottom: 10px solid transparent;
}
.frequency-display {
width: 100px;
text-align: center;
font-size: 1.2rem;
font-weight: bold;
color: #ff6b6b;
}
.controls {
display: flex;
justify-content: center;
gap: 15px;
margin-top: 30px;
}
button {
padding: 12px 25px;
border: none;
border-radius: 50px;
background: #fdbb2d;
color: #1a2a6c;
font-weight: bold;
cursor: pointer;
transition: all 0.3s ease;
font-size: 1rem;
}
button:hover {
background: #ffcc44;
transform: translateY(-3px);
box-shadow: 0 5px 15px rgba(0, 0, 0, 0.3);
}
.play-btn {
background: #4ecdc4;
color: #fff;
}
.instructions {
margin-top: 30px;
padding: 20px;
background: rgba(255, 255, 255, 0.1);
border-radius: 10px;
line-height: 1.6;
}
.instructions h2 {
margin-bottom: 10px;
color: #fdbb2d;
}
.instructions ol {
padding-left: 20px;
}
.instructions li {
margin-bottom: 10px;
}
@media (max-width: 600px) {
.string-container {
flex-direction: column;
align-items: flex-start;
}
.tuner {
width: 100%;
margin-top: 10px;
}
.controls {
flex-direction: column;
align-items: center;
}
button {
width: 100%;
}
}
</style>
</head>
<body>
<header>
<h1>เครื่องตั้งสายกีตาร์แบบ Chromatic</h1>
<p class="description">ใช้เครื่องมือนี้เพื่อตั้งสายกีตาร์ของคุณให้ตรงกับความถี่ที่ถูกต้อง</p>
</header>
<div class="container">
<div class="string-container">
<div class="string-label">สายที่ 1 <span class="string-name">(E)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">329.63 Hz</div>
<button class="play-btn" data-frequency="329.63">เล่นเสียง</button>
</div>
<div class="string-container">
<div class="string-label">สายที่ 2 <span class="string-name">(B)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">246.94 Hz</div>
<button class="play-btn" data-frequency="246.94">เล่นเสียง</button>
</div>
<div class="string-container">
<div class="string-label">สายที่ 3 <span class="string-name">(G)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">196.00 Hz</div>
<button class="play-btn" data-frequency="196.00">เล่นเสียง</button>
</div>
<div class="string-container">
<div class="string-label">สายที่ 4 <span class="string-name">(D)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">146.83 Hz</div>
<button class="play-btn" data-frequency="146.83">เล่นเสียง</button>
</div>
<div class="string-container">
<div class="string-label">สายที่ 5 <span class="string-name">(A)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">110.00 Hz</div>
<button class="play-btn" data-frequency="110.00">เล่นเสียง</button>
</div>
<div class="string-container">
<div class="string-label">สายที่ 6 <span class="string-name">(E)</span></div>
<div class="tuner">
<div class="needle"></div>
</div>
<div class="frequency-display">82.41 Hz</div>
<button class="play-btn" data-frequency="82.41">เล่นเสียง</button>
</div>
<div class="controls">
<button id="play-all">เล่นทั้งหมด</button>
<button id="reset">รีเซ็ต</button>
</div>
</div>
<div class="container instructions">
<h2>วิธีการใช้</h2>
<ol>
<li>กดปุ่ม "เล่นเสียง" สำหรับแต่ละสายเพื่อฟังเสียงอ้างอิง</li>
<li>ดีดสายกีตาร์ที่ต้องการตั้งเสียง</li>
<li>ปรับสายกีตาร์จนกว่าเข็มจะอยู่ตรงกลาง (แสดงว่าตั้งเสียงถูกต้องแล้ว)</li>
<li>ทำซ้ำสำหรับแต่ละสายจนกว่าทุกสายจะตั้งเสียงถูกต้อง</li>
<li>ใช้ปุ่ม "เล่นทั้งหมด" เพื่อฟังเสียงทั้งหมดต่อเนื่องกัน</li>
</ol>
</div>
<script>
// ฟังก์ชันสร้างเสียง sine wave
function playFrequency(frequency, duration = 1500) {
const audioContext = new (window.AudioContext || window.webkitAudioContext)();
const oscillator = audioContext.createOscillator();
const gainNode = audioContext.createGain();
oscillator.type = 'sine';
oscillator.frequency.value = frequency;
gainNode.gain.value = 0.3;
oscillator.connect(gainNode);
gainNode.connect(audioContext.destination);
oscillator.start();
// ลดเสียงลงก่อนหยุดเพื่อป้องกันเสียงตัดขาด
gainNode.gain.exponentialRampToValueAtTime(0.0001, audioContext.currentTime + duration / 1000);
setTimeout(() => {
oscillator.stop();
}, duration);
}
// เพิ่ม event listener ให้กับปุ่มเล่นเสียงแต่ละสาย
document.querySelectorAll('.play-btn').forEach(button => {
button.addEventListener('click', () => {
const frequency = parseFloat(button.getAttribute('data-frequency'));
playFrequency(frequency);
// แอนิเมชันเข็ม
const needle = button.parentElement.querySelector('.needle');
needle.style.transform = 'translateX(0)';
setTimeout(() => {
needle.style.transform = 'translateX(20px)';
}, 100);
setTimeout(() => {
needle.style.transform = 'translateX(-20px)';
}, 500);
setTimeout(() => {
needle.style.transform = 'translateX(0)';
}, 900);
});
});
// เพิ่ม event listener ให้กับปุ่มเล่นทั้งหมด
document.getElementById('play-all').addEventListener('click', () => {
const frequencies = [329.63, 246.94, 196.00, 146.83, 110.00, 82.41];
let delay = 0;
frequencies.forEach((freq, index) => {
setTimeout(() => {
playFrequency(freq, 1200);
// เน้นสายที่กำลังเล่น
const stringContainer = document.querySelectorAll('.string-container')[index];
stringContainer.style.background = 'rgba(78, 205, 196, 0.3)';
setTimeout(() => {
stringContainer.style.background = '';
}, 1200);
}, delay);
delay += 1200;
});
});
// เพิ่ม event listener ให้กับปุ่มรีเซ็ต
document.getElementById('reset').addEventListener('click', () => {
document.querySelectorAll('.needle').forEach(needle => {
needle.style.transform = 'translateX(0)';
});
});
// จำลองการตั้งสาย (สำหรับการสาธิต)
document.querySelectorAll('.string-container').forEach(container => {
container.addEventListener('click', (e) => {
if (!e.target.classList.contains('play-btn')) {
const needle = container.querySelector('.needle');
const randomOffset = (Math.random() * 40 - 20) + 'px';
needle.style.transform = `translateX(${randomOffset})`;
}
});
});
</script>
</body>
</html>
To embed this project on your website, copy the following code and paste it into your website's HTML: