Skip to content

Commit

Permalink
Merge pull request #47 from JurajNyiri/template_entities
Browse files Browse the repository at this point in the history
Add: Template support
  • Loading branch information
JurajNyiri authored Oct 16, 2021
2 parents 677f005 + 403a081 commit 3f48eb1
Show file tree
Hide file tree
Showing 4 changed files with 206 additions and 53 deletions.
126 changes: 101 additions & 25 deletions dist/plex-meets-homeassistant.js
Original file line number Diff line number Diff line change
Expand Up @@ -19399,7 +19399,7 @@ const isScrolledIntoView = (elem) => {
};

class PlayController {
constructor(card, hass, plex, entity, runBefore, runAfter, libraryName) {
constructor(card, hass, plex, entity, runBefore, runAfter, libraryName, entityRegistry) {
this.playButtons = [];
this.readyPlayersForType = {};
this.entityStates = {};
Expand All @@ -19409,6 +19409,7 @@ class PlayController {
this.supported = supported;
this.playActionButton = document.createElement('button');
this.playActionClickFunction = false;
this.entityRegistry = [];
this.getKodiSearchResults = async () => {
return JSON.parse((await getState(this.hass, 'sensor.kodi_media_sensor_search')).attributes.data);
};
Expand Down Expand Up @@ -19879,36 +19880,75 @@ class PlayController {
await sleep(1000);
}
};
this.exportEntity = (entityID, key) => {
const entities = [];
if (lodash.isEqual(key, 'inputSelect') || lodash.isEqual(key, 'inputText')) {
// special processing for templates
if (lodash.isArray(entityID)) {
for (let i = 0; i < entityID.length; i += 1) {
const realEntityID = lodash.get(this.entityStates[entityID[i]], 'state');
let realEntityKey = 'unknown';
lodash.forEach(this.entityRegistry, entityInRegister => {
if (lodash.isEqual(entityInRegister.entity_id, realEntityID)) {
realEntityKey = entityInRegister.platform;
}
});
entities.push({
value: realEntityID,
key: realEntityKey
});
}
}
else {
const realEntityID = lodash.get(this.entityStates[entityID], 'state');
let realEntityKey = 'unknown';
lodash.forEach(this.entityRegistry, entityInRegister => {
if (lodash.isEqual(entityInRegister.entity_id, realEntityID)) {
realEntityKey = entityInRegister.platform;
}
});
entities.push({
value: realEntityID,
key: realEntityKey
});
}
}
else if (lodash.isArray(entityID)) {
lodash.forEach(entityID, entity => {
entities.push({
value: entity,
key
});
});
}
else {
entities.push({
value: entityID,
key
});
}
return entities;
};
this.getPlayService = (data, forceRefresh = false) => {
if (!lodash.isNil(this.readyPlayersForType[data.type]) && forceRefresh === false) {
return this.readyPlayersForType[data.type];
}
let service = {};
lodash.forEach(this.entity, (value, key) => {
if (lodash.isEmpty(service)) {
const entityVal = value;
if (lodash.isArray(entityVal)) {
lodash.forEach(entityVal, entity => {
if (lodash.includes(this.supported[key], data.type)) {
if ((key === 'kodi' && this.isKodiSupported(entity)) ||
(key === 'androidtv' && this.isAndroidTVSupported(entity)) ||
(key === 'plexPlayer' && this.isPlexPlayerSupported(entity)) ||
(key === 'cast' && this.isCastSupported(entity))) {
service = { key, value: entity };
return false;
}
const entities = this.exportEntity(value, key);
lodash.forEach(entities, entity => {
if (lodash.includes(this.supported[entity.key], data.type)) {
// todo: load info in this.entityStates otherwise this will never work for input selects and templates
if ((entity.key === 'kodi' && this.isKodiSupported(entity.value)) ||
(entity.key === 'androidtv' && this.isAndroidTVSupported(entity.value)) ||
(entity.key === 'plexPlayer' && this.isPlexPlayerSupported(entity.value)) ||
(entity.key === 'cast' && this.isCastSupported(entity.value))) {
service = { key: entity.key, value: entity.value };
return false;
}
});
}
else if (lodash.includes(this.supported[key], data.type)) {
if ((key === 'kodi' && this.isKodiSupported(entityVal)) ||
(key === 'androidtv' && this.isAndroidTVSupported(entityVal)) ||
(key === 'plexPlayer' && this.isPlexPlayerSupported(entityVal)) ||
(key === 'cast' && this.isCastSupported(entityVal))) {
service = { key, value: entityVal };
return false;
}
}
});
}
});
this.readyPlayersForType[data.type] = service;
Expand Down Expand Up @@ -19989,6 +20029,23 @@ class PlayController {
catch (err) {
// pass
}
// get values for template entities
for (const [key, value] of Object.entries(this.entity)) {
if (lodash.isEqual(key, 'inputSelect') || lodash.isEqual(key, 'inputText')) {
const entities = this.exportEntity(value, key);
for (const entity of entities) {
if (!lodash.isNil(this.hass.states[entity.value])) {
try {
// eslint-disable-next-line no-await-in-loop
this.entityStates[entity.value] = await getState(this.hass, entity.value);
}
catch (err) {
// pass
}
}
}
}
}
return this.entityStates;
};
this.getPlexPlayerMachineIdentifier = (entity) => {
Expand Down Expand Up @@ -20051,6 +20108,7 @@ class PlayController {
this.entityStates[entityName].attributes.adb_response !== undefined) ||
!lodash.isEqual(this.runBefore, false));
};
this.entityRegistry = entityRegistry;
this.card = card;
this.hass = hass;
this.plex = plex;
Expand Down Expand Up @@ -20291,7 +20349,9 @@ class PlexMeetsHomeAssistantEditor extends HTMLElement {
lodash.forEach(this.entitiesRegistry, entityRegistry => {
if (lodash.isEqual(entityRegistry.platform, 'cast') ||
lodash.isEqual(entityRegistry.platform, 'kodi') ||
lodash.isEqual(entityRegistry.platform, 'androidtv')) {
lodash.isEqual(entityRegistry.platform, 'androidtv') ||
lodash.isEqual(entityRegistry.platform, 'input_select') ||
lodash.isEqual(entityRegistry.platform, 'input_text')) {
const entityName = `${entityRegistry.platform} | ${entityRegistry.entity_id}`;
entities.appendChild(addDropdownItem(entityName));
addedEntityStrings.push(entityName);
Expand Down Expand Up @@ -21859,7 +21919,9 @@ class PlexMeetsHomeAssistant extends HTMLElement {
}
else if (lodash.startsWith(entityString, 'androidtv | ') ||
lodash.startsWith(entityString, 'kodi | ') ||
lodash.startsWith(entityString, 'cast | ')) {
lodash.startsWith(entityString, 'cast | ') ||
lodash.startsWith(entityString, 'input_select | ') ||
lodash.startsWith(entityString, 'input_text | ')) {
// eslint-disable-next-line prefer-destructuring
realEntityString = entityString.split(' | ')[1];
isPlexPlayer = false;
Expand Down Expand Up @@ -21896,6 +21958,20 @@ class PlexMeetsHomeAssistant extends HTMLElement {
}
entityObj.kodi.push(entityInRegister.entity_id);
break;
case 'input_select':
if (lodash.isNil(entityObj.inputSelect)) {
// eslint-disable-next-line no-param-reassign
entityObj.inputSelect = [];
}
entityObj.inputSelect.push(entityInRegister.entity_id);
break;
case 'input_text':
if (lodash.isNil(entityObj.inputText)) {
// eslint-disable-next-line no-param-reassign
entityObj.inputText = [];
}
entityObj.inputText.push(entityInRegister.entity_id);
break;
// pass
}
}
Expand All @@ -21917,7 +21993,7 @@ class PlexMeetsHomeAssistant extends HTMLElement {
this.renderPage();
try {
if (this.plex && this.hassObj) {
this.playController = new PlayController(this, this.hassObj, this.plex, entity, this.runBefore, this.runAfter, this.config.libraryName);
this.playController = new PlayController(this, this.hassObj, this.plex, entity, this.runBefore, this.runAfter, this.config.libraryName, this.entityRegistry);
if (this.playController) {
await this.playController.init();
}
Expand Down
4 changes: 3 additions & 1 deletion src/editor.ts
Original file line number Diff line number Diff line change
Expand Up @@ -278,7 +278,9 @@ class PlexMeetsHomeAssistantEditor extends HTMLElement {
if (
_.isEqual(entityRegistry.platform, 'cast') ||
_.isEqual(entityRegistry.platform, 'kodi') ||
_.isEqual(entityRegistry.platform, 'androidtv')
_.isEqual(entityRegistry.platform, 'androidtv') ||
_.isEqual(entityRegistry.platform, 'input_select') ||
_.isEqual(entityRegistry.platform, 'input_text')
) {
const entityName = `${entityRegistry.platform} | ${entityRegistry.entity_id}`;
entities.appendChild(addDropdownItem(entityName));
Expand Down
108 changes: 83 additions & 25 deletions src/modules/PlayController.ts
Original file line number Diff line number Diff line change
Expand Up @@ -38,15 +38,19 @@ class PlayController {

card: any;

entityRegistry: Array<Record<string, any>> = [];

constructor(
card: any,
hass: HomeAssistant,
plex: Plex,
entity: Record<string, any>,
runBefore: string,
runAfter: string,
libraryName: string
libraryName: string,
entityRegistry: Array<Record<string, any>>
) {
this.entityRegistry = entityRegistry;
this.card = card;
this.hass = hass;
this.plex = plex;
Expand Down Expand Up @@ -581,39 +585,76 @@ class PlayController {
}
};

private exportEntity = (entityID: Array<any> | string, key: string): Array<Record<string, any>> => {
const entities: Array<Record<string, any>> = [];
if (_.isEqual(key, 'inputSelect') || _.isEqual(key, 'inputText')) {
// special processing for templates
if (_.isArray(entityID)) {
for (let i = 0; i < entityID.length; i += 1) {
const realEntityID = _.get(this.entityStates[entityID[i]], 'state');
let realEntityKey = 'unknown';
_.forEach(this.entityRegistry, entityInRegister => {
if (_.isEqual(entityInRegister.entity_id, realEntityID)) {
realEntityKey = entityInRegister.platform;
}
});
entities.push({
value: realEntityID,
key: realEntityKey
});
}
} else {
const realEntityID = _.get(this.entityStates[entityID], 'state');
let realEntityKey = 'unknown';
_.forEach(this.entityRegistry, entityInRegister => {
if (_.isEqual(entityInRegister.entity_id, realEntityID)) {
realEntityKey = entityInRegister.platform;
}
});
entities.push({
value: realEntityID,
key: realEntityKey
});
}
} else if (_.isArray(entityID)) {
_.forEach(entityID, entity => {
entities.push({
value: entity,
key
});
});
} else {
entities.push({
value: entityID,
key
});
}
return entities;
};

private getPlayService = (data: Record<string, any>, forceRefresh = false): Record<string, string> => {
if (!_.isNil(this.readyPlayersForType[data.type]) && forceRefresh === false) {
return this.readyPlayersForType[data.type];
}
let service: Record<string, string> = {};
_.forEach(this.entity, (value, key) => {
if (_.isEmpty(service)) {
const entityVal = value;
if (_.isArray(entityVal)) {
_.forEach(entityVal, entity => {
if (_.includes(this.supported[key], data.type)) {
if (
(key === 'kodi' && this.isKodiSupported(entity)) ||
(key === 'androidtv' && this.isAndroidTVSupported(entity)) ||
(key === 'plexPlayer' && this.isPlexPlayerSupported(entity)) ||
(key === 'cast' && this.isCastSupported(entity))
) {
service = { key, value: entity };
return false;
}
const entities = this.exportEntity(value, key);

_.forEach(entities, entity => {
if (_.includes(this.supported[entity.key], data.type)) {
// todo: load info in this.entityStates otherwise this will never work for input selects and templates
if (
(entity.key === 'kodi' && this.isKodiSupported(entity.value)) ||
(entity.key === 'androidtv' && this.isAndroidTVSupported(entity.value)) ||
(entity.key === 'plexPlayer' && this.isPlexPlayerSupported(entity.value)) ||
(entity.key === 'cast' && this.isCastSupported(entity.value))
) {
service = { key: entity.key, value: entity.value };
return false;
}
});
} else if (_.includes(this.supported[key], data.type)) {
if (
(key === 'kodi' && this.isKodiSupported(entityVal)) ||
(key === 'androidtv' && this.isAndroidTVSupported(entityVal)) ||
(key === 'plexPlayer' && this.isPlexPlayerSupported(entityVal)) ||
(key === 'cast' && this.isCastSupported(entityVal))
) {
service = { key, value: entityVal };
return false;
}
}
});
}
});
this.readyPlayersForType[data.type] = service;
Expand Down Expand Up @@ -707,6 +748,23 @@ class PlayController {
// pass
}

// get values for template entities
for (const [key, value] of Object.entries(this.entity)) {
if (_.isEqual(key, 'inputSelect') || _.isEqual(key, 'inputText')) {
const entities = this.exportEntity(value, key);
for (const entity of entities) {
if (!_.isNil(this.hass.states[entity.value])) {
try {
// eslint-disable-next-line no-await-in-loop
this.entityStates[entity.value] = await getState(this.hass, entity.value);
} catch (err) {
// pass
}
}
}
}
}

return this.entityStates;
};

Expand Down
21 changes: 19 additions & 2 deletions src/plex-meets-homeassistant.ts
Original file line number Diff line number Diff line change
Expand Up @@ -290,7 +290,9 @@ class PlexMeetsHomeAssistant extends HTMLElement {
} else if (
_.startsWith(entityString, 'androidtv | ') ||
_.startsWith(entityString, 'kodi | ') ||
_.startsWith(entityString, 'cast | ')
_.startsWith(entityString, 'cast | ') ||
_.startsWith(entityString, 'input_select | ') ||
_.startsWith(entityString, 'input_text | ')
) {
// eslint-disable-next-line prefer-destructuring
realEntityString = entityString.split(' | ')[1];
Expand Down Expand Up @@ -327,6 +329,20 @@ class PlexMeetsHomeAssistant extends HTMLElement {
}
entityObj.kodi.push(entityInRegister.entity_id);
break;
case 'input_select':
if (_.isNil(entityObj.inputSelect)) {
// eslint-disable-next-line no-param-reassign
entityObj.inputSelect = [];
}
entityObj.inputSelect.push(entityInRegister.entity_id);
break;
case 'input_text':
if (_.isNil(entityObj.inputText)) {
// eslint-disable-next-line no-param-reassign
entityObj.inputText = [];
}
entityObj.inputText.push(entityInRegister.entity_id);
break;
default:
// pass
}
Expand Down Expand Up @@ -356,7 +372,8 @@ class PlexMeetsHomeAssistant extends HTMLElement {
entity,
this.runBefore,
this.runAfter,
this.config.libraryName
this.config.libraryName,
this.entityRegistry
);
if (this.playController) {
await this.playController.init();
Expand Down

0 comments on commit 3f48eb1

Please sign in to comment.