Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Maximize option to card #1688

Closed
wants to merge 17 commits into from
2 changes: 1 addition & 1 deletion panel/template/react/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,7 +67,7 @@ def _update_render_items(self, event):
if x1 is None: x1 = 12
if y0 is None: y0 = 0
if y1 is None: y1 = self.main.nrows
layouts.append({'x': x0, 'y': y0, 'w': x1-x0, 'h': y1-y0, 'i': str(i+1)})
layouts.append({'x': x0, 'y': y0, 'w': x1-x0, 'h': y1-y0, 'i': str(i+1), 'minH':4})
self._render_variables['layouts'] = {'lg': layouts, 'md': layouts}

@depends('cols', 'breakpoints', 'row_height', 'compact', watch=True)
Expand Down
6 changes: 5 additions & 1 deletion panel/template/react/dark.css
Original file line number Diff line number Diff line change
Expand Up @@ -47,6 +47,10 @@ body {
background-image: url('');
}

.maximize{
background-image: url('');
}

html {
scrollbar-face-color: #646464;
scrollbar-base-color: #646464;
Expand All @@ -64,4 +68,4 @@ html {
::-webkit-scrollbar-track-piece { background-color: #000;}
::-webkit-scrollbar-thumb { height: 50px; background-color: #666; border-radius: 3px;}
::-webkit-scrollbar-corner { background-color: #646464;}}
::-webkit-resizer { background-color: #666;}
::-webkit-resizer { background-color: #666;}
23 changes: 22 additions & 1 deletion panel/template/react/react.css
Original file line number Diff line number Diff line change
Expand Up @@ -94,7 +94,7 @@ a.navbar-brand {

p.bk.card-button {
display: none;
}
}

.sidenav {
overflow-x: hidden;
Expand Down Expand Up @@ -198,6 +198,27 @@ p.bk.card-button {
opacity: 1;
}

.maximize{
position: absolute;
top: 2px;
right: 2px;
width: 24px;
height: 24px;
z-index: 10000;
background-image: url('');
opacity: 0;
transition-delay: 0.5s;
transition: 0.5s;
}

.maximize:hover {
transition: 0.5s;
opacity: 1;
}

.maximize:focus{opacity: 1;
}

.react-resizable {
position: relative;
}
Expand Down
194 changes: 149 additions & 45 deletions panel/template/react/react.html
Original file line number Diff line number Diff line change
Expand Up @@ -126,56 +126,160 @@

<script type="text/babel">
const divStyle = {borderRadius: '5px'};
const ResponsiveGridLayout = ReactGridLayout.WidthProvider(ReactGridLayout.Responsive);

const ResponsiveGridLayout = ReactGridLayout.WidthProvider(ReactGridLayout.Responsive)

function resize_layout(obj, old_, new_, p, e, element) {
window.dispatchEvent(new Event("resize"))
}

function resize_layout_stop(obj, old_, new_, p, e, element) {
setTimeout(function(){
window.dispatchEvent(new Event("resize"))
}, 50);
}

function ResponsiveGrid(props) {
const layouts = {{layouts}};
return (
<ResponsiveGridLayout
className={"layout"}
cols={ {{cols}} }
breakpoints={ {{breakpoints}} }
draggableHandle={".drag-handle"}
onResize={resize_layout}
onResizeStop={resize_layout_stop}
draggableCancel={".bk-root"}
rowHeight={ {{rowHeight}} }
layouts={layouts}
{% if compact == "both" %}
compactType={"horizontal"}
{% elif compact == "horizontal" %}
verticalCompact={false}
compactType={"horizontal"}
{% elif compact is none %}
verticalCompact={false}
{% endif %}
>
{% set count = [] %}
{% for doc in docs %}{% for root in doc.roots %}{% if "main" in root.tags %}
<div key="{{count|length + 1 }}" style={divStyle} >
<span className="drag-handle"></span>
{{ embed(root) | indent(4) | replace("class", "className") }}
</div>
{% set __ = count.append(1) %}
{% endif %}{% endfor %}{% endfor %}
</ResponsiveGridLayout>
);
}

class ResponsiveGrid extends React.Component {
constructor(props) {
super(props);
this.state= {
layouts: {{layouts}},
resizedWidgets: {},
};
}


resize_layout(resized_layouts) {
const { layouts, resizedWidgets } = this.state;

const newLayout = layouts;
newLayout['lg']=resized_layouts;
newLayout['md']=resized_layouts
// reinitilize all widgets maximized state to false if there is some kind of manual resizing event
Object.keys(resizedWidgets).map(function(key, index) {
resizedWidgets[key] = false; });

this.setState({layouts:newLayout, });
window.dispatchEvent(new Event("resize"));
}


resize_layout_stop(resized_layouts) {
console.log(resized_layouts, window.innerWidth);
const { layouts, resizedWidgets } = this.state;

const newLayout = layouts;
newLayout['lg']=resized_layouts;
newLayout['md']=resized_layouts

var myObject = resizedWidgets;
Object.keys(myObject).map(function(key, index) {
myObject[key] = false;
});
console.log(myObject);

this.setState({layouts:newLayout, });
setTimeout(function(){
window.dispatchEvent(new Event("resize"));
}, 100);
}


toggleWidget(id){
const newState = {...this.state.resizedWidgets};
const maximized = typeof newState[id] === 'boolean' ? newState[id] : false;
newState[id] = !maximized;

this.setState({resizedWidgets: newState,}, () => {
window.dispatchEvent(new Event("resize"));
});
}


getModifiedLayouts() {
const N_COLS = {{cols}}
const ROW_HEIGHT = {{rowHeight}}
const { layouts, resizedWidgets } = this.state;

const layout_LG = layouts.lg
const layout_MD = layouts.md

const height_header = 70
const margin_rgl = 10
const N_viewport_rows = parseInt((window.innerHeight-height_header)/(ROW_HEIGHT+margin_rgl))
const N_trues = Object.values(resizedWidgets).reduce((a, item) => a + item, 0)

const newLayouts_LG = layout_LG.map(layout => {
const newLayout = { ...layout };

if (N_trues > 0) {
if (resizedWidgets[newLayout.i]) {
console.log('int', parseInt(window.innerHeight/ROW_HEIGHT));
newLayout.h = N_viewport_rows ;
newLayout.w = N_COLS['lg'];
newLayout.y= 0;
}
else {
newLayout.y= 100
};
};
return newLayout;
});

const newLayouts_MD = layout_MD.map(layout => {
const newLayout = { ...layout };
console.log('modified',newLayout);
if (N_trues > 0) {
if (resizedWidgets[newLayout.i]) {
console.log('int', parseInt(window.innerHeight/ROW_HEIGHT));
newLayout.h = N_viewport_rows ;
newLayout.w = N_COLS['md'];
newLayout.y= 0;
}
else {
newLayout.y= 100
};
};
return newLayout;
});

const newLayouts = {};
newLayouts['lg'] = newLayouts_LG
newLayouts['md'] = newLayouts_MD
return newLayouts

}


render() {
const modifiedLayouts = this.getModifiedLayouts();
return (
<ResponsiveGridLayout
className = {"layout"}
{...this.props}
cols = { {{cols}} }
breakpoints = { {{breakpoints}} }
draggableHandle = {".drag-handle"}
onResize = {this.resize_layout.bind(this)}
onResizeStop = {this.resize_layout_stop.bind(this)}
draggableCancel = {".bk-root"}
rowHeight = { {{rowHeight}} }
layouts = {modifiedLayouts}
verticalCompact = {true}
compactType = {"horizontal"}
>

{% set count = [] %}
{% for doc in docs %}{% for root in doc.roots %}{% if "main" in root.tags %}
<div key="{{count|length + 1 }}" id = {'reactcard' + "{{ count|length + 1}}"} style={divStyle} >
<span className="drag-handle"></span>
<span className="maximize" onClick={() =>
this.toggleWidget("{{count|length + 1 }}")}></span>
{{ embed(root) | indent(4) | replace("class", "className") }}
</div>
{% set __ = count.append(1) %}
{% endif %}{% endfor %}{% endfor %}

</ResponsiveGridLayout>
);
};
}

ReactDOM.render(<ResponsiveGrid />, document.getElementById('responsive-grid'))

</script>


{{ embed(roots.js_area) }}
{{ embed(roots.location) }}

Expand Down