/* @tldraw/editor */
.tl-container {
width: 100%;
height: 100%;
font-size: 12px;
/* Spacing */
--space-1: 2px;
--space-2: 4px;
--space-3: 8px;
--space-4: 12px;
--space-5: 16px;
--space-6: 20px;
--space-7: 28px;
--space-8: 32px;
--space-9: 64px;
--space-10: 72px;
/* Radius */
--radius-0: 2px;
--radius-1: 4px;
--radius-2: 6px;
--radius-3: 9px;
--radius-4: 11px;
/* Z Index */
--layer-background: 100;
--layer-grid: 150;
--layer-culled-shapes: 175;
--layer-canvas: 200;
--layer-shapes: 300;
--layer-overlays: 400;
--layer-in-front: 500;
--layer-following-indicator: 1000;
--layer-blocker: 10000;
/* z index for text editors */
--layer-text-container: 1;
--layer-text-content: 3;
--layer-text-editor: 4;
/* Misc */
--tl-zoom: 1;
/* Cursor SVGs */
--tl-cursor-none: none;
--tl-cursor-default: url("data:image/svg+xml,")
12 8,
default;
--tl-cursor-pointer: url("data:image/svg+xml,")
14 10,
pointer;
--tl-cursor-cross: url("data:image/svg+xml,")
16 16,
crosshair;
--tl-cursor-move: url("data:image/svg+xml,")
16 16,
move;
--tl-cursor-grab: url("data:image/svg+xml,")
16 16,
grab;
--tl-cursor-grabbing: url("data:image/svg+xml,")
16 16,
grabbing;
--tl-cursor-text: url("data:image/svg+xml,")
4 10,
text;
--tl-cursor-zoom-in: url("data:image/svg+xml,")
16 16,
zoom-in;
--tl-cursor-zoom-out: url("data:image/svg+xml,")
16 16,
zoom-out;
/* These cursor values get programmatically overridden */
/* They're just here to help your editor autocomplete */
--tl-cursor: var(--tl-cursor-default);
--tl-cursor-resize-edge: ew-resize;
--tl-cursor-resize-corner: nesw-resize;
--tl-cursor-ew-resize: ew-resize;
--tl-cursor-ns-resize: ns-resize;
--tl-cursor-nesw-resize: nesw-resize;
--tl-cursor-nwse-resize: nwse-resize;
--tl-cursor-rotate: pointer;
--tl-cursor-nwse-rotate: pointer;
--tl-cursor-nesw-rotate: pointer;
--tl-cursor-senw-rotate: pointer;
--tl-cursor-swne-rotate: pointer;
--tl-scale: calc(1 / var(--tl-zoom));
/* fonts */
--tl-font-draw: 'tldraw_draw', sans-serif;
--tl-font-sans: 'tldraw_sans', sans-serif;
--tl-font-serif: 'tldraw_serif', serif;
--tl-font-mono: 'tldraw_mono', monospace;
/* fext outline */
--a: calc(min(0.5, 1 / var(--tl-zoom)) * 2px);
--b: calc(min(0.5, 1 / var(--tl-zoom)) * -2px);
--tl-text-outline: 0 var(--b) 0 var(--color-background), 0 var(--a) 0 var(--color-background),
var(--b) var(--b) 0 var(--color-background), var(--a) var(--b) 0 var(--color-background),
var(--a) var(--a) 0 var(--color-background), var(--b) var(--a) 0 var(--color-background);
/* own properties */
position: relative;
inset: 0px;
height: 100%;
width: 100%;
overflow: clip;
}
.tl-theme__light {
--color-accent: hsl(0, 76%, 60%);
--color-background: hsl(210, 20%, 98%);
--color-brush-fill: hsl(0, 0%, 56%, 10.2%);
--color-brush-stroke: hsl(0, 0%, 56%, 25.1%);
--color-grid: hsl(0, 0%, 43%);
--color-low: hsl(204, 16%, 94%);
--color-low-border: hsl(204, 16%, 92%);
--color-culled: hsl(204, 14%, 93%);
--color-muted-none: hsl(0, 0%, 0%, 0%);
--color-muted-0: hsl(0, 0%, 0%, 2%);
--color-muted-1: hsl(0, 0%, 0%, 10%);
--color-muted-2: hsl(0, 0%, 0%, 4.3%);
--color-hint: hsl(0, 0%, 0%, 5.5%);
--color-overlay: hsl(0, 0%, 0%, 20%);
--color-divider: hsl(0, 0%, 91%);
--color-panel-contrast: hsl(0, 0%, 100%);
--color-panel-overlay: hsl(0, 0%, 100%, 82%);
--color-panel: hsl(0, 0%, 99%);
--color-focus: hsl(214, 100%, 29%);
--color-selected: hsl(214, 84%, 56%);
--color-selected-contrast: hsl(0, 0%, 100%);
--color-selection-fill: hsl(210, 100%, 56%, 24%);
--color-selection-stroke: hsl(214, 84%, 56%);
--color-text-0: hsl(0, 0%, 11%);
--color-text-1: hsl(0, 0%, 18%);
--color-text-3: hsl(220, 2%, 65%);
--color-text-shadow: hsl(0, 0%, 100%);
--color-primary: hsl(214, 84%, 56%);
--color-success: hsl(123, 46%, 34%);
--color-info: hsl(201, 98%, 41%);
--color-warning: hsl(27, 98%, 47%);
--color-error: hsl(0, 65%, 51%);
--color-warn: hsl(0, 90%, 43%);
--color-text: hsl(0, 0%, 0%);
--color-laser: hsl(0, 100%, 50%);
/* Shadows */
--shadow-1: 0px 1px 2px hsl(0, 0%, 0%, 25%), 0px 1px 3px hsl(0, 0%, 0%, 9%);
--shadow-2: 0px 0px 2px hsl(0, 0%, 0%, 16%), 0px 2px 3px hsl(0, 0%, 0%, 24%),
0px 2px 6px hsl(0, 0%, 0%, 0.1), inset 0px 0px 0px 1px var(--color-panel-contrast);
--shadow-3: 0px 1px 2px hsl(0, 0%, 0%, 28%), 0px 2px 6px hsl(0, 0%, 0%, 14%),
inset 0px 0px 0px 1px var(--color-panel-contrast);
--shadow-4: 0px 0px 3px hsl(0, 0%, 0%, 19%), 0px 5px 4px hsl(0, 0%, 0%, 16%),
0px 2px 16px hsl(0, 0%, 0%, 6%), inset 0px 0px 0px 1px var(--color-panel-contrast);
}
.tl-theme__dark {
--color-accent: hsl(0, 76%, 60%);
--color-background: hsl(240, 5%, 6.5%);
--color-brush-fill: hsl(0, 0%, 71%, 5.1%);
--color-brush-stroke: hsl(0, 0%, 71%, 25.1%);
--color-grid: hsl(0, 0%, 40%);
--color-low: hsl(260, 4.5%, 10.5%);
--color-low-border: hsl(207, 10%, 10%);
--color-culled: hsl(210, 11%, 19%);
--color-muted-none: hsl(0, 0%, 100%, 0%);
--color-muted-0: hsl(0, 0%, 100%, 2%);
--color-muted-1: hsl(0, 0%, 100%, 10%);
--color-muted-2: hsl(0, 0%, 100%, 5%);
--color-hint: hsl(0, 0%, 100%, 7%);
--color-overlay: hsl(0, 0%, 0%, 50%);
--color-divider: hsl(240, 9%, 22%);
--color-panel-contrast: hsl(245, 12%, 23%);
--color-panel: hsl(235, 6.8%, 13.5%);
--color-panel-overlay: hsl(210, 10%, 24%, 82%);
--color-focus: hsl(217, 76%, 80%);
--color-selected: hsl(217, 89%, 61%);
--color-selected-contrast: hsl(0, 0%, 100%);
--color-selection-fill: hsl(209, 100%, 57%, 20%);
--color-selection-stroke: hsl(214, 84%, 56%);
--color-text-0: hsl(0, 9%, 94%);
--color-text-1: hsl(0, 0%, 85%);
--color-text-3: hsl(210, 6%, 45%);
--color-text-shadow: hsl(210, 13%, 18%);
--color-primary: hsl(214, 84%, 56%);
--color-success: hsl(123, 38%, 57%);
--color-info: hsl(199, 92%, 56%);
--color-warning: hsl(36, 100%, 57%);
--color-error: hsl(4, 90%, 58%);
--color-warn: hsl(0, 81%, 66%);
--color-text: hsl(210, 17%, 98%);
--color-laser: hsl(0, 100%, 50%);
/* Shadows */
--shadow-1: 0px 1px 2px hsl(0, 0%, 0%, 16.1%), 0px 1px 3px hsl(0, 0%, 0%, 22%),
inset 0px 0px 0px 1px var(--color-panel-contrast);
--shadow-2: 0px 1px 3px hsl(0, 0%, 0%, 66.6%), 0px 2px 6px hsl(0, 0%, 0%, 33%),
inset 0px 0px 0px 1px var(--color-panel-contrast);
--shadow-3: 0px 1px 3px hsl(0, 0%, 0%, 50%), 0px 2px 12px hsl(0, 0%, 0%, 50%),
inset 0px 0px 0px 1px var(--color-panel-contrast);
}
.tl-counter-scaled {
transform: scale(var(--tl-scale));
transform-origin: top left;
width: calc(100% * var(--tl-zoom));
height: calc(100% * var(--tl-zoom));
}
.tl-container,
.tl-container * {
-webkit-touch-callout: none;
-webkit-tap-highlight-color: transparent;
scrollbar-highlight-color: transparent;
-webkit-user-select: none;
user-select: none;
box-sizing: border-box;
outline: none;
}
.tl-container a {
-webkit-touch-callout: initial;
}
.tl-container:focus-within {
outline: 1px solid var(--color-low);
}
input,
*[contenteditable],
*[contenteditable] * {
-webkit-user-select: text;
}
/* -------------------------------------------------- */
/* Canvas */
/* -------------------------------------------------- */
.tl-canvas {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
color: var(--color-text);
z-index: var(--layer-canvas);
cursor: var(--tl-cursor);
overflow: clip;
content-visibility: auto;
touch-action: none;
contain: strict;
}
.tl-culled-shapes {
width: 100%;
height: 100%;
z-index: var(--layer-culled-shapes);
position: absolute;
pointer-events: none;
contain: size layout;
}
.tl-culled-shapes__canvas {
width: 100%;
height: 100%;
}
.tl-shapes {
position: relative;
z-index: var(--layer-shapes);
}
.tl-overlays {
position: absolute;
top: 0px;
left: 0px;
height: 100%;
width: 100%;
contain: strict;
pointer-events: none;
z-index: var(--layer-overlays);
}
.tl-overlays__item {
position: absolute;
top: 0px;
left: 0px;
overflow: visible;
pointer-events: none;
transform-origin: top left;
}
.tl-svg-context {
position: absolute;
top: 0px;
left: 0px;
width: 100%;
height: 100%;
pointer-events: none;
}
.tl-front {
z-index: 600;
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
overflow: clip;
content-visibility: auto;
touch-action: none;
pointer-events: none;
}
/* ------------------- Background ------------------- */
.tl-background__wrapper {
z-index: var(--layer-background);
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
}
.tl-background {
background-color: var(--color-background);
width: 100%;
height: 100%;
}
/* --------------------- Grid Layer --------------------- */
.tl-grid {
position: absolute;
inset: 0px;
width: 100%;
height: 100%;
touch-action: none;
pointer-events: none;
z-index: var(--layer-grid);
contain: strict;
}
.tl-grid-dot {
fill: var(--color-grid);
}
/* --------------------- Layers --------------------- */
.tl-html-layer {
position: absolute;
top: 0px;
left: 0px;
width: 1px;
height: 1px;
contain: layout style size;
}
/* ---------------------- Brush --------------------- */
.tl-brush {
stroke-width: calc(var(--tl-scale) * 1px);
contain: size layout;
}
.tl-brush__default {
stroke: var(--color-brush-stroke);
fill: var(--color-brush-fill);
}
/* -------------------- Scribble -------------------- */
.tl-scribble {
stroke-linejoin: round;
stroke-linecap: round;
pointer-events: none;
contain: size layout;
}
/* ---------------------- Shape --------------------- */
.tl-shape {
position: absolute;
pointer-events: none;
overflow: visible;
transform-origin: top left;
contain: size layout;
}
.tl-shape__culled {
position: absolute;
pointer-events: none;
overflow: visible;
transform-origin: top left;
contain: size layout;
background-color: var(--color-culled);
z-index: 0;
}
/* ---------------- Shape Containers ---------------- */
.tl-svg-container {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
pointer-events: none;
stroke-linecap: round;
stroke-linejoin: round;
transform-origin: top left;
overflow: visible;
}
.tl-html-container {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
pointer-events: none;
stroke-linecap: round;
stroke-linejoin: round;
/* content-visibility: auto; */
transform-origin: top left;
color: inherit;
}
/* --------------- Overlay Stack --------------- */
/* back of the stack, behind user's stuff */
.tl-collaborator__scribble {
z-index: 10;
}
.tl-collaborator__brush {
z-index: 20;
}
.tl-collaborator__shape-indicator {
z-index: 30;
}
.tl-user-scribble {
z-index: 40;
}
.tl-user-brush {
z-index: 50;
}
.tl-user-indicator__selected {
z-index: 60;
}
.tl-user-indicator__hovered {
z-index: 70;
}
.tl-user-handles {
z-index: 80;
}
.tl-user-snapline {
z-index: 90;
}
.tl-selection__fg {
pointer-events: none;
z-index: 100;
}
.tl-user-indicator__hint {
z-index: 110;
stroke-width: calc(2.5px * var(--tl-scale));
}
/* behind collaborator cursor */
.tl-collaborator__cursor-hint {
z-index: 120;
}
.tl-collaborator__cursor {
z-index: 130;
}
.tl-cursor {
overflow: visible;
}
/* -------------------- Indicator ------------------- */
.tl-shape-indicator {
transform-origin: top left;
fill: none;
stroke-width: calc(1.5px * var(--tl-scale));
contain: size layout;
}
/* ------------------ SelectionBox ------------------ */
.tl-selection__bg {
position: absolute;
top: 0px;
left: 0px;
transform-origin: top left;
background-color: transparent;
pointer-events: all;
}
.tl-selection__fg__outline {
fill: none;
pointer-events: none;
stroke: var(--color-selection-stroke);
stroke-width: calc(1.5px * var(--tl-scale));
}
.tl-corner-handle {
pointer-events: none;
stroke: var(--color-selection-stroke);
fill: var(--color-background);
stroke-width: calc(1.5px * var(--tl-scale));
}
.tl-text-handle {
pointer-events: none;
fill: var(--color-selection-stroke);
}
.tl-corner-crop-handle {
pointer-events: none;
fill: none;
stroke: var(--color-selection-stroke);
}
.tl-corner-crop-edge-handle {
pointer-events: none;
fill: none;
stroke: var(--color-selection-stroke);
}
.tl-mobile-rotate__bg {
pointer-events: all;
cursor: var(--tl-cursor-grab);
}
.tl-mobile-rotate__fg {
pointer-events: none;
stroke: var(--color-selection-stroke);
fill: var(--color-background);
stroke-width: calc(1.5px * var(--tl-scale));
}
.tl-transparent {
fill: transparent;
stroke: transparent;
}
/* --------------------- Handles -------------------- */
.tl-handle {
pointer-events: all;
}
.tl-handle__bg {
fill: transparent;
stroke: transparent;
pointer-events: all;
}
.tl-handle__fg {
fill: var(--color-selected-contrast);
stroke: var(--color-selection-stroke);
stroke-width: calc(1.5px * var(--tl-scale));
pointer-events: none;
}
.tl-handle__create {
opacity: 0;
}
.tl-handle__clone > .tl-handle__fg {
fill: var(--color-selection-stroke);
stroke: none;
}
.tl-handle__bg:active {
fill: none;
}
@media (pointer: coarse) {
.tl-handle__bg:active {
fill: var(--color-selection-fill);
}
.tl-handle__create {
opacity: 1;
}
}
/* ------------------ Bounds Detail ----------------- */
.tl-image,
.tl-video {
object-fit: cover;
background-size: cover;
width: 100%;
height: 100%;
}
.tl-video-container,
.tl-image-container,
.tl-embed-container {
width: 100%;
height: 100%;
pointer-events: all;
/* background-color: var(--color-background); */
display: flex;
justify-content: center;
align-items: center;
}
.tl-image__tg {
--scale: calc(min(2, var(--tl-scale)));
position: absolute;
top: calc(var(--scale) * 8px);
right: calc(var(--scale) * 8px);
font-size: 10px;
transform-origin: top right;
background-color: var(--color-background);
padding: 2px 4px;
border-radius: var(--radius-1);
}
/* --------------------- Nametag -------------------- */
.tl-collaborator-cursor {
position: absolute;
}
.tl-nametag {
position: absolute;
top: 16px;
left: 13px;
width: fit-content;
height: fit-content;
max-width: 120px;
padding: 3px 6px;
white-space: nowrap;
position: absolute;
overflow: hidden;
text-overflow: ellipsis;
font-size: 12px;
font-family: var(--font-body);
border-radius: var(--radius-2);
color: var(--color-selected-contrast);
}
.tl-nametag-title {
position: absolute;
top: -2px;
left: 13px;
width: fit-content;
height: fit-content;
padding: 0px 6px;
max-width: 120px;
white-space: nowrap;
position: absolute;
overflow: hidden;
text-overflow: ellipsis;
font-size: 12px;
font-family: var(--font-body);
text-shadow: var(--tl-text-outline);
color: var(--color-selected-contrast);
}
.tl-nametag-chat {
position: absolute;
top: 16px;
left: 13px;
width: fit-content;
height: fit-content;
color: var(--color-selected-contrast);
white-space: nowrap;
position: absolute;
padding: 3px 6px;
font-size: 12px;
font-family: var(--font-body);
opacity: 1;
border-radius: var(--radius-2);
}
.tl-cursor-chat {
position: absolute;
color: var(--color-selected-contrast);
white-space: nowrap;
padding: 3px 6px;
font-size: 12px;
font-family: var(--font-body);
pointer-events: none;
z-index: var(--layer-cursor);
margin-top: 16px;
margin-left: 13px;
opacity: 1;
border: none;
user-select: text;
border-radius: var(--radius-2);
}
.tl-cursor-chat .tl-cursor-chat__bubble {
padding-right: 12px;
}
.tl-cursor-chat::selection {
background: var(--color-selected);
color: var(--color-selected-contrast);
text-shadow: none;
}
.tl-cursor-chat-fade {
/* Setting to zero causes it to immediately disappear */
/* Setting to near-zero causes it to fade out gradually */
opacity: 0.0001;
transition: opacity 5s ease-in-out;
}
.tl-cursor-chat::placeholder {
color: var(--color-selected-contrast);
opacity: 0.7;
}
/* -------------------------------------------------- */
/* Spinner */
/* -------------------------------------------------- */
@keyframes spinner {
to {
transform: rotate(360deg);
}
}
/* ------------------- Text Shape ------------------- */
.tl-text-shape-label {
position: relative;
font-weight: normal;
min-width: 1px;
padding: 0px;
margin: 0px;
border: none;
width: fit-content;
height: fit-content;
font-variant: normal;
font-style: normal;
pointer-events: all;
white-space: pre-wrap;
overflow-wrap: break-word;
text-shadow: var(--tl-text-outline);
}
.tl-text-wrapper[data-font='draw'] {
font-family: var(--tl-font-draw);
}
.tl-text-wrapper[data-font='sans'] {
font-family: var(--tl-font-sans);
}
.tl-text-wrapper[data-font='serif'] {
font-family: var(--tl-font-serif);
}
.tl-text-wrapper[data-font='mono'] {
font-family: var(--tl-font-mono);
}
.tl-text-wrapper[data-align='start'],
.tl-text-wrapper[data-align='start-legacy'] {
text-align: left;
}
.tl-text-wrapper[data-align='middle'],
.tl-text-wrapper[data-align='middle-legacy'] {
text-align: center;
}
.tl-text-wrapper[data-align='end'],
.tl-text-wrapper[data-align='end-legacy'] {
text-align: right;
}
.tl-text-wrapper[data-isediting='true'] .tl-text-content {
opacity: 0;
}
.tl-text {
/* remove overflow from textarea on windows */
margin: 0px;
padding: 0px;
border: 0px;
color: inherit;
caret-color: var(--color-text);
background: none;
border-image: none;
font-size: inherit;
font-family: inherit;
font-weight: inherit;
line-height: inherit;
font-variant: inherit;
font-style: inherit;
text-align: inherit;
letter-spacing: inherit;
text-shadow: inherit;
outline: none;
white-space: pre-wrap;
word-wrap: break-word;
overflow-wrap: break-word;
text-rendering: auto;
text-transform: none;
text-indent: 0px;
display: inline-block;
appearance: auto;
column-count: initial !important;
writing-mode: horizontal-tb !important;
word-spacing: 0px;
}
.tl-text-measure {
position: absolute;
z-index: -999999;
top: 0px;
left: 0px;
opacity: 0;
width: max-content;
box-sizing: border-box;
pointer-events: none;
line-break: normal;
white-space: pre-wrap;
word-wrap: break-word;
overflow-wrap: break-word;
resize: none;
border: none;
user-select: none;
contain: style paint;
-webkit-user-select: none;
}
.tl-text-input,
.tl-text-content {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
min-width: 1px;
min-height: 1px;
overflow: visible;
outline: none;
}
.tl-text-content__wrapper {
position: relative;
width: fit-content;
height: fit-content;
display: flex;
align-items: center;
justify-content: center;
pointer-events: none;
min-height: auto;
}
.tl-text-content {
pointer-events: none;
}
.tl-text-input {
resize: none;
user-select: all;
-webkit-user-select: text;
overflow: hidden;
cursor: var(--tl-cursor-text);
}
.tl-text-wrapper[data-isediting='false'] .tl-text-input,
.tl-arrow-label[data-isediting='false'] .tl-text-input {
opacity: 0;
cursor: var(--tl-cursor-default);
}
.tl-text-input::selection {
background: var(--color-selected);
color: var(--color-selected-contrast);
text-shadow: none;
}
/* ------------------- Snap Lines ------------------- */
.tl-snap-indicator {
stroke: var(--color-accent);
stroke-width: calc(1px * var(--tl-scale));
fill: none;
}
.tl-snap-point {
stroke: var(--color-accent);
stroke-width: calc(1px * var(--tl-scale));
fill: none;
}
/* -------------------- Groups ------------------ */
.tl-group {
stroke: var(--color-text);
stroke-width: calc(1px * var(--tl-scale));
opacity: 0.5;
}
/* ------------------- Bookmark Shape ------------------- */
.tl-bookmark__container {
width: 100%;
height: 100%;
position: relative;
border: 1px solid var(--color-panel-contrast);
background-color: var(--color-panel);
border-radius: var(--radius-2);
display: flex;
flex-direction: column;
overflow: hidden;
}
.tl-bookmark__image_container {
flex: 1;
overflow: hidden;
border-top-left-radius: var(--radius-1);
border-top-right-radius: var(--radius-1);
width: 100%;
height: 100%;
display: flex;
justify-content: flex-end;
align-items: flex-start;
}
.tl-bookmark__image_container > .tl-hyperlink-button::after {
background-color: var(--color-panel);
}
.tl-bookmark__placeholder {
width: 100%;
height: 100%;
background-color: var(--color-muted-2);
border-bottom: 1px solid var(--color-muted-2);
}
.tl-bookmark__image {
width: 100%;
height: 100%;
object-fit: cover;
object-position: center;
}
.tl-bookmark__copy_container {
background-color: var(--color-muted);
padding: var(--space-4);
pointer-events: all;
}
.tl-bookmark__heading,
.tl-bookmark__description,
.tl-bookmark__link {
margin: 0px;
width: 100%;
font-family: inherit;
}
.tl-bookmark__heading {
font-size: 16px;
font-weight: bold;
padding-bottom: var(--space-2);
margin: 8px 0px;
}
.tl-bookmark__description {
font-size: 12px;
padding-bottom: var(--space-4);
}
.tl-bookmark__link {
font-size: 14px;
pointer-events: all;
z-index: 999;
overflow: hidden;
display: block;
color: var(--color-text);
text-overflow: ellipsis;
text-decoration: none;
color: var(--color-text-1);
cursor: var(--tl-cursor-pointer);
}
/* ---------------- Hyperlink Button ---------------- */
.tl-hyperlink-button {
background: none;
margin: 0px;
position: absolute;
top: 0px;
right: 0px;
height: 44px;
width: 44px;
display: flex;
align-items: center;
justify-content: center;
z-index: 200;
font-size: 12px;
font-weight: 400;
color: var(--color-text-1);
padding: 13px;
cursor: var(--tl-cursor-pointer);
border: none;
outline: none;
pointer-events: all;
}
.tl-hyperlink-button::after {
content: '';
z-index: -1;
position: absolute;
right: 6px;
bottom: 6px;
display: block;
width: calc(100% - 12px);
height: calc(100% - 12px);
border-radius: var(--radius-1);
background-color: var(--color-background);
pointer-events: none;
}
.tl-hyperlink-button:focus-visible {
color: var(--color-selected);
}
.tl-hyperlink-button__icon {
width: 18px;
height: 18px;
background-color: currentColor;
pointer-events: none;
}
.tl-hyperlink-button__hidden {
display: none;
}
/* ---------------- Geo shape ---------------- */
.tl-text-label {
display: flex;
justify-content: center;
align-items: center;
color: var(--color-text);
text-shadow: var(--tl-text-outline);
line-height: inherit;
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
z-index: 10;
}
.tl-text-label[data-hastext='false'][data-isediting='false'] > .tl-text-label__inner {
width: 40px;
height: 40px;
}
.tl-text-label[data-hastext='true'][data-isediting='false'] .tl-text-content {
pointer-events: all;
}
.tl-text-wrapper .tl-text-content {
pointer-events: all;
z-index: var(--layer-text-content);
}
.tl-text-label__inner > .tl-text-content {
position: relative;
top: 0px;
left: 0px;
padding: 16px;
height: fit-content;
width: fit-content;
border-radius: var(--radius-1);
max-width: 100%;
}
.tl-text-label__inner > .tl-text-input {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
padding: 16px;
}
.tl-text-wrapper[data-isselected='true'] .tl-text-input {
z-index: var(--layer-text-editor);
pointer-events: all;
}
/* This part of the rule helps preserve the occlusion rules for the shapes so we
* don't click on shapes that are behind other shapes.
* One extra nuance is we don't use this behavior for:
* - arrows which have weird geometry and just gets in the way.
* - draw/line shapes, because it feels restrictive to have them be 'in the way' of clicking on a textfield
* - shapes that are not filled
*/
.tl-canvas:is([data-iseditinganything='true'], [data-isselectinganything='true'])
.tl-shape:not(
[data-shape-type='arrow'],
[data-shape-type='draw'],
[data-shape-type='line'],
[data-shape-type='highlight'],
[data-shape-is-filled='false']
) {
pointer-events: all;
}
.tl-text-label[data-textwrap='true'] > .tl-text-label__inner {
max-width: 100%;
}
.tl-text-label[data-isediting='true'] {
background-color: transparent;
min-height: auto;
}
.tl-text-label[data-isediting='true'] p {
opacity: 0;
}
.tl-arrow-hint {
stroke: var(--color-text-1);
fill: none;
stroke-linecap: round;
overflow: visible;
}
/* ------------------- Arrow Shape ------------------ */
.tl-arrow-label {
position: absolute;
top: -1px;
left: -1px;
width: 2px;
height: 2px;
padding: 0px;
display: flex;
justify-content: center;
align-items: center;
text-align: center;
color: var(--color-text);
text-shadow: var(--tl-text-outline);
}
.tl-arrow-label[data-isediting='true'] p {
opacity: 0;
}
.tl-arrow-label[data-isediting='true'] > .tl-arrow-label__inner {
background-color: var(--color-background);
}
.tl-arrow-label__inner {
border-radius: var(--radius-1);
box-sizing: content-box;
position: relative;
height: max-content;
width: max-content;
pointer-events: none;
display: flex;
justify-content: center;
align-items: center;
}
.tl-arrow-label .tl-arrow {
position: relative;
height: max-content;
padding: 4px;
overflow: visible;
}
.tl-arrow-label textarea {
padding: 4px;
/* Don't allow textarea to be zero width */
min-width: 4px;
}
/* -------------------- NoteShape ------------------- */
.tl-note__container {
position: relative;
width: 100%;
height: 100%;
pointer-events: all;
opacity: 1;
z-index: var(--layer-text-container);
border-radius: 1px;
}
.tl-note__container > .tl-text-label {
text-shadow: none;
color: currentColor;
}
/* --------------------- Loading -------------------- */
.tl-loading {
background-color: var(--color-background);
color: var(--color-text-1);
height: 100%;
width: 100%;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
gap: var(--space-2);
font-size: 14px;
font-weight: 500;
opacity: 0;
animation: fade-in 0.2s ease-in-out forwards;
animation-delay: 0.2s;
}
@keyframes fade-in {
0% {
opacity: 0;
}
100% {
opacity: 1;
}
}
/* -------------------- FrameShape ------------------- */
.tl-frame__body {
stroke-width: calc(1px * var(--tl-scale));
}
.tl-frame__creating {
stroke: var(--color-selected);
fill: none;
}
.tl-frame-heading {
display: flex;
align-items: center;
position: absolute;
transform-origin: 0% 100%;
overflow: hidden;
max-width: 100%;
min-width: 32px;
height: auto;
font-size: 12px;
padding-bottom: 4px;
pointer-events: all;
}
.tl-frame-heading-hit-area {
pointer-events: all;
/* scale from bottom left corner so we can pin it to the top left corner of the frame */
transform-origin: 0% 100%;
display: flex;
height: 100%;
width: 100%;
align-items: center;
border-radius: var(--radius-1);
background-color: var(--color-background);
}
.tl-frame-label {
pointer-events: all;
overflow: hidden;
text-overflow: ellipsis;
padding: var(--space-3) var(--space-3);
position: relative;
font-size: inherit;
white-space: pre;
border: 1px solid transparent;
}
.tl-frame-label__editing {
color: transparent;
white-space: pre;
width: auto;
overflow: visible;
background-color: var(--color-panel);
border-radius: var(--radius-1);
border-color: var(--color-selected);
}
.tl-frame-name-input {
position: absolute;
border: none;
background: none;
outline: none;
padding: var(--space-3) var(--space-3);
inset: 0px;
height: 100%;
width: 100%;
font-size: inherit;
font-family: inherit;
font-weight: inherit;
width: 100%;
color: var(--color-text-1);
border-radius: var(--radius-1);
user-select: all;
-webkit-user-select: text;
white-space: pre;
cursor: var(--tl-cursor-text);
}
/* If mobile use 16px as font size */
/* On iOS, font size under 16px in an input will make the page zoom into the input 🤦♂️ */
/* https://css-tricks.com/16px-or-larger-text-prevents-ios-form-zoom/ */
@media (max-width: 600px) {
.tl-frame-heading {
font-size: 16px;
}
}
/* ------------------ iFrames Detail ----------------- */
.tl-embed {
border: none;
border-radius: var(--radius-2);
}
/* -------------- Shape Error Boundary -------------- */
.tl-shape-error-boundary {
width: 100%;
height: 100%;
background-color: var(--color-muted-1);
border-width: calc(1px * var(--tl-scale));
border-color: var(--color-muted-1);
border-style: solid;
border-radius: calc(var(--radius-1) * var(--tl-scale));
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
text-align: left;
position: relative;
pointer-events: all;
overflow: hidden;
padding: var(--space-2);
}
.tl-shape-error-boundary::before {
transform: scale(var(--tl-scale));
content: 'Error';
font-size: 12px;
font-family: inherit;
color: var(--color-text-0);
}
/* ----------------- Error Boundary ----------------- */
.tl-error-boundary {
width: 100%;
height: 100%;
display: flex;
align-items: center;
justify-content: center;
padding: var(--space-4);
background-color: var(--color-background);
color: var(--color-text-1);
position: absolute;
z-index: 600;
}
.tl-error-boundary__overlay {
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
z-index: 500;
background-color: var(--color-overlay);
}
.tl-error-boundary__content * {
user-select: all;
-webkit-user-select: text;
pointer-events: all;
}
.tl-error-boundary__canvas {
pointer-events: none;
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
z-index: -1;
}
/* some browsers seem to have some weird interactions between stacking contexts
and pointer-events. this ::after pseudo element covers the canvas and prevents
it from receiving any pointer events or affecting the cursor. */
.tl-error-boundary__canvas::after {
content: ' ';
display: block;
position: absolute;
inset: 0px;
height: 100%;
width: 100%;
z-index: 600;
pointer-events: all;
}
.tl-error-boundary__content {
width: fit-content;
height: fit-content;
max-width: 100%;
width: 400px;
max-height: 100%;
background-color: var(--color-panel);
padding: 16px;
border-radius: 16px;
box-shadow: var(--shadow-2);
font-size: 14px;
font-weight: 400;
display: flex;
flex-direction: column;
overflow: auto;
z-index: 600;
gap: 12px;
}
.tl-error-boundary__content__expanded {
width: 600px;
}
.tl-error-boundary__content h2 {
font-size: 16px;
margin: 0px;
font-weight: 500;
}
.tl-error-boundary__content h4 {
border: 1px solid var(--color-low-border);
margin: -6px 0 0 0;
padding: var(--space-5);
border-radius: var(--radius-2);
font-weight: normal;
}
.tl-error-boundary__content p {
line-height: 1.5;
margin: 0px;
}
.tl-error-boundary__content pre {
background-color: var(--color-muted-2);
margin-top: 0;
padding: var(--space-5);
border-radius: var(--radius-2);
overflow: auto;
font-size: 12px;
max-height: 320px;
}
.tl-error-boundary__content button {
background: none;
border: none;
font-family: inherit;
font-size: 14px;
font-weight: 500;
padding: var(--space-4);
border-radius: var(--radius-3);
cursor: var(--tl-cursor-pointer);
color: inherit;
background-color: transparent;
}
.tl-error-boundary__content a {
color: var(--color-text-1);
font-weight: 500;
text-decoration: none;
}
.tl-error-boundary__content__error {
position: relative;
margin: -6px 0 0 0;
}
.tl-error-boundary__content__error button {
position: absolute;
top: var(--space-2);
right: var(--space-2);
font-size: 12px;
padding: var(--space-2) var(--space-3);
background-color: var(--color-panel);
border-radius: var(--radius-1);
}
.tl-error-boundary__content__actions {
display: flex;
justify-content: space-between;
gap: var(--space-4);
margin: 0px;
margin-left: -4px;
}
.tl-error-boundary__content__actions__group {
display: flex;
gap: var(--space-4);
}
.tl-error-boundary__content .tl-error-boundary__reset {
color: var(--color-warn);
}
.tl-error-boundary__content .tl-error-boundary__refresh {
background-color: var(--color-primary);
color: var(--color-selected-contrast);
}
/* --------------------- Coarse --------------------- */
.tl-hidden {
opacity: 0;
pointer-events: none;
}
.debug__ui-logger {
position: absolute;
top: 62px;
left: 16px;
color: #555;
font-size: 12px;
font-family: monospace;
}
/* ---------------- Hit test blocker ---------------- */
.tl-hit-test-blocker {
position: absolute;
z-index: var(--layer-blocker);
inset: 0px;
width: 100%;
height: 100%;
pointer-events: all;
}
.tl-hit-test-blocker__hidden {
display: none;
}
@media (hover: hover) {
.tl-handle__create:hover {
opacity: 1;
}
.tl-handle__bg:hover {
cursor: var(--tl-cursor-grab);
fill: var(--color-selection-fill);
}
.tl-bookmark__link:hover {
color: var(--color-selected);
}
.tl-hyperlink-button:hover {
color: var(--color-selected);
}
.tl-error-boundary__content button:hover {
background-color: var(--color-low);
}
.tl-error-boundary__content a:hover {
color: var(--color-text-1);
}
.tl-error-boundary__content .tl-error-boundary__refresh:hover {
background-color: var(--color-primary);
opacity: 0.9;
}
/* These three rules help preserve clicking into specific points in text areas *while*
* already in edit mode when jumping from shape to shape. */
.tl-canvas[data-iseditinganything='true'] .tl-text-wrapper:hover .tl-text-input {
z-index: var(--layer-text-editor);
pointer-events: all;
}
}