Yepyeni Media Session API ile artık web uygulamanızın oynattığı medya için meta veri sağlayarak medya bildirimlerini özelleştirebilirsiniz. Ayrıca, bildirimlerden veya medya tuşlarından gelebilecek sarma veya parça değiştirme gibi medya ile ilgili etkinlikleri yönetmenize olanak tanır. Heyecanlandınız mı? Resmi medya oturumu örneklerini deneyin.
Media Session API, Chrome 57'de desteklenir (Şubat 2017'de beta, Mart 2017'de kararlı sürüm).
Gimme what I want
MediaSession API'yi zaten biliyor ve utanmadan bazı standart kodları kopyalayıp yapıştırmak için mi geri geldiniz? İşte burada.
if ('mediaSession' in navigator) {
navigator.mediaSession.metadata = new MediaMetadata({
title: 'Never Gonna Give You Up',
artist: 'Rick Astley',
album: 'Whenever You Need Somebody',
artwork: [
{ src: 'https://dummyimage.com/96x96', sizes: '96x96', type: 'image/png' },
{ src: 'https://dummyimage.com/128x128', sizes: '128x128', type: 'image/png' },
{ src: 'https://dummyimage.com/192x192', sizes: '192x192', type: 'image/png' },
{ src: 'https://dummyimage.com/256x256', sizes: '256x256', type: 'image/png' },
{ src: 'https://dummyimage.com/384x384', sizes: '384x384', type: 'image/png' },
{ src: 'https://dummyimage.com/512x512', sizes: '512x512', type: 'image/png' },
]
});
navigator.mediaSession.setActionHandler('play', function() {});
navigator.mediaSession.setActionHandler('pause', function() {});
navigator.mediaSession.setActionHandler('seekbackward', function() {});
navigator.mediaSession.setActionHandler('seekforward', function() {});
navigator.mediaSession.setActionHandler('previoustrack', function() {});
navigator.mediaSession.setActionHandler('nexttrack', function() {});
}
Kodu inceleme
Haydi oynayalım 🎷
Web sayfanıza basit bir <audio>
öğesi ekleyin ve tarayıcının en iyi performansı sunanı seçebilmesi için birkaç medya kaynağı atayın.
<audio controls>
<source src="audio.mp3" type="audio/mp3"/>
<source src="audio.ogg" type="audio/ogg"/>
</audio>
Bildiğiniz gibi autoplay
, Android için Chrome'daki ses öğeleri için devre dışıdır. Bu nedenle, ses öğesinin play()
yöntemini kullanmamız gerekir. Bu yöntem, dokunma veya fare tıklaması gibi bir kullanıcı hareketiyle tetiklenmelidir.
Yani pointerup
, click
ve touchend
etkinliklerini dinlemek Diğer bir deyişle, web uygulamanız ses çıkarabilmek için kullanıcının bir düğmeyi tıklaması gerekir.
playButton.addEventListener('pointerup', function(event) {
let audio = document.querySelector('audio');
// User interacted with the page. Let's play audio...
audio.play()
.then(_ => { /* Set up media session... */ })
.catch(error => { console.log(error) });
});
İlk etkileşimden hemen sonra ses çalmak istemiyorsanız ses öğesinin load()
yöntemini kullanmanızı öneririz. Bu, tarayıcının kullanıcının öğeyle etkileşimde bulunup bulunmadığını izlemesinin bir yoludur. İçerik önceden yüklendiği için oynatmanın sorunsuz olmasını da sağlayabileceğini unutmayın.
let audio = document.querySelector('audio');
welcomeButton.addEventListener('pointerup', function(event) {
// User interacted with the page. Let's load audio...
<strong>audio.load()</strong>
.then(_ => { /* Show play button for instance... */ })
.catch(error => { console.log(error) });
});
// Later...
playButton.addEventListener('pointerup', function(event) {
<strong>audio.play()</strong>
.then(_ => { /* Set up media session... */ })
.catch(error => { console.log(error) });
});
Bildirimi özelleştirme
Web uygulamanız ses çalarken bildirim tepsisinde bir medya bildirimi görebilirsiniz. Android'de Chrome, belgenin başlığını ve bulabileceği en büyük simge resmini kullanarak uygun bilgileri göstermek için elinden geleni yapar.
Meta verileri ayarlama
Media Session API ile başlık, sanatçı, albüm adı ve poster gibi bazı medya oturumu meta verilerini ayarlayarak bu medya bildirimini nasıl özelleştireceğinizi görelim.
// When audio starts playing...
if ('mediaSession' in navigator) {
navigator.mediaSession.metadata = new MediaMetadata({
title: 'Never Gonna Give You Up',
artist: 'Rick Astley',
album: 'Whenever You Need Somebody',
artwork: [
{ src: 'https://dummyimage.com/96x96', sizes: '96x96', type: 'image/png' },
{ src: 'https://dummyimage.com/128x128', sizes: '128x128', type: 'image/png' },
{ src: 'https://dummyimage.com/192x192', sizes: '192x192', type: 'image/png' },
{ src: 'https://dummyimage.com/256x256', sizes: '256x256', type: 'image/png' },
{ src: 'https://dummyimage.com/384x384', sizes: '384x384', type: 'image/png' },
{ src: 'https://dummyimage.com/512x512', sizes: '512x512', type: 'image/png' },
]
});
}
Oynatma işlemi tamamlandığında bildirim otomatik olarak kaybolacağından medya oturumunu "bırakmanız" gerekmez. Oynatma işlemi başladığında mevcut navigator.mediaSession.metadata
değerinin kullanılacağını unutmayın. Bu nedenle, medya bildirimlerinde her zaman alakalı bilgiler göstermek için bu sayfayı güncellemeniz gerekir.
Önceki parça / sonraki parça
Web uygulamanızda şarkı listesi varsa kullanıcının "Önceki Parça" ve "Sonraki Parça" simgelerini kullanarak doğrudan medya bildiriminde şarkı listenizde gezinmesine izin verebilirsiniz.
let audio = document.createElement('audio');
let playlist = ['audio1.mp3', 'audio2.mp3', 'audio3.mp3'];
let index = 0;
navigator.mediaSession.setActionHandler('previoustrack', function() {
// User clicked "Previous Track" media notification icon.
index = (index - 1 + playlist.length) % playlist.length;
playAudio();
});
navigator.mediaSession.setActionHandler('nexttrack', function() {
// User clicked "Next Track" media notification icon.
index = (index + 1) % playlist.length;
playAudio();
});
function playAudio() {
audio.src = playlist[index];
audio.play()
.then(_ => { /* Set up media session... */ })
.catch(error => { console.log(error); });
}
playButton.addEventListener('pointerup', function(event) {
playAudio();
});
Medya işlemi işleyicilerinin devam edeceğini unutmayın. Bu, etkinlik dinleyici kalıbına çok benzer. Tek fark, bir etkinliğin işlenmesi durumunda tarayıcının varsayılan davranışı durdurması ve bunu web uygulamanızın medya işlemini desteklediğinin bir sinyali olarak kullanmasıdır. Bu nedenle, uygun işlem işleyiciyi ayarlamadığınız sürece medya işlemi kontrolleri gösterilmez.
Bir medya işlemi işleyicisinin ayarını kaldırmak, null
değerine atama kadar kolaydır.
Geri / ileri sarma
Media Session API, atlanan süreyi kontrol etmek istiyorsanız "Geri Sar" ve "İleri Sar" medya bildirim simgelerini göstermenize olanak tanır.
let skipTime = 10; // Time to skip in seconds
navigator.mediaSession.setActionHandler('seekbackward', function() {
// User clicked "Seek Backward" media notification icon.
audio.currentTime = Math.max(audio.currentTime - skipTime, 0);
});
navigator.mediaSession.setActionHandler('seekforward', function() {
// User clicked "Seek Forward" media notification icon.
audio.currentTime = Math.min(audio.currentTime + skipTime, audio.duration);
});
Oynat / duraklat
"Oynat/Duraklat" simgesi medya bildirimde her zaman gösterilir ve ilgili etkinlikler tarayıcı tarafından otomatik olarak yönetilir. Varsayılan davranış herhangi bir nedenle işe yaramazsa "Oynat" ve "Duraklat" medya etkinliklerini yine de işleyebilirsiniz.
navigator.mediaSession.setActionHandler('play', function() {
// User clicked "Play" media notification icon.
// Do something more than just playing current audio...
});
navigator.mediaSession.setActionHandler('pause', function() {
// User clicked "Pause" media notification icon.
// Do something more than just pausing current audio...
});
Her yerde bildirimler
Media Session API'nin en iyi özelliklerinden biri, medya meta verilerinin ve denetimlerinin yalnızca bildirim tepsisinde görünmemesidir. Medya bildirimi, eşlenen tüm giyilebilir cihazlarla otomatik olarak senkronize edilir. Ayrıca kilit ekranlarında da gösterilir.
Çevrimdışı oynarken sorun yaşamamak için
Şu anda ne düşündüğünüzü biliyorum. Hizmet çalışanı imdadınıza yetişiyor!
Doğru, ancak öncelikle bu yapılacaklar listesinde yer alan tüm öğelerin işaretlendiğinden emin olmanız gerekir:
- Tüm medya ve poster dosyaları, uygun
Cache-Control
HTTP üst bilgisiyle birlikte yayınlanır. Bu, tarayıcının daha önce getirilen kaynakları önbelleğe almasına ve yeniden kullanmasına olanak tanır. Önbelleğe alma yapılacaklar listesini inceleyin. - Tüm medya ve poster dosyalarının
Allow-Control-Allow-Origin: *
HTTP üst bilgisiyle yayınlandığından emin olun. Bu, üçüncü taraf web uygulamalarının web sunucunuzdan HTTP yanıtları almasına ve kullanmasına olanak tanır.
Hizmet çalışanı önbelleğe alma stratejisi
Medya dosyalarıyla ilgili olarak, Jake Archibald tarafından gösterildiği gibi basit bir "Önbelleğe alma, ağa geri dönme" stratejisini öneririm.
Poster için ise biraz daha ayrıntılı bir yaklaşım seçerdim:
If
posteri zaten önbellekte. Önbellekten sunun.Else
posteri ağdan getiriyorIf
getirme işlemi başarılıysa ağ posterini önbelleğe ekleyin ve yayınlayınElse
Yedek posteri önbellekten sunma
Böylece, tarayıcı posterleri getiremediğinde bile medya bildirimlerinde her zaman güzel bir poster simgesi olur. Bunu nasıl uygulayabileceğiniz aşağıda açıklanmıştır:
const FALLBACK_ARTWORK_URL = 'fallbackArtwork.png';
addEventListener('install', event => {
self.skipWaiting();
event.waitUntil(initArtworkCache());
});
function initArtworkCache() {
caches.open('artwork-cache-v1')
.then(cache => cache.add(FALLBACK_ARTWORK_URL));
}
addEventListener('fetch', event => {
if (/artwork-[0-9]+\.png$/.test(event.request.url)) {
event.respondWith(handleFetchArtwork(event.request));
}
});
function handleFetchArtwork(request) {
// Return cache request if it's in the cache already, otherwise fetch
// network artwork.
return getCacheArtwork(request)
.then(cacheResponse => cacheResponse || getNetworkArtwork(request));
}
function getCacheArtwork(request) {
return caches.open('artwork-cache-v1')
.then(cache => cache.match(request));
}
function getNetworkArtwork(request) {
// Fetch network artwork.
return fetch(request)
.then(networkResponse => {
if (networkResponse.status !== 200) {
return Promise.reject('Network artwork response is not valid');
}
// Add artwork to the cache for later use and return network response.
addArtworkToCache(request, networkResponse.clone())
return networkResponse;
})
.catch(error => {
// Return cached fallback artwork.
return getCacheArtwork(new Request(FALLBACK_ARTWORK_URL))
});
}
function addArtworkToCache(request, response) {
return caches.open('artwork-cache-v1')
.then(cache => cache.put(request, response));
}
Kullanıcının önbelleği kontrol etmesine izin verme
Kullanıcı web uygulamanızdan içerik tüketirken medya ve poster dosyaları cihazında çok fazla yer kaplıyor olabilir. Ne kadar önbelleğin kullanıldığını göstermek ve kullanıcılara önbelleği temizleme olanağı tanımak sizin sorumluluğunuzdadır. Neyse ki bunu Cache API ile yapmak oldukça kolay.
// Here's how I'd compute how much cache is used by artwork files...
caches.open('artwork-cache-v1')
.then(cache => cache.matchAll())
.then(responses => {
let cacheSize = 0;
let blobQueue = Promise.resolve();
responses.forEach(response => {
let responseSize = response.headers.get('content-length');
if (responseSize) {
// Use content-length HTTP header when possible.
cacheSize += Number(responseSize);
} else {
// Otherwise, use the uncompressed blob size.
blobQueue = blobQueue.then(_ => response.blob())
.then(blob => { cacheSize += blob.size; blob.close(); });
}
});
return blobQueue.then(_ => {
console.log('Artwork cache is about ' + cacheSize + ' Bytes.');
});
})
.catch(error => { console.log(error); });
// And here's how to delete some artwork files...
const artworkFilesToDelete = ['artwork1.png', 'artwork2.png', 'artwork3.png'];
caches.open('artwork-cache-v1')
.then(cache => Promise.all(artworkFilesToDelete.map(artwork => cache.delete(artwork))))
.catch(error => { console.log(error); });
Uygulama notları
- Android için Chrome, medya bildirimlerini göstermek için yalnızca medya dosyası süresi en az 5 saniye olduğunda "tam" ses odağını ister.
- Bildirim posterleri, blob URL'lerini ve veri URL'lerini destekler.
- Poster tanımlanmamışsa ve istenen boyutta bir simge resmi varsa medya bildirimleri bu resmi kullanır.
- Android için Chrome'da bildirim posteri boyutu
512x512
'tür. Düşük özellikli cihazlar için256x256
değeri geçerlidir. audio.src = ''
ile medya bildirimlerini kapatabilirsiniz.- Web Audio API, geçmiş nedenlerden dolayı Android ses odağını istemediğinden, Media Session API ile çalışmasını sağlamanın tek yolu, Web Audio API'ye giriş kaynağı olarak bir
<audio>
öğesi bağlamaktır. Önerilen Web AudioFocus API'nin yakın gelecekte bu durumu iyileştireceğini umuyoruz. - Medya oturumu çağrıları, yalnızca medya kaynağıyla aynı çerçeveden gelirse medya bildirimlerini etkiler. Aşağıdaki snippet'i inceleyin.
<iframe id="iframe">
<audio>...</audio>
</iframe>
<script>
iframe.contentWindow.navigator.mediaSession.metadata = new MediaMetadata({
title: 'Never Gonna Give You Up',
...
});
</script>
Destek
Bu makalenin yazıldığı sırada Media Session API'yi destekleyen tek platform Android için Chrome'dur. Tarayıcı uygulama durumuyla ilgili en güncel bilgileri Chrome Platform Durumu'nda bulabilirsiniz.
Örnekler ve demolar
Blender Foundation ve Jan Morgenstern'ın çalışmalarının yer aldığı resmi Chrome medya oturumu örneklerimize göz atın.
Kaynaklar
Medya Oturumu Spesifikasyonu: wicg.github.io/mediasession
Spesifikasyon sorunları: github.com/WICG/mediasession/issues
Chrome Hataları: crbug.com