Compare commits
10 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| cda9914d50 | |||
| 3acefbfbe8 | |||
| 4d10c5c91e | |||
| 810ff6fe85 | |||
| b2d3d940f2 | |||
| 8b1d828c59 | |||
| 824a9e5cad | |||
| 1cbc204e88 | |||
| 561c323e67 | |||
| 5c604b40c6 |
15
CHANGELOG.md
15
CHANGELOG.md
@@ -11,6 +11,21 @@ Sections:
|
|||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
|
## [0.5.9] - 2026-01-17
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- README is now collapsible with a preview by default (Show more / Show less).
|
||||||
|
- Improved mobile readability by keeping long README content contained without affecting the page layout.
|
||||||
|
|
||||||
|
## [0.5.8] - 2026-01-17
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Mobile UI layout stabilized to prevent horizontal shifting.
|
||||||
|
- README rendering no longer expands the page width on mobile devices.
|
||||||
|
- Tables and code blocks inside README now scroll within their container.
|
||||||
|
- Floating action buttons removed to avoid UI overlap on small screens.
|
||||||
|
- Header icon buttons improved for better visibility in light and dark mode.
|
||||||
|
|
||||||
## [0.5.7] - 2026-01-17
|
## [0.5.7] - 2026-01-17
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
|
|||||||
2
bcs.yaml
2
bcs.yaml
@@ -4,7 +4,7 @@ description: >
|
|||||||
Supports GitHub, GitLab, Gitea and Bahmcloud repositories with
|
Supports GitHub, GitLab, Gitea and Bahmcloud repositories with
|
||||||
a central index, UI panel and API, similar to HACS but independent.
|
a central index, UI panel and API, similar to HACS but independent.
|
||||||
|
|
||||||
category: Store
|
category: integration
|
||||||
|
|
||||||
author: Bahmcloud
|
author: Bahmcloud
|
||||||
maintainer: Bahmcloud
|
maintainer: Bahmcloud
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"domain": "bahmcloud_store",
|
"domain": "bahmcloud_store",
|
||||||
"name": "Bahmcloud Store",
|
"name": "Bahmcloud Store",
|
||||||
"version": "0.5.7",
|
"version": "0.5.9",
|
||||||
"documentation": "https://git.bahmcloud.de/bahmcloud/bahmcloud_store",
|
"documentation": "https://git.bahmcloud.de/bahmcloud/bahmcloud_store",
|
||||||
"platforms": ["update"],
|
"platforms": ["update"],
|
||||||
"requirements": [],
|
"requirements": [],
|
||||||
|
|||||||
@@ -19,12 +19,15 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
|
|
||||||
this._detailRepoId = null;
|
this._detailRepoId = null;
|
||||||
this._detailRepo = null;
|
this._detailRepo = null;
|
||||||
|
|
||||||
this._readmeLoading = false;
|
this._readmeLoading = false;
|
||||||
this._readmeText = null;
|
this._readmeText = null;
|
||||||
this._readmeHtml = null;
|
this._readmeHtml = null;
|
||||||
this._readmeError = null;
|
this._readmeError = null;
|
||||||
|
|
||||||
|
// README UX (E2)
|
||||||
|
this._readmeExpanded = false;
|
||||||
|
this._readmeCanToggle = false;
|
||||||
|
|
||||||
this._refreshing = false;
|
this._refreshing = false;
|
||||||
|
|
||||||
this._installingRepoId = null;
|
this._installingRepoId = null;
|
||||||
@@ -201,6 +204,7 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
this._readmeText = null;
|
this._readmeText = null;
|
||||||
this._readmeHtml = null;
|
this._readmeHtml = null;
|
||||||
this._readmeError = null;
|
this._readmeError = null;
|
||||||
|
this._readmeExpanded = false;
|
||||||
this._update();
|
this._update();
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@@ -265,6 +269,8 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
this._readmeText = null;
|
this._readmeText = null;
|
||||||
this._readmeHtml = null;
|
this._readmeHtml = null;
|
||||||
this._readmeError = null;
|
this._readmeError = null;
|
||||||
|
this._readmeExpanded = false;
|
||||||
|
this._readmeCanToggle = false;
|
||||||
|
|
||||||
this._update();
|
this._update();
|
||||||
this._loadReadme(repoId);
|
this._loadReadme(repoId);
|
||||||
@@ -282,15 +288,23 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
if (resp?.ok && typeof resp.readme === "string" && resp.readme.trim()) {
|
if (resp?.ok && typeof resp.readme === "string" && resp.readme.trim()) {
|
||||||
this._readmeText = resp.readme;
|
this._readmeText = resp.readme;
|
||||||
this._readmeHtml = typeof resp.html === "string" && resp.html.trim() ? resp.html : null;
|
this._readmeHtml = typeof resp.html === "string" && resp.html.trim() ? resp.html : null;
|
||||||
|
|
||||||
|
const lines = resp.readme.split(/\r?\n/).length;
|
||||||
|
this._readmeCanToggle = lines > 24 || resp.readme.length > 1600;
|
||||||
|
this._readmeExpanded = !this._readmeCanToggle;
|
||||||
} else {
|
} else {
|
||||||
this._readmeText = null;
|
this._readmeText = null;
|
||||||
this._readmeHtml = null;
|
this._readmeHtml = null;
|
||||||
this._readmeError = this._safeText(resp?.message) || "README not found.";
|
this._readmeError = this._safeText(resp?.message) || "README not found.";
|
||||||
|
this._readmeCanToggle = false;
|
||||||
|
this._readmeExpanded = false;
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
this._readmeText = null;
|
this._readmeText = null;
|
||||||
this._readmeHtml = null;
|
this._readmeHtml = null;
|
||||||
this._readmeError = e?.message ? String(e.message) : String(e);
|
this._readmeError = e?.message ? String(e.message) : String(e);
|
||||||
|
this._readmeCanToggle = false;
|
||||||
|
this._readmeExpanded = false;
|
||||||
} finally {
|
} finally {
|
||||||
this._readmeLoading = false;
|
this._readmeLoading = false;
|
||||||
this._update();
|
this._update();
|
||||||
@@ -319,8 +333,8 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
display:flex; align-items:center; justify-content:center;
|
display:flex; align-items:center; justify-content:center;
|
||||||
/* Ensure icon visibility on app header (light & dark modes) */
|
/* Ensure icon visibility on app header (light & dark modes) */
|
||||||
color: var(--app-header-text-color, var(--primary-text-color));
|
color: var(--app-header-text-color, var(--primary-text-color));
|
||||||
border: 1px solid rgba(255,255,255,0.25);
|
border: 1px solid rgba(255,255,255,0.35);
|
||||||
background: rgba(0,0,0,0.18);
|
background: rgba(255,255,255,0.16);
|
||||||
cursor:pointer; user-select:none;
|
cursor:pointer; user-select:none;
|
||||||
}
|
}
|
||||||
.iconbtn:hover{ filter:brightness(0.98); }
|
.iconbtn:hover{ filter:brightness(0.98); }
|
||||||
@@ -338,9 +352,12 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
|
|
||||||
.grid{ display:grid; gap:12px; grid-template-columns: repeat(1, minmax(0, 1fr)); }
|
.grid{ display:grid; gap:12px; grid-template-columns: repeat(1, minmax(0, 1fr)); }
|
||||||
@media (min-width: 900px){ .grid{ grid-template-columns: repeat(2, minmax(0, 1fr)); } }
|
@media (min-width: 900px){ .grid{ grid-template-columns: repeat(2, minmax(0, 1fr)); } }
|
||||||
|
/* Prevent grid children from forcing horizontal overflow (mobile) */
|
||||||
|
.grid > *{ min-width:0; }
|
||||||
|
|
||||||
.grid2{ display:grid; gap:12px; grid-template-columns: 1fr; }
|
.grid2{ display:grid; gap:12px; grid-template-columns: 1fr; }
|
||||||
@media (min-width: 1024px){ .grid2{ grid-template-columns: 1.2fr .8fr; } }
|
@media (min-width: 1024px){ .grid2{ grid-template-columns: 1.2fr .8fr; } }
|
||||||
|
.grid2 > *{ min-width:0; }
|
||||||
|
|
||||||
.card{
|
.card{
|
||||||
padding:14px 14px;
|
padding:14px 14px;
|
||||||
@@ -413,7 +430,8 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
line-height: 1.4;
|
line-height: 1.4;
|
||||||
}
|
}
|
||||||
|
|
||||||
.md{ overflow-wrap:anywhere; word-break:break-word; }
|
/* Markdown can contain very wide content (tables/images). Keep it within the viewport. */
|
||||||
|
.md{ overflow-wrap:anywhere; word-break:break-word; max-width:100%; min-width:0; overflow-x:auto; -webkit-overflow-scrolling:touch; }
|
||||||
.md :is(h1,h2,h3){ margin-top: 12px; }
|
.md :is(h1,h2,h3){ margin-top: 12px; }
|
||||||
.md img{ max-width:100%; height:auto; }
|
.md img{ max-width:100%; height:auto; }
|
||||||
.md code{
|
.md code{
|
||||||
@@ -432,15 +450,65 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
.md table{
|
.md table{
|
||||||
width:100%;
|
width:100%;
|
||||||
border-collapse: collapse;
|
border-collapse: collapse;
|
||||||
overflow:auto;
|
overflow-x:auto;
|
||||||
|
overflow-y:hidden;
|
||||||
|
-webkit-overflow-scrolling: touch;
|
||||||
display:block;
|
display:block;
|
||||||
max-width:100%;
|
max-width:100%;
|
||||||
}
|
}
|
||||||
|
.readmeWrap{
|
||||||
|
margin-top:12px;
|
||||||
|
max-width:100%;
|
||||||
|
border:1px solid var(--divider-color);
|
||||||
|
border-radius:14px;
|
||||||
|
padding:12px;
|
||||||
|
background: rgba(0,0,0,.02);
|
||||||
|
position:relative;
|
||||||
|
overflow:hidden;
|
||||||
|
}
|
||||||
|
.readmeWrap.collapsed{
|
||||||
|
max-height:260px;
|
||||||
|
}
|
||||||
|
@media (min-width: 1024px){
|
||||||
|
.readmeWrap.collapsed{ max-height:340px; }
|
||||||
|
}
|
||||||
|
.readmeWrap.collapsed::after{
|
||||||
|
content:"";
|
||||||
|
position:absolute;
|
||||||
|
left:0; right:0; bottom:0;
|
||||||
|
height:64px;
|
||||||
|
background: linear-gradient(to bottom, rgba(0,0,0,0), var(--card-background-color));
|
||||||
|
pointer-events:none;
|
||||||
|
}
|
||||||
|
.readmeWrap.expanded{
|
||||||
|
max-height:70vh;
|
||||||
|
overflow:auto;
|
||||||
|
-webkit-overflow-scrolling:touch;
|
||||||
|
}
|
||||||
|
.readmeActions{
|
||||||
|
display:flex; justify-content:flex-end;
|
||||||
|
margin-top:10px;
|
||||||
|
}
|
||||||
|
button.link{
|
||||||
|
border:1px solid transparent;
|
||||||
|
background: transparent;
|
||||||
|
color: var(--bcs-accent);
|
||||||
|
padding:8px 10px;
|
||||||
|
}
|
||||||
|
button.link:hover{
|
||||||
|
border-color: rgba(30,136,229,.25);
|
||||||
|
background: rgba(30,136,229,.06);
|
||||||
|
}
|
||||||
|
|
||||||
.md th, .md td{
|
.md th, .md td{
|
||||||
border: 1px solid var(--divider-color);
|
border: 1px solid var(--divider-color);
|
||||||
padding: 8px;
|
padding: 8px;
|
||||||
text-align:left;
|
text-align:left;
|
||||||
|
max-width:100%;
|
||||||
|
overflow-wrap:anywhere;
|
||||||
|
word-break:break-word;
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
|
|
||||||
<div class="mobilebar">
|
<div class="mobilebar">
|
||||||
@@ -726,7 +794,14 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
<div class="muted small">Rendered Markdown</div>
|
<div class="muted small">Rendered Markdown</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div id="readmePretty" class="md" style="margin-top:12px;"></div>
|
<div class="readmeWrap ${this._readmeExpanded ? "expanded" : "collapsed"}">
|
||||||
|
<div id="readmePretty" class="md"></div>
|
||||||
|
</div>
|
||||||
|
${this._readmeCanToggle ? `
|
||||||
|
<div class="readmeActions">
|
||||||
|
<button class="link" id="btnReadmeToggle">${this._readmeExpanded ? "Show less" : "Show more"}</button>
|
||||||
|
</div>
|
||||||
|
` : ``}
|
||||||
|
|
||||||
<details>
|
<details>
|
||||||
<summary>Show raw Markdown</summary>
|
<summary>Show raw Markdown</summary>
|
||||||
@@ -829,6 +904,7 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
const btnUpdate = root.getElementById("btnUpdate");
|
const btnUpdate = root.getElementById("btnUpdate");
|
||||||
const btnUninstall = root.getElementById("btnUninstall");
|
const btnUninstall = root.getElementById("btnUninstall");
|
||||||
const btnRestart = root.getElementById("btnRestart");
|
const btnRestart = root.getElementById("btnRestart");
|
||||||
|
const btnReadmeToggle = root.getElementById("btnReadmeToggle");
|
||||||
|
|
||||||
if (btnInstall) {
|
if (btnInstall) {
|
||||||
btnInstall.addEventListener("click", () => {
|
btnInstall.addEventListener("click", () => {
|
||||||
@@ -855,6 +931,13 @@ class BahmcloudStorePanel extends HTMLElement {
|
|||||||
btnRestart.addEventListener("click", () => this._restartHA());
|
btnRestart.addEventListener("click", () => this._restartHA());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (btnReadmeToggle) {
|
||||||
|
btnReadmeToggle.addEventListener("click", () => {
|
||||||
|
this._readmeExpanded = !this._readmeExpanded;
|
||||||
|
this._update();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
const mount = root.getElementById("readmePretty");
|
const mount = root.getElementById("readmePretty");
|
||||||
if (!mount) return;
|
if (!mount) return;
|
||||||
|
|
||||||
|
|||||||
@@ -1,10 +1,24 @@
|
|||||||
body { font-family: system-ui, sans-serif; margin:0; }
|
body { font-family: system-ui, sans-serif; margin:0; }
|
||||||
.wrap { padding: 16px; max-width: 1000px; margin: 0 auto; }
|
.wrap { padding: 16px; max-width: 1000px; margin: 0 auto; }
|
||||||
.card { border: 1px solid #ddd; border-radius: 10px; padding: 12px; margin: 10px 0; }
|
.card { border: 1px solid #ddd; border-radius: 10px; padding: 12px; margin: 10px 0; }
|
||||||
.row { display:flex; justify-content:space-between; gap: 12px; align-items: flex-start; }
|
.row { display:flex; justify-content:space-between; gap: 12px; align-items: flex-start; flex-wrap: wrap; min-width:0; }
|
||||||
.badge { border: 1px solid #bbb; border-radius: 999px; padding: 2px 8px; font-size: 12px; height: fit-content; }
|
.badge { border: 1px solid #bbb; border-radius: 999px; padding: 2px 8px; font-size: 12px; height: fit-content; }
|
||||||
.muted { color: #666; font-size: 13px; margin-top: 4px; }
|
.muted { color: #666; font-size: 13px; margin-top: 4px; }
|
||||||
.actions { display:flex; gap: 8px; margin-top: 10px; }
|
.actions { display:flex; gap: 8px; margin-top: 10px; }
|
||||||
button { padding: 8px 12px; cursor:pointer; }
|
button { padding: 8px 12px; cursor:pointer; }
|
||||||
button[disabled] { opacity: 0.6; cursor: not-allowed; }
|
button[disabled] { opacity: 0.6; cursor: not-allowed; }
|
||||||
a { color: inherit; }
|
a { color: inherit; }
|
||||||
|
|
||||||
|
/* Basic markdown safety (in case styles.css is used by older panels) */
|
||||||
|
.md { max-width: 100%; overflow-x: auto; }
|
||||||
|
.md table { display:block; max-width:100%; overflow-x:auto; }
|
||||||
|
.md img { max-width: 100%; height: auto; }
|
||||||
|
|
||||||
|
/* README UX (E2): collapsible preview (standalone page only) */
|
||||||
|
.readmeWrap{ border:1px solid #ddd; border-radius:10px; padding:12px; background: #f7f7f7; max-width:100%; }
|
||||||
|
.readmeWrap.collapsed{ max-height:260px; overflow:hidden; position:relative; }
|
||||||
|
.readmeWrap.collapsed::after{ content:""; position:absolute; left:0; right:0; bottom:0; height:56px; background: linear-gradient(to bottom, rgba(247,247,247,0), #f7f7f7); pointer-events:none; }
|
||||||
|
.readmeWrap.expanded{ max-height:70vh; overflow:auto; }
|
||||||
|
.readmeActions{ display:flex; justify-content:flex-end; margin-top:10px; }
|
||||||
|
button.link{ border:none; background:transparent; padding:6px 10px; color:#1E88E5; }
|
||||||
|
button.link:hover{ text-decoration:underline; }
|
||||||
|
|||||||
Reference in New Issue
Block a user