From 9346daae15bb30890c3e6720b2721538dbed28c1 Mon Sep 17 00:00:00 2001 From: Ozan Karaali Date: Sun, 11 Dec 2022 01:35:14 +0100 Subject: [PATCH] HLS/M3U Player --- README.md | 8 ++- package.json | 4 +- src/App.jsx | 151 +++++++++++++++++++++++++++++++++++++------------- src/Modal.jsx | 16 ++++-- src/server.js | 31 ++++++++--- 5 files changed, 156 insertions(+), 54 deletions(-) diff --git a/README.md b/README.md index 6a056ec..b04cf61 100644 --- a/README.md +++ b/README.md @@ -20,10 +20,12 @@ You need to enter your IPTV provider's details to Settings. When you save, if yo Screen Shot 2021-03-04 at 00 20 09 Screen Shot 2021-03-04 at 00 20 20 -## Future Work +## Disclaimer + +This application bundles [a list of publicly available IPTV channels](https://github.com/iptv-org/iptv) from around the world. The channels are not hosted by this application or respective repository. The application simply creates a convenient way to browse a publicly available media database. The developer of this application has no affiliation with the content providers. The content provided can be removed at any time and we have no control over it. The developer assumes no liability and is not responsible for any legal issues caused by the misuse of this application. + +No video files are stored in this repository, the application bundles open-sourced [iptv-org](https://github.com/iptv-org/iptv) playlist for quick startup, users can delete that playlist entry if they want to from their computer. If any links/channels in this application infringe on your rights as a copyright holder, they may be removed by sending a [pull request](https://github.com/iptv-org/iptv/pulls) or opening an [issue](https://github.com/iptv-org/iptv/issues/new?assignees=freearhey&labels=removal+request&template=--removal-request.yml&title=Remove%3A+). However, note that we have **no control** over the destination of the link, and just removing the link from the playlist will not remove its contents from the web. Note that linking does not directly infringe copyright because no copy is made on the site providing the link, and thus this is **not** a valid reason to send a DMCA notice to GitHub. To remove this content from the web, you should contact the web host that's actually hosting the content (**not** GitHub, nor the maintainers of this repository). -- UI work will be done -- Need to check Linux HWAccels ## Contributing diff --git a/package.json b/package.json index a022859..465fa18 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "PiTV", - "version": "1.0.3-beta1", + "version": "1.1.0", "description": "A cross-platform STB (IPTV) player client.", "main": ".webpack/main", "scripts": { @@ -43,12 +43,12 @@ "bulma-prefers-dark": "^0.1.0-beta.1", "classic-level": "^1.2.0", "cors": "^2.8.5", + "electron-squirrel-startup": "^1.0.0", "express": "^4.18.2", "ffmpeg-static": "^5.1.0", "fluent-ffmpeg": "^2.1.2", "react": "^18.2.0", "react-dom": "^18.2.0", - "electron-squirrel-startup": "^1.0.0", "sass": "^1.56.1", "through2": "^4.0.2", "video.js": "^7.20.3" diff --git a/src/App.jsx b/src/App.jsx index dbb6e5d..29db73a 100644 --- a/src/App.jsx +++ b/src/App.jsx @@ -6,24 +6,21 @@ import './App.scss'; const App = () => { // stb or m3u => if true it is stb, else it is m3u - const [stb, setStb] = useState(false) const [modalState, setModalState] = useState(false) // take json state as account for the server url and mac address - const [currentStbAccount, setCurrentStbAccount] = useState({ url: "", mac: "" }) - const [stbAccounts, setStbAccounts] = useState([]) - + let [config, setConfig] = useState(undefined) + const [selected, setSelected] = useState(0) + const [reload, setReload] = useState(false) // get selected channel from channelList - const [selectedChannel, setSelectedChannel] = useState(undefined) - - const [serverUrl, setServerUrl] = useState("") - const [macAddress, setMacAddress] = useState("") - const [fullScreen, setFullScreen] = useState(true) - const [reload, setReload] = useState(true) const toggleModal = () => { setModalState(!modalState) } + useEffect(() => { + loadData(); + }, [modalState]) + useEffect(() => { loadData(); }, []) @@ -31,8 +28,8 @@ const App = () => { const loadData = async () => { const response = await fetch("http://localhost:8000/config") const data = await response.json() - setMacAddress(data.data[0].mac) - setServerUrl(data.data[0].url) + setConfig(data) + setSelected(data.selected) } const saveData = async () => { @@ -41,17 +38,9 @@ const App = () => { headers: { 'Content-Type': 'application/json', }, - body: JSON.stringify({ - "selected": 1, - "data": [ - { "type": "STB", "url": serverUrl, "mac": macAddress }, - { "type": "STB", "url": "", "mac": "00:1A:79:xx:xx:xx", "channel_list": [] }, - { "type": "M3U8", "url": "" } - ] - }) + body: JSON.stringify(config), }) const data = await response.json() - setReload(!reload) setModalState(!modalState) } @@ -59,11 +48,14 @@ const App = () => {
- + { + setReload(!reload) // TODO: better reload + } + }/>
-
@@ -76,24 +68,109 @@ const App = () => { toggleModal()} saveModal={() => saveData()} + + addServer={() => { + let configx = JSON.parse(JSON.stringify(config)); + configx.data.push({ url: "", type: "STB" }); + setSelected(configx.data.length - 1) + setConfig(configx); + }} + + deleteServer={() => { + let configx = JSON.parse(JSON.stringify(config)); + configx.selected = selected - 1; + configx.data.splice(selected, 1); + setSelected(selected - 1) + setConfig(configx); + }} + modalState={modalState} title="PiTV Settings" > -
-

Server URL (http://example.com:1234):

- { - setServerUrl(e.target.value) - }}> -
+ Servers: + { + config && config.data.map((item, index) => { + return ( +
+ +
+ ) + }) + } +

-
-

MAC Address (00:1A:79:xx:xx:xx):

- { - setMacAddress(e.target.value) - }}> -
-
-
+ Stream Type: + { + console.log(config && config.data[selected].type)}{ + config && config.data[selected].type && ["STB", "M3UPLAYLIST", "M3USTREAM"].map((item, index) => { + return ( +
+ +
+ ) + }) + } +

+ + { + config && config.data[selected] && ( +
+

Server URL (http://example.com:1234):

+ { + let configx = JSON.parse(JSON.stringify(config)); + configx.data[selected].url = e.target.value; + setConfig(configx); + } + }> +
+ ) + } + + { + config && config.data[selected].type === "STB" && ( +
+

MAC Address (00:1A:79:xx:xx:xx):

+ { + let configx = JSON.parse(JSON.stringify(config)); + configx.data[selected].mac = e.target.value; + setConfig(configx); + } + }> +
+ ) + } + + {/* { NOT YET SUPPORTED + config && config.data[selected].type === "M3UPLAYLIST" && ( +
+

File:

+ { + let configx = JSON.parse(JSON.stringify(config)) + configx.data[selected].url = e.target.files[0].name; + setConfig(configx); + }}> +
+ ) + } */} + + +
); } export default App; diff --git a/src/Modal.jsx b/src/Modal.jsx index d02e025..68b1609 100644 --- a/src/Modal.jsx +++ b/src/Modal.jsx @@ -1,6 +1,6 @@ import React from "react"; -const Modal = ({ children, closeModal, saveModal, modalState, title }) => { +const Modal = ({ children, closeModal, saveModal, modalState, title, addServer, deleteServer}) => { if (!modalState) { return null; } @@ -16,13 +16,21 @@ const Modal = ({ children, closeModal, saveModal, modalState, title }) => {
diff --git a/src/server.js b/src/server.js index bb249e5..d397be4 100644 --- a/src/server.js +++ b/src/server.js @@ -49,6 +49,10 @@ import ffmpegStatic from 'ffmpeg-static'; // "type": "M3UPLAYLIST", // "url": "https://iptv-org.github.io/iptv/index.m3u" // } +// { +// "type": "M3USTREAM", +// "url": "http://:8080/c//stream.m3u8" +// } // ] // } // type: STB, M3UPLAYLIST, M3USTREAM @@ -98,7 +102,16 @@ try { console.log("options", options) } catch (e) { - console.log("error", e) + let config = { + "selected": 0, + "data": [ + { + "type": "M3UPLAYLIST", + "url": "https://iptv-org.github.io/iptv/index.m3u" + } + ] + } + db.put('config', config); } const do_handshake = async (url, mac) => { @@ -294,27 +307,29 @@ app.get('/allChannels', async (req, res) => { // Split the text into lines const result = parseM3U(data); // Return the result array - config.data[config.selected].data = result; - await db.put('config', config); + // config.data[config.selected].data = result; + // await db.put('config', config); res.send(result); } else if (file) { const data = fs.readFileSync(file, 'utf8'); const result = parseM3U(data); - config.data[config.selected].data = result; - await db.put('config', config); + // config.data[config.selected].data = result; + // await db.put('config', config); res.send(result); } - res.send([]); + else { + res.send([]); + } } else if (type === 'M3USTREAM') { const { url } = config.data[config.selected]; if (url) { const channel = { id: 1, name: 'Stream ' + url, cmd: url }; const result = [channel]; // Return the result array - config.data[config.selected].data = result; - await db.put('config', config); + // config.data[config.selected].data = result; + // await db.put('config', config); res.send(result); }