Research
Security News
Quasar RAT Disguised as an npm Package for Detecting Vulnerabilities in Ethereum Smart Contracts
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
@lightningjs/solid
Advanced tools
Solid-Lightning is a UI framework for Lightning Renderer built with SolidJS Universal Renderer. It allows you to declaratively construct lightning nodes with reactive primitives, just as you would construct a DOM tree in SolidJS.
Clone starter template:
> npx degit lightning-js/solid-starter-template my-app
> cd my-app
> npm i # or yarn or pnpm
> npm start # or yarn or pnpm
Most of the things you do with Solid will carry over to using Solid-Lightning with some key differences as Lightning does not use HTML / DOM / CSS / Mouse Input.
import { render, Canvas, Text } from '@lightningjs/solid';
render(() => (
<Canvas>
<Text>Hello World</Text>
</Canvas>
));
The element boots up the Lightning Renderer. This should be the first component passed into the render function. It takes an options
param which is passed to the Lightning Renderer.
Everything is built with two primitive components: and . Think of like div tag for HTML, all encompassing. Whenever you want to display text, wrap it in a tag like so <Text>Hello World</Text>
import { View, Text } from '@lightningjs/solid';
<View style={OverviewContainer}>
<Text style={Title}>Hello World!</Text>
</View>;
Also included is a Row and Column component which handles key navigation between children by automatically calling setFocus on selected child.
import { Column, Row, View, Text } from '@lightningjs/solid';
<Row y={400} style={styles.Row} gap={12} justifyContent="flexStart">
<Button autofocus>TV Shows</Button>
<Button>Movies</Button>
<Button>Sports</Button>
<Button>News</Button>
</Row>;
activeElement is a global Solid Signal. At any time there is one element that can be the activeElement. You can also setActiveElement at any time to any element.
import { createEffect, on } from "solid-js";
import { activeElement, setActiveElement } from "@lightningjs/solid";
// Get notified whenever the activeElement changes
createEffect(on(activeElement, (elm) => {
focusRingRef.x = elm.x;
}, { defer: true}))
// autofocus will setActiveElement on this when intially created
<Button autofocus>TV Shows</Button>
let myButton;
onMount(() => {
setActiveElement(myButton)
//or
myButton.setFocus();
})
<Button ref={myButton}>Sports</Button>
useFocusManager
adds key handling, focusPath tracking, and focus and blur events on components. You can do this once in your App component. It returns a signal, focusPath which is an array of elements that currently have focus. When activeElement
changes, the focusPath will be recalculated. During which all elements in focus will have a focus
state added and onFocus(currentFocusedElm, prevFocusedElm) event called. Elements losing focus will have focus
state removed and onBlur(currentFocusedElm, prevFocusedElm) called.
import { useFocusManager } from "@lightningjs/solid";
const App = () => {
// Only need to do this once in Application, but you can call it anywhere
// if you need to get the focusPath signal
const focusPath = useFocusManager();
return ...
}
The calculated focusPath is used for handling key events. When a key is pressed, the Config.keyMap
looks for the keyName and corresponding value to call on${key}
then onKeyPress
on each element until one handles the event.
import { Config } from '@lightningjs/solid';
Config.keyMap.m = 'Menu';
Config.keyMap.t = 'Text';
Config.keyMap.b = 'Buttons';
<View
onText={() => navigate('/text')}
onButtons={() => navigate('/buttons')}
onMenu={() => navigate('/')}
/>;
When keys m, t, b are pressed - onMenu, onText, onButtons will be called respectively.
You can add styles to your JSX components using object notation or applying the properties directly to the JSX or via a ref:
import { createEffect, createSignal } from 'solid-js';
let columnRef;
const Column = {
display: 'flex',
flexDirection: 'column',
justifyContent: 'flexStart',
width: 1760,
height: 500,
gap: 50,
y: 200,
x: 80,
color: '00000000',
};
createEffect(() => {
columnRef.x = 200;
});
const [alpha, setAlpha] = createSignal(1);
<View ref={columnRef} alpha={alpha()} y={90} style={Column}>
// ...add some children
</View>;
The style attribute takes an object of properties and passes them to the Lightning Renderer on initial creation of the component. The style object will not be reapplied if it is changed after creation. This keeps the style object as Read Only in the templating system allowing you to use it for multiple components. Additionally, when the style object is applied any properties on the JSX will have greater precedent so you can override styles on individual componets. After the component is created, you can further change props via signals or imperatively with the ref to the component.
components require a width and height value. X and y will default to 0, 0 if not specified but are required by the renderer. component does not require any properties.
Can be HEX string ('#ffffff') or RGBA number 0x00000000 or string 'RRGGBBAA'
border
and borderRadius
are special props which create effects for the DynamicShader found in the Lightning Renderer. These props can be set on the JSX or style object. The order in which you set the props determine how they are applied in the shader. Meaning you probably want to set borderRadius first. You can also set individual borders via borderLeft
, borderRight
, borderTop
, borderBottom
. These properties do not support animations.
const style = {
borderRadius: 30,
border: { width: 10, color: '#000000' }
}
// or
const style = {
borderLeft: { width: 10, color: '#000000' },
borderRight: { width: 10, color: '#000000' }
}
At the moment there is a very barebone flex implementation (display: flex
) made for one level of children. It only supports flexDirection
, justifyContent
and gap
at the moment. But very useful for laying out rows and columns.
import { Column, Row, View, Text } from '@lightningjs/solid';
const RowStyles = {
display: 'flex',
justifyContent: 'flexStart',
width: 1760,
height: 300,
gap: 26,
y: 400,
};
<Row gap={12} style={RowStyles}>
<Button autofocus>TV Shows</Button>
<Button>Movies</Button>
<Button>Sports</Button>
<Button>News</Button>
</Row>;
Additionally, flex will automatically layout Text nodes. Anytime a View with display: flex has children which are text nodes it adds a listener for the text to load to set the width and height of the text elements and then calls updateLayout on the container to recalculate the flex layout.
<View gap={10} style={OverviewContainer}>
<Text style={Title}>{data().title || data().name}</Text>
<Text style={Overview}>{data().overview}</Text>
<View gap={8} style={SupportContainer}>
<Text style={Subline}>Support Text</Text>
<Text style={Subline}>{data().release_date}</Text>
<View width={30} height={30} src={'/assets/rt-popcorn.png'}></View>
<Text style={Subline}>90%</Text>
</View>
</View>
Adding an animate
attribute to a component will cause any property changes (after initial render) to be animated. This is useful for simple animations where you want to resize, move, or change alpha of a component. You can set animationSettings
with an object to control the duration and timing function of property changes.
createEffect(on(activeElement, (elm) => {
focusRingRef.x = elm.x;
}, { defer: true}))
<FocusRing animate animationSettings={{duration: 1500}} ref={focusRingRef} />
You can also animate elements by setting their value to an array [newValue, animationSettings]; The second param is optional and will use the animationSettings on the JSX element or Config.animationSettings.
let button;
onMount(() => {
button.alpha = [1, { duration: 500 }];
});
<Button ref={button}>Movies</Button>;
For more complicated animations, you can access the Lightning renderer animate API directly:
let button;
onMount(() => {
button.animate({ alpha: 1 }, { duration: 500 });
});
<Button ref={button}>Movies</Button>;
You can set default animation settings for all transitions globally via Config.
import { Config } from '@lightningjs/solid';
Config.animationSettings = {
duration: 250,
easing: 'ease-in-out',
};
The style object can also be used to style components based on their state. You can add any keys with states you'd like applied like so:
const Button = {
width: 386,
height: 136,
color: '#546160',
alpha: 0.5,
scale: 1,
focus: {
color: '#58807d',
scale: [1.1, { duration: 500 }],
alpha: 1,
},
disabled: {
color: '#333333',
},
};
When Button is focused, the focus styles will be applied. And when focus is removed, the original styles on the element will be set, meaning you need defaults on the original style to fallback to.
To apply a state to a component:
<Button states={{ active: true, happy: false, disabled: false }}>Movies</Button>
<Button states={'active'}>Sports</Button>
<Button states='happy'>News</Button>
Or imperatively
let myButton;
createEffect(() => {
myButton.states.add('focus');
// Check for a state
if(myButton.states.has('focus')) {
myButton.states.remove('focus')
}
myButton.states.add('disabled');
// is and has are identical
myButton.states.is('disabled');
// toggle disabled on / off
myButton.states.toggle('disabled');
})
<View ref={myButton} style={Button} />
The focus
state is automatically added and removed with the useFocusManager
primitive. Also note if elements are animating and another state is applied during the animation which uses the animated value (say alpha or color) - when that state is removed it will return to some value during the animation. Be careful not to set state with styles that are also being animated.
When you want the state to also be applied to children elements, you can add forwardStates
attribute to the parent element. Any states set on the parent will be add / removed from the children as well. This is useful for functional components where you need to change styles of children as well.
function Button(props) {
const ButtonContainer = {
width: 386,
height: 136,
color: '#000000',
alpha: 0.3,
scale: 1,
focus: {
color: ['#58807d', { duration: 2000 }],
scale: 1.2,
alpha: [1, { duration: 1500, delay: 200, timing: 'easy-in' }],
},
};
const ButtonText = {
fontSize: 32,
lineHeight: Button.height,
contain: 'width',
textAlign: 'center',
mountY: -0.35,
color: '#F6F6F9',
height: Button.height,
width: Button.width,
focus: {
color: '#FFFFFF',
},
};
return (
<View {...props} forwardStates animate style={ButtonContainer}>
<Text style={ButtonText}>{props.children}</Text>
</View>
);
}
The shader prop allows you to specify a custom shader. Most of the common use ones have shortcuts like borderRadius
, border
.
const RoundedRectangle = ['RoundedRectangle', { radius: 6 }];
function Button(props) {
return (
<View
{...props}
forwardStates
animate
style={buttonStyles.container}
shader={RoundedRectangle}
>
<View style={buttonStyles.topBar} shader={RoundedRectangle}></View>
<Text style={buttonStyles.text}>{props.children}</Text>
</View>
);
}
Allows setup of defaults for the app:
import { Config } from '@lightningjs/solid';
// Log out solid rendering information
// This is removed for Prod builds for performance
Config.debug = false;
// Set defaults for all <Text>
Config.fontSettings.fontFamily = 'Ubuntu';
Config.fontSettings.color = 0xffffffff;
Config.fontSettings.fontSize = 100;
// Set key handling map for on{Name}
Config.keyMap.m = 'Menu';
Config.keyMap.t = 'Text';
Config.keyMap.b = 'Buttons';
FAQs
Lightning renderer for solid universal
The npm package @lightningjs/solid receives a total of 16 weekly downloads. As such, @lightningjs/solid popularity was classified as not popular.
We found that @lightningjs/solid demonstrated a healthy version release cadence and project activity because the last version was released less than a year ago. It has 6 open source maintainers collaborating on the project.
Did you know?
Socket for GitHub automatically highlights issues in each pull request and monitors the health of all your open source dependencies. Discover the contents of your packages and block harmful activity before you install or update your dependencies.
Research
Security News
Socket researchers uncover a malicious npm package posing as a tool for detecting vulnerabilities in Etherium smart contracts.
Security News
Research
A supply chain attack on Rspack's npm packages injected cryptomining malware, potentially impacting thousands of developers.
Research
Security News
Socket researchers discovered a malware campaign on npm delivering the Skuld infostealer via typosquatted packages, exposing sensitive data.