Skip to content

Commit 6cdc617

Browse files
committed
Live URLS feature
1 parent f8f9d36 commit 6cdc617

File tree

9 files changed

+134
-13
lines changed

9 files changed

+134
-13
lines changed

docs/posts/main/configuration/client.mdx

+15
Original file line numberDiff line numberDiff line change
@@ -16,6 +16,8 @@ Before we explain all the possible options here is the client configuration Type
1616
```ts
1717
type RdtClientConfig = {
1818
position: "top-left" | "top-right" | "bottom-left" | "bottom-right" | "middle-left" | "middle-right";
19+
liveUrls: { name: string, url: string }[];
20+
liveUrlsPosition: "top-left" | "top-right" | "bottom-left" | "bottom-right";
1921
defaultOpen: boolean;
2022
expansionLevel: number;
2123
height: number;
@@ -31,6 +33,19 @@ type RdtClientConfig = {
3133
3234
Let's go through each option and see what it does.
3335
36+
## Live URLs
37+
38+
This option is used to set the live urls that will be displayed in the bottom left corner of the screen. The default value is an empty array.
39+
It allows you to specify multiple live urls that you can use to open the current page in a new tab.
40+
41+
## Live URLs position
42+
43+
This option is used to set the position of the live urls that will be displayed in the bottom left corner of the screen. The possible values are:
44+
- `top-left` - the live urls will be positioned at the top left corner of the screen
45+
- `top-right` - the live urls will be positioned at the top right corner of the screen
46+
- `bottom-left` - the live urls will be positioned at the bottom left corner of the screen
47+
- `bottom-right` - the live urls will be positioned at the bottom right corner of the screen
48+
3449
## Position
3550
3651
This option is used to set the position of the Remix Development Tools trigger (the button that opens the panel). The possible values are:

package.json

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,7 @@
22
"name": "remix-development-tools",
33
"description": "Remix development tools - a set of tools for developing/debugging Remix.run apps",
44
"author": "Alem Tuzlak",
5-
"version": "4.3.3",
5+
"version": "4.4.0",
66
"license": "MIT",
77
"keywords": [
88
"remix",

src/client/RemixDevTools.tsx

+26-3
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import { RDTContextProvider, RdtClientConfig } from "./context/RDTContext.js";
33
import { Tab } from "./tabs/index.js";
44
import { useTimelineHandler } from "./hooks/useTimelineHandler.js";
55
import { useDetachedWindowControls, usePersistOpen, useSettingsContext } from "./context/useRDTContext.js";
6-
import { useLocation } from "@remix-run/react";
6+
import { Link, useLocation } from "@remix-run/react";
77
import { Trigger } from "./components/Trigger.js";
88
import { MainPanel } from "./layout/MainPanel.js";
99
import { Tabs } from "./layout/Tabs.js";
@@ -26,6 +26,7 @@ import { useDebounce } from "./hooks/useDebounce.js";
2626
import { useListenToRouteChange } from "./hooks/detached/useListenToRouteChange.js";
2727
import { RdtPlugin } from "../index.js";
2828
import { useHotkeys } from "react-hotkeys-hook";
29+
import clsx from "clsx";
2930

3031
const recursivelyChangeTabIndex = (node: Element | HTMLElement, remove = true) => {
3132
if(remove){
@@ -38,6 +39,27 @@ const recursivelyChangeTabIndex = (node: Element | HTMLElement, remove = true) =
3839
}
3940
};
4041

42+
const LiveUrls = () =>{
43+
const { settings } = useSettingsContext();
44+
const location = useLocation();
45+
const envsPosition = settings.liveUrlsPosition;
46+
const envsClassName = {
47+
"rdt-bottom-0": envsPosition === "bottom-left" || envsPosition === "bottom-right",
48+
"rdt-top-0": envsPosition === "top-left" || envsPosition === "top-right",
49+
"rdt-right-0": envsPosition === "bottom-right" || envsPosition === "top-right",
50+
"rdt-left-0": envsPosition === "bottom-left" || envsPosition === "top-left",
51+
}
52+
if(settings.liveUrls.length === 0) return null;
53+
return <div className={clsx("rdt-flex rdt-fixed rdt-items-center rdt-gap-2 rdt-px-2", envsClassName)}>
54+
{settings.liveUrls.map((env) => {
55+
return <Link key={env.name} referrerPolicy="no-referrer" target="_blank" to={env.url+location.pathname} className="rdt-flex rdt-transition-all hover:rdt-text-black rdt-items-center rdt-gap-2 rdt-text-sm rdt-font-semibold rdt-text-gray-400">
56+
{env.name}
57+
</Link>
58+
})}
59+
</div>
60+
61+
}
62+
4163
const DevTools = ({ plugins: pluginArray }: RemixDevToolsProps) => {
4264
useTimelineHandler();
4365
useResetDetachmentCheck();
@@ -85,17 +107,18 @@ const DevTools = ({ plugins: pluginArray }: RemixDevToolsProps) => {
85107
}
86108

87109
return (
88-
<>
110+
89111
<div id={REMIX_DEV_TOOLS} className="remix-dev-tools">
90112
<Trigger isOpen={isOpen} setIsOpen={setIsOpen} />
113+
<LiveUrls />
91114
<MainPanel isOpen={isOpen}>
92115
<div className="rdt-flex rdt-h-full">
93116
<Tabs plugins={plugins} setIsOpen={setIsOpen} />
94117
<ContentPanel leftSideOriented={leftSideOriented} plugins={plugins} />
95118
</div>
96119
</MainPanel>
97120
</div>
98-
</>
121+
99122
);
100123
};
101124

src/client/components/RouteNode.tsx

+11-4
Original file line numberDiff line numberDiff line change
@@ -2,38 +2,45 @@ import clsx from "clsx";
22
import { RouteWildcards } from "../context/rdtReducer.js";
33
import { ExtendedRoute, getRouteColor } from "../utils/routing.js";
44
import { CustomNodeElementProps } from "../../external/react-d3-tree/index.js";
5+
import type { useNavigate } from "@remix-run/react";
56

67
export const RouteNode = ({
78
nodeDatum,
89
hierarchyPointNode,
910
toggleNode,
1011
setActiveRoute,
1112
activeRoutes,
13+
navigate
1214
}: CustomNodeElementProps & {
1315
routeWildcards: RouteWildcards;
1416
setActiveRoute: (e: ExtendedRoute) => void;
1517
activeRoutes: string[];
18+
navigate: ReturnType<typeof useNavigate>;
1619
}) => {
20+
1721
const parent = hierarchyPointNode.parent?.data;
1822
const parentName = parent && parent?.name !== "/" ? parent.name : "";
1923
const name = nodeDatum.name.replace(parentName, "") ?? "/";
2024
const route = { ...nodeDatum, ...nodeDatum.attributes } as any as ExtendedRoute;
2125
return (
22-
<g className="rdt-flex">
26+
<g className="rdt-flex">
2327
<circle
2428
x={20}
25-
onClick={toggleNode}
29+
onClick={toggleNode}
2630
className={clsx(
2731
getRouteColor(route),
2832
"rdt-stroke-white",
2933
nodeDatum.__rd3t.collapsed && nodeDatum.children?.length && "rdt-fill-gray-800"
3034
)}
3135
r={12}
3236
></circle>
33-
<g>
34-
<foreignObject y={-15} x={17} width={110} height={140}>
37+
<g >
38+
<foreignObject y={-15} x={17} width={110} height={140}>
3539
<p
3640
onClick={() => setActiveRoute(route)}
41+
onDoubleClickCapture={() => {
42+
navigate(route.url)
43+
}}
3744
style={{ width: 100, fontSize: 14 }}
3845
className={clsx(
3946
"rdt-w-full rdt-break-all rdt-fill-white rdt-stroke-transparent",

src/client/context/RDTContext.tsx

+4-4
Original file line numberDiff line numberDiff line change
@@ -70,22 +70,22 @@ export const getExistingStateFromStorage = (config?: RdtClientConfig) => {
7070
const settings = getSettings();
7171
const { detachedWindow, detachedWindowOwner } = detachedModeSetup();
7272
const state: RemixDevToolsState = {
73-
...initialState,
74-
73+
...initialState,
7574
...(existingState ? JSON.parse(existingState) : {}),
7675
settings: {
7776
...initialState.settings,
7877
...config,
7978
...settings,
79+
liveUrls: config?.liveUrls ?? initialState.settings.liveUrls,
8080
},
8181
detachedWindow,
8282
detachedWindowOwner,
8383
};
84-
84+
8585
return state;
8686
};
8787

88-
export type RdtClientConfig = Pick<RemixDevToolsState["settings"], "defaultOpen" | "expansionLevel" | "position" | "height" | "minHeight" | "maxHeight" | "hideUntilHover" | "panelLocation" | "requireUrlFlag" | "urlFlag" | "routeBoundaryGradient">
88+
export type RdtClientConfig = Pick<RemixDevToolsState["settings"], "defaultOpen" | "expansionLevel" | "liveUrls" | "position" | "height" | "minHeight" | "maxHeight" | "hideUntilHover" | "panelLocation" | "requireUrlFlag" | "urlFlag" | "routeBoundaryGradient">
8989

9090

9191
export const RDTContextProvider = ({ children, config }: ContextProps) => {

src/client/context/rdtReducer.ts

+56
Original file line numberDiff line numberDiff line change
@@ -69,25 +69,79 @@ export type RemixDevToolsState = {
6969
timeline: TimelineEvent[];
7070
terminals: Terminal[];
7171
settings: {
72+
/**
73+
* The live urls to show in the corner which allow you to open the app in a different environment (eg. staging, production)
74+
* @default []
75+
*/
76+
liveUrls: { url: string, name: string }[];
77+
/**
78+
* The position of the live urls
79+
* @default "bottom-left"
80+
*/
81+
liveUrlsPosition: "bottom-left" | "bottom-right" | "top-left" | "top-right";
82+
/**
83+
* The route boundary gradient color to use
84+
* @default "silver"
85+
*/
7286
routeBoundaryGradient: keyof typeof ROUTE_BOUNDARY_GRADIENTS;
7387
routeWildcards: RouteWildcards;
7488
activeTab: Tabs;
7589
shouldConnectWithForge: boolean;
7690
port: number;
7791
height: number;
92+
/**
93+
* The maximum height of the panel
94+
* @default 800
95+
*/
7896
maxHeight: number;
97+
/**
98+
* The minimum height of the panel
99+
* @default 200
100+
*/
79101
minHeight: number;
102+
/**
103+
* Whether the dev tools should be open by default
104+
* @default false
105+
*/
80106
defaultOpen: boolean;
107+
/**
108+
* Whether the dev tools trigger should be hidden until the user hovers over it
109+
* @default false
110+
*/
81111
hideUntilHover: boolean;
112+
/**
113+
* The position of the trigger button
114+
* @default "bottom-right"
115+
*/
82116
position: TriggerPosition;
117+
/**
118+
* The initial expansion level of the JSON viewer objects
119+
* @default 0
120+
*/
83121
expansionLevel: number;
84122
hoveredRoute: string;
85123
isHoveringRoute: boolean;
86124
routeViewMode: "list" | "tree";
125+
/**
126+
* The location of the panel once it is open
127+
* @default "bottom"
128+
*/
87129
panelLocation: "top" | "bottom";
88130
withServerDevTools: boolean;
131+
/**
132+
* The hotkey to open the dev tools
133+
* @default "shift+a"
134+
*/
89135
openHotkey: string;
136+
/**
137+
* Whether to require the URL flag to open the dev tools
138+
* @default false
139+
*/
90140
requireUrlFlag: boolean;
141+
/**
142+
* The URL flag to open the dev tools, used in conjunction with requireUrlFlag (if set to true)
143+
* @default "rdt"
144+
*/
91145
urlFlag: string;
92146
};
93147
htmlErrors: HTMLError[];
@@ -102,6 +156,8 @@ export const initialState: RemixDevToolsState = {
102156
terminals: [{ id: 0, locked: false, output: [], history: [] }],
103157
server: undefined,
104158
settings: {
159+
liveUrls: [],
160+
liveUrlsPosition: "bottom-left",
105161
routeBoundaryGradient: "silver",
106162
routeWildcards: {},
107163
activeTab: "page",

src/client/tabs/RoutesTab.tsx

+1
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,7 @@ const RoutesTab = () => {
5252
routeWildcards,
5353
setActiveRoute,
5454
activeRoutes,
55+
navigate
5556
})
5657
}
5758
orientation="vertical"

src/client/tabs/SettingsTab.tsx

+13
Original file line numberDiff line numberDiff line change
@@ -136,6 +136,19 @@ export const SettingsTab = () => {
136136
]}
137137
hint="This will determine where your trigger position on the screen is when the tools are collapsed."
138138
/>
139+
<SelectWithOptions
140+
label="Environments position"
141+
onSelect={(value) => setSettings({ liveUrlsPosition: value as any })}
142+
value={settings.position}
143+
className="rdt-w-full"
144+
options={[
145+
{ label: "Bottom Right", value: "bottom-right" },
146+
{ label: "Bottom Left", value: "bottom-left" },
147+
{ label: "Top Right", value: "top-right" },
148+
{ label: "Top Left", value: "top-left" },
149+
]}
150+
hint="This will determine where your environments position on the screen is."
151+
/>
139152
<SelectWithOptions
140153
label="Panel position"
141154
onSelect={(value) => setSettings({ panelLocation: value })}

src/test-apps/remix-vite/vite.config.ts

+7-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,13 @@ const config = defineRdtConfig({
88
defaultOpen: false,
99
panelLocation: "top",
1010
position: "top-right",
11-
requireUrlFlag: false
11+
requireUrlFlag: false,
12+
liveUrls: [
13+
{ url: "https://forge42.dev", name: "Production" },
14+
{
15+
url: "https://forge42.dev/staging",
16+
name: "Staging",
17+
}],
1218
},server: {
1319
silent: true,
1420
},

0 commit comments

Comments
 (0)