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.
@glue42/launchpad-ui-react
Advanced tools
This @glue42/launchpad-react-ui component library contains Glue42 enabled React components that let you create a Glue42 toolbar. It exports two main components: <Launchpad/>
and <GlobalSearch/>
. The primary purpose of a Glue42 toolbar is managing the applications, workspaces and various Glue42 Enterprise settings available to the user. In addition to that, you can create an application that implements the Search Provider
protocol and be able to send your own custom search results to the <GlobalSearch/>
component, which can do whatever you program them to do, when a user clicks on them!
Currently @glue42/launchpad-react-ui components only work with Glue42 Enterprise. Glue Desktop for Enterprise should already be running in order for the toolbar to be usable.
You need to a create a React
application using create-react-app
cli tool. Run the following command in a directory of your choice:
npx create-react-app awesome-glue-toolbar
The toolbar is a Glue Desktop Application
that is configured to run on http://localhost:3008
by default. The <Launchpad />
component is served on the URL path /launchpad
and the <GlobalSearch />
component is served on the URL path /global-search
. You must instruct create-react-app
to launch the awesome-glue-toolbar
application on port 3008. You can do that by creating a .env
file at the root of the project directory with the following text:
PORT=3008
SKIP_PREFLIGHT_CHECK=true
After setting up on which port the application should run on, install the following dependencies:
npm i -S @glue42/launchpad-ui-react @glue42/react-hooks react-router-dom
The toolbar expects that the Glue object is initialized using the @glue42/react-hooks
library. You need to import the <GlueProvider />
component from it. It uses the React Context API
to pass the initialized Glue
object down the React
tree. You need to pass the following object { appManager: 'full', layouts: 'full'
to the config
prop of the <GlueProvider />
, because they are required for proper operation.
Go to the index.js
file of the awesome-glue-toolbar
project and add the following code:
import React from 'react';
import ReactDOM from 'react-dom';
import '@glue42/launchpad-ui-react/dist/vnext.css';
import Glue from '@glue42/desktop';
import { GlueProvider } from '@glue42/react-hooks';
import { LaunchPad, GlobalSearch } from '@glue42/launchpad-ui-react';
ReactDOM.render(
<GlueProvider config={{ appManager: 'full', layouts: 'full' }} glueFactory={Glue}>
<BrowserRouter>
<Routes>
<Route path="/launchpad" element={<LaunchPad/>} />
<Route path="/global-search" element={<GlobalSearch placeholder="Search..."/>} />
</Routes>
</BrowserRouter>
</GlueProvider>,
document.getElementById('root')
);
and run the following command:
npm start
You should now be able to open Launchpad when you press the Wnd + G
keyboard combination and Global Search when you press the Ctrl + Space
keyboard combination.
Global
and Workspace
in <Launchpad/>
which are not hiddenUpon starting the application that uses the <Launchpad/>
component, it will load all of the Glue42 applications and layouts that are avaliable to the current user.
Folder names can be specified in the configuration files of different layouts and applications which Launchpad
will automatically detect and group the applications and layouts by their corresponding folders. Layouts and applications without folders will be grouped at the bottom of the list
Applications and layouts can be rearranged by setting the userProperties.appManagerOrder
in each applications config. By default each application has a sortOrder of 1000. When two applications have the same sort order, they will be ordered lexicographically. Folders inherit the minimum of the sort orders of the child applications.
At the top there is a search bar which allows for searching applications and layouts by name.
When you click on the Show layouts only
button next to the search bar, <Launchpad/>
will show you only the layouts whose name includes the search term.
You can mark every application and layout in the list as favorite by clicking on the star icon next to it's name, so that you can have easy access to it in the future. Clicking on the star icon again will unmark the given application as favorite.
Every time you click on applications and layouts they will be marked as recently used.
You can right click on items in the list of applications and layouts, on the favorite and recently used applications and layouts as well, which will bring up a context-sensitive menu. You can mark an application or layout as favorite, unmark it, and you can even delete layouts using this menu, when you click on a layout
When you have a few open applications, you can save them as a layout by pressing the keyboard combination Ctrl + S while the Launchpad part of the toolbar is open. This will bring a dialog which will prompt you to enter the layout's name and then save it when you click on the save button. If you enter a layout's name which already exists, the dialog will offer to replace the existing layout. You can close the dialog ot any time by clicking on the Cancel
button, by clicking on the close button at the top of the window, or by pressing the Escape
key.
You have a bell icon at the top left corner of the Launchpad
window which shows the current number of unread notifications. By clicking on it, you can view them in the notifications bar, which will be opened.
Next to the notifications bell, there is a hamburger menu which contains useful information and a couple of features. The information included is:
The functionality provided by the hamburger menu is:
Global Search
will make a request to all Glue 42 Search Providers
when you start typing in the search box. All the results will be displayed in a list at the bottom of the search box.
You can mark the returned results as favorites by clicking on the star icon next to their name. They will be presented in Launchpad
.
Whenever a search result is clicked or opened in any way through Global Search
, it will be marked as recently used in Launchpad
.
You can create your own Search Provider
! Global Search
follows a protocol when making requests to and getting results from search providers. The protocol uses the Glue42 Interop API
as a communication medium. If you properly implement the given protocol and supply data in the appropriate format, you can send results to Global Search
whenever someone starts searching for something!
The protocol is not limited to just returning items as search results. You can specify an action that should happen when someone clicks on the result that has been returned by your search provider! The only limitation is that the action must happen through the Glue42 Interop API
.
You can toggle the showFavoritesAndRecentsPanel
flag to show/hide the panel with the favorite and recent applications:
<LaunchPad
showFavoritesAndRecentsPanel={false}
/>
The above example will hide the favorites and recents panel.
You can toggle the showSaveButton
flag to show/hide the save button at the top of the application:
<LaunchPad
showSaveButton={false}
/>
The above example will hide the save layout button.
You can toggle the showNotificationsButton
button to show/hide the notifications button at the top of the application:
<LaunchPad
showNotificationsButton={false}
/>
The above example will hide the notifications button.
The following props are implemented using React's render prop pattern. Refer to the documentation for a thorough explanation of what it is and how it works.
You can reorder the items in the hamburger menu and you can even add you own custom elements using <LaunchPad/>
's hamburgerMenuItems
props.
import { AboutMenuItem } from '@glue42/launchpad-ui-react';
<LaunchPad
components={{
hamburgerMenuItems: [
AboutMenuItem
]
}}
/>
The code snippet above will remove all items from the hamburger menu and leave only the AboutMenuItem
, which when clicked gives information about the application.
All snippets can be imported from the @glue42/launchpad-ui-react
library. Here is a list of all the menu items that you can add to the hamburgerMenuItems
array prop:
Global Search
applicationYou can add your own custom menu items like so:
import { FeedbackMenuItem } from '@glue42/launchpad-ui-react';
<LaunchPad
components={{
hamburgerMenuItems: [
FeedbackMenuItem,
() => <p>This is my custom hamburger menu item component</p>
]
}}
/>
This will show the send feedback button and your own custom hamburger menu component below it. Note: In order to have a properly formatted menu items you need to use the <DropdownItem/>
component which we export. If you do not wrap your menu item JSX with this component, you are responsible for the proper styling of your custom menu item.
Here is how to insert a custom menu item component that is styled like the existing build-in menu item components:
import { DropdownItem } from '@glue42/launchpad-ui-react';
<LaunchPad
components={{
hamburgerMenuItems: [
() => <DropdownItem>This is FIRST custom hamburger menu item component</DropdownItem>,
() => <DropdownItem>This is SECOND custom hamburger menu item component</DropdownItem>
]
}}
/>
This will display your custom menu items with styling that makes them visually appear like the built-in menu items.
You can create custom dialog content for when the user clicks the shutdown or restart button to exit the application. The following examples will show how to create such content:
<LaunchPad
components={{
renderShutdownDialog: ({ closeDialog }) => {
return (
<div>
<p>Do you want to shutdown LaunchPad?</p>
<button onClick={() => glue.appManager.exit({showDialogs: false})}>Yes</button>
<button onClick={closeDialog}>No</button>
</div>
)
}
}}
/>
<LaunchPad
components={{
renderRestartDialog: ({ closeDialog }) => {
return (
<div>
<p>Do you want to restart LaunchPad?</p>
<button onClick={() => glue.appManager.restart()}>Yes</button>
<button onClick={closeDialog}>No</button>
</div>
)
}
}}
/>
You can add your own custom logo that will be shown on the upper right corner of the UI. You can change the logo by using the logo
property that is part of the components
prop on the <LaunchPad />
component like in the following example:
<LaunchPad
components={{
logo={() => <img src={/my/awesome/company/logo}>}
}}
/>
Additionally, you can customize how the list items on the left (the ones that show applications and layouts) are rendered by passing your own renderer to the <LaunchPad />
component like so:
<LaunchPad
components={{
renderApplicationsAndLayoutsItem={props => <p>This is my custom list item!</p>}
}}
/>
The custom component that you pass will receive the following properties from the <LaunchPad />
components:
Glue42 Applications
and Glue42 Layouts (global and workspace)
that LaunchPad
uses. It is the current app of the current row that is being rendered.number
that represents the position at which the current list item should receive focus. It is used internally by focus manager of <LaunchPad />
. In order for your renderer to participate in focus switching, you must register it with the FocusArea
React context. Otherwise, your renderer will work, but it will not receive focus. Example given below:number
that represents the nesting level of the currently rendered application. The levels start from 0, which means that the application is not in any folder, and increases by 1 for each nesting level. Currently we can't nest folders but this will probably be added in the near future. const appRef = useRef<HTMLElement | null>(null);
const { addFocusElement } = useContext(FocusAreaContext);
useEffect(() => {
if (addFocusElement && appRef && (focusOrder !== undefined)) {
addFocusElement({ focusOrder, elementRef: appRef });
}
return;
}, [addFocusElement, appRef, focusOrder]);
return (<div ref={appRef}>My custom HTML</div>)
<div
onMouseOver={() => setShowcaseApp(app)}
onMouseLeave={() => setShowcaseApp(undefined)}
...
This will display the application in the special showcase component inside of <LaunchPad />
when the user hovers over your custom list item renderer and will clear the application from the showcase component when the user's mouse leaves the div element.
<div
onClick={() => markRecentlyUsed(app)}
<LaunchPad />
to display the application in the Favorites grid to the right of the list Example:<div
onClick={() => markFavorite(app)}
<LaunchPad />
to remove the application from the favorites grid to the right of the list items. This example will unmark the application as favorite whenever the user right clicks on it:<div
onContextMenu={() => unmarkFavorite(app)}
<GlobalSearch />
. Because <LaunchPad />
and <GlobalSearch />
are used as separate applications, internally they use the Glue42 Interop API
to understand which applications have been marked as recent or favorite. Note: Every application has a 'persistable' form. You get this from when you call the getPersistableForm() method. You must send the persistable form and from each application or the Glue42 Interop API will throw an error!. Example usage:<div
onKeyDown={e => {
if (e.key === "Enter") {
markFavorite(app);
// ATTENTION: SEND THE PERSISTABLE FORM
syncFavWithLP(app.getPersistableForm());
}
}}
<div
onKeyDown={e => {
if (e.key === "Enter") {
unmarkFavorite(app);
// ATTENTION: SEND THE PERSISTABLE FORM
syncUnfavWithLP(app.getPersistableForm());
}
}}
The above examples will mark/unmark the application as favorite in <LaunchPad />
and synchronize the change with <GlobalSearch />
.
The next three methods work with the LaunchPad's context menu. You pass an application to it and the coordinates at which you want the context menu to appear at.
setIsContextMenuOpen - tell <LaunchPad />
to open the context menu.
setContextMenuCoordinates - the coordinates at which to render the context menu (usually the current mouse x and y coordinates).
setAppForContextMenu - the application which the context menu analyses and display different options based on the state of the application (hence the name "context" menu).
Example usage of all of the above mentioned methods:
<div
onContextMenu={e => {
setAppForContextMenu(app);
setContextMenuCoordinates({ x: e.pageX, y: e.pageY });
setIsContextMenuOpen(true);
}}
This example will open the context menu with the current application selected at the user's current mouse coordinates, when the user clicks on the list item with the right mouse button.
Additionally, you can customize how the favorite items are being rendered on the right (the ones that show applications and layouts) are rendered by passing your own renderer to the <LaunchPad />
component like so:
<LaunchPad
components={{
renderFavoriteItem={props => <p>This is my custom favorite item!</p>}
}}
/>
Glue42 Applications
and Glue42 Layouts (global and workspace)
that LaunchPad
uses. It is the current app of the current row that is being rendered.number
that represents the position at which the current list item should receive focus. It is used internally by focus manager of <LaunchPad />
. In order for your renderer to participate in focus switching, you must register it with the FocusArea
React context. Otherwise, your renderer will work, but it will not receive focus. Example given below: const appRef = useRef<HTMLElement | null>(null);
const { addFocusElement } = useContext(FocusAreaContext);
useEffect(() => {
if (addFocusElement && appRef && (focusOrder !== undefined)) {
addFocusElement({ focusOrder, elementRef: appRef });
}
return;
}, [addFocusElement, appRef, focusOrder]);
return (<div ref={appRef}>My custom HTML</div>)
<div
onMouseOver={() => setShowcaseApp(app)}
onMouseLeave={() => setShowcaseApp(undefined)}
...
This will display the application in the special showcase component inside of <LaunchPad />
when the user hovers over your custom list item renderer and will clear the application from the showcase component when the user's mouse leaves the div element.
<div
onClick={() => markRecentlyUsed(app)}
The next three methods work with the LaunchPad's context menu. You pass an application to it and the coordinates at which you want the context menu to appear at.
setIsContextMenuOpen - tell <LaunchPad />
to open the context menu.
setContextMenuCoordinates - the coordinates at which to render the context menu (usually the current mouse x and y coordinates).
setAppForContextMenu - the application which the context menu analyses and display different options based on the state of the application (hence the name "context" menu).
Example usage of all of the above mentioned methods:
<div
onContextMenu={e => {
setAppForContextMenu(app);
setContextMenuCoordinates({ x: e.pageX, y: e.pageY });
setIsContextMenuOpen(true);
}}
This example will open the context menu with the current application selected at the user's current mouse coordinates, when the user clicks on the list item with the right mouse button.
Additionally, you can customize how the recent items are being rendered on the right (the ones that show applications and layouts) are rendered by passing your own renderer to the <LaunchPad />
component like so:
<LaunchPad
components={{
renderRecentItem={props => <p>This is my recent item!</p>}
}}
/>
Glue42 Applications
and Glue42 Layouts (global and workspace)
that LaunchPad
uses. It is the current app of the current row that is being rendered.number
that represents the position at which the current list item should receive focus. It is used internally by focus manager of <LaunchPad />
. In order for your renderer to participate in focus switching, you must register it with the FocusArea
React context. Otherwise, your renderer will work, but it will not receive focus. Example given below: const appRef = useRef<HTMLElement | null>(null);
const { addFocusElement } = useContext(FocusAreaContext);
useEffect(() => {
if (addFocusElement && appRef && (focusOrder !== undefined)) {
addFocusElement({ focusOrder, elementRef: appRef });
}
return;
}, [addFocusElement, appRef, focusOrder]);
return (<div ref={appRef}>My custom HTML</div>)
<div
onMouseOver={() => setShowcaseApp(app)}
onMouseLeave={() => setShowcaseApp(undefined)}
...
This will display the application in the special showcase component inside of <LaunchPad />
when the user hovers over your custom list item renderer and will clear the application from the showcase component when the user's mouse leaves the div element.
<div
onClick={() => markRecentlyUsed(app)}
The next three methods work with the LaunchPad's context menu. You pass an application to it and the coordinates at which you want the context menu to appear at.
setIsContextMenuOpen - tell <LaunchPad />
to open the context menu.
setContextMenuCoordinates - the coordinates at which to render the context menu (usually the current mouse x and y coordinates).
setAppForContextMenu - the application which the context menu analyses and display different options based on the state of the application (hence the name "context" menu).
Example usage of all of the above mentioned methods:
<div
onContextMenu={e => {
setAppForContextMenu(app);
setContextMenuCoordinates({ x: e.pageX, y: e.pageY });
setIsContextMenuOpen(true);
}}
This example will open the context menu with the current application selected at the user's current mouse coordinates, when the user clicks on the list item with the right mouse button.
This custom component can be added to the top of <LaunchPad />
to display whatever you like. You will not receive any props. Note you must be aware ot the css styling of in order not to break the UI.
<LaunchPad
components={{
renderHeaderContents={() => <p>This will show at the very top of LaunchPad</p>}
}}
/>
The only thing you can currently customize in <GlobalSearch />
is the list item in the search results:
<GlobalSearch
components={{
renderGlobalSearchItem{props => <p>My custom Global Search list item</p>}
}}
/>
You will receive the following props from <GlobalSearch />
:
Glue42 Applications
and Glue42 Layouts (global and workspace)
that LaunchPad
uses. It is the current app of the current row that is being rendered.number
that represents the position at which the current list item should receive focus. It is used internally by focus manager of <GlobalSearch />
. In order for your renderer to participate in focus switching, you must register it with the FocusArea
React context. Otherwise, your renderer will work, but it will not receive focus. Example given below: const appRef = useRef<HTMLElement | null>(null);
const { addFocusElement } = useContext(FocusAreaContext);
useEffect(() => {
if (addFocusElement && appRef && (focusOrder !== undefined)) {
addFocusElement({ focusOrder, elementRef: appRef });
}
return;
}, [addFocusElement, appRef, focusOrder]);
return (<div ref={appRef}>My custom HTML</div>)
<div
onMouseOver={() => setShowcaseApp(app)}
onMouseLeave={() => setShowcaseApp(undefined)}
...
This will display the application in the special showcase component inside of <GlobalSearch />
when the user hovers over your custom list item renderer and will clear the application from the showcase component when the user's mouse leaves the div element.
<LaunchPad />
. Currently, <GlobalSearch />
does not formally keep track of the recently used applications, so you only need to sync them with the other application:<div
onClick={() => syncRecentWithLP(app)}
<GlobalSearch />
to display the application in the Favorites grid to the right of the list Example:<div
onClick={() => markFavorite(app)}
<GlobalSearch />
to remove the application from the favorites grid to the right of the list items. This example will unmark the application as favorite whenever the user right clicks on it:<div
onContextMenu={() => unmarkFavorite(app)}
<LaunchPad />
. Because <LaunchPad />
and <GlobalSearch />
are used as separate applications, internally they use the Glue42 Interop API
to understand which applications have been marked as recent or favorite. Note: Every application has a 'persistable' form. You get this from when you call the getPersistableForm() method. You must send the persistable form and from each application or the Glue42 Interop API will throw an error!. Example usage:<div
onKeyDown={e => {
if (e.key === "Enter") {
markFavorite(app);
// ATTENTION: SEND THE PERSISTABLE FORM
syncFavWithLP(app.getPersistableForm());
}
}}
<div
onKeyDown={e => {
if (e.key === "Enter") {
unmarkFavorite(app);
// ATTENTION: SEND THE PERSISTABLE FORM
syncUnfavWithLP(app.getPersistableForm());
}
}}
The above examples will mark/unmark the application as favorite in <GlobalSearch />
and synchronize the change with <LaunchPad />
.
The placeholder can be changed using the placeholder
prop.
<GlobalSearch
placeholder="Search..."
/>
FAQs
Glue42 Launchpad and GlobalSearch React Components
We found that @glue42/launchpad-ui-react demonstrated a not healthy version release cadence and project activity because the last version was released a year ago. It has 8 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.