> ## Documentation Index
> Fetch the complete documentation index at: https://docs.livepeer.org/llms.txt
> Use this file to discover all available pages before exploring further.

# Lips Changelog

> Changelog of updates, improvements, and changes related to the Lips implementation

export const CustomCardTitle = ({icon, title, variant = "card", iconSize, style = {}, className = "", ...rest}) => {
  const variants = {
    card: {
      display: 'flex',
      alignItems: 'center',
      gap: "var(--lp-spacing-2)",
      marginBottom: "var(--lp-spacing-3)",
      color: 'var(--lp-color-text-primary)',
      fontSize: '1rem',
      fontWeight: 600
    },
    accordion: {
      display: 'inline-flex',
      alignItems: 'center',
      gap: "var(--lp-spacing-2)"
    },
    tab: {
      display: 'inline-flex',
      alignItems: 'center',
      gap: '0.4rem',
      fontSize: '0.875rem'
    }
  };
  const sizes = {
    card: 20,
    accordion: 18,
    tab: 14
  };
  const size = iconSize || sizes[variant] || 20;
  const baseStyle = variants[variant] || variants.card;
  return variant === 'card' ? <div className={className} style={{
    ...baseStyle,
    ...style
  }} {...rest}>
      {typeof icon === 'string' ? <Icon icon={icon} size={size} color="var(--lp-color-accent)" /> : icon}
      {title}
    </div> : <span className={className} style={{
    ...baseStyle,
    ...style
  }} {...rest}>
      {typeof icon === 'string' ? <Icon icon={icon} size={size} color="var(--lp-color-accent)" /> : icon}
      {title}
    </span>;
};

export const Subtitle = ({style = {}, text, children, variant = 'default', fontSize = '', fontWeight = '', fontStyle = '', marginTop = '', marginBottom = '', color = '', className = '', ...rest}) => {
  const renderInlineCode = (value, keyPrefix) => {
    return value.split(/(`[^`]+`)/g).map((segment, index) => {
      if (segment.startsWith('`') && segment.endsWith('`')) {
        return <code key={`${keyPrefix}-code-${index}`}>{segment.slice(1, -1)}</code>;
      }
      return segment;
    });
  };
  const renderInlineMarkup = (value, keyPrefix = 'subtitle') => {
    if (typeof value !== 'string') {
      return value;
    }
    return value.split(/(\*\*[\s\S]+?\*\*)/g).map((segment, index) => {
      if (segment.startsWith('**') && segment.endsWith('**')) {
        const inner = segment.slice(2, -2);
        return <strong key={`${keyPrefix}-strong-${index}`}>
            {renderInlineCode(inner, `${keyPrefix}-strong-${index}`)}
          </strong>;
      }
      return renderInlineCode(segment, `${keyPrefix}-${index}`);
    });
  };
  const renderContent = (value, keyPrefix) => {
    if (Array.isArray(value)) {
      return value.map((item, index) => renderContent(item, `${keyPrefix}-${index}`));
    }
    return renderInlineMarkup(value, keyPrefix);
  };
  const variants = {
    default: {
      fontSize: '1rem',
      fontStyle: 'italic',
      color: 'var(--lp-color-accent)',
      marginBottom: 0
    },
    changelog: {
      fontSize: '0.8rem',
      fontStyle: 'normal',
      fontWeight: 700,
      color: 'var(--lp-color-text-primary)',
      marginBottom: 0
    }
  };
  const base = variants[variant] || variants.default;
  return <span className={className} style={{
    ...base,
    ...fontSize ? {
      fontSize
    } : {},
    ...fontWeight ? {
      fontWeight
    } : {},
    ...fontStyle ? {
      fontStyle
    } : {},
    ...marginTop ? {
      marginTop
    } : {},
    ...marginBottom ? {
      marginBottom
    } : {},
    ...color ? {
      color
    } : {},
    ...style
  }} {...rest}>
      {renderContent(text, 'text')}
      {renderContent(children, 'children')}
    </span>;
};

export const LazyLoad = ({children, height = "200px", offset = "200px", fadeDuration = 400, className = "", style = {}, ...rest}) => {
  const ref = useRef(null);
  const [visible, setVisible] = useState(false);
  const [ready, setReady] = useState(false);
  useEffect(() => {
    const el = ref.current;
    if (!el) return;
    const observer = new IntersectionObserver(([entry]) => {
      if (entry.isIntersecting) {
        setVisible(true);
        observer.disconnect();
      }
    }, {
      rootMargin: offset
    });
    observer.observe(el);
    return () => observer.disconnect();
  }, []);
  useEffect(() => {
    if (!visible) return;
    const frameId = requestAnimationFrame(() => {
      setReady(true);
    });
    return () => cancelAnimationFrame(frameId);
  }, [visible]);
  const placeholder = <div ref={ref} className={className} style={{
    minHeight: height,
    ...style
  }} {...rest} />;
  if (!visible) return placeholder;
  return <div ref={ref} className={className} style={{
    opacity: ready ? 1 : 0,
    transition: `opacity ${fadeDuration}ms ease-in`,
    ...style
  }} {...rest}>
      {children}
    </div>;
};

export const ScrollBox = ({children, maxHeight = 300, showHint = true, ariaLabel = "Scrollable content", style = {}, className = "", ...rest}) => {
  const contentRef = useRef(null);
  const [isOverflowing, setIsOverflowing] = useState(false);
  useEffect(() => {
    const checkOverflow = () => {
      if (contentRef.current) {
        const maxHeightPx = typeof maxHeight === "number" ? maxHeight : parseInt(maxHeight, 10) || 300;
        setIsOverflowing(contentRef.current.scrollHeight > maxHeightPx);
      }
    };
    checkOverflow();
    window.addEventListener("resize", checkOverflow);
    return () => window.removeEventListener("resize", checkOverflow);
  }, [maxHeight, children]);
  return <div className={className} style={{
    position: "relative",
    ...style
  }} {...rest}>
      <div ref={contentRef} role="region" tabIndex={0} aria-label={ariaLabel} style={{
    maxHeight: typeof maxHeight === "number" ? `${maxHeight}px` : maxHeight,
    overflowY: "auto",
    paddingRight: 4
  }} onScroll={e => {
    const el = e.target;
    const atBottom = el.scrollHeight - el.scrollTop <= el.clientHeight + 10;
    const hint = el.parentNode.querySelector("[data-scroll-hint]");
    if (hint) hint.style.opacity = atBottom ? "0" : "1";
  }}>
        {children}
      </div>
      {showHint && isOverflowing && <div data-scroll-hint style={{
    fontSize: 11,
    color: "var(--lp-color-text-muted)",
    textAlign: "center",
    marginTop: 8,
    transition: "opacity 0.2s"
  }}>
          Scroll for more ↓
        </div>}
    </div>;
};

export const DoubleIconLink = ({label = '', labelColor, href = '#', text = '', iconLeft = 'github', iconLeftColor, iconRight = 'arrow-up-right', iconRightColor = 'var(--lp-color-accent)', className = '', style = {}, ...rest}) => {
  return <span className={className} style={{
    whiteSpace: 'nowrap',
    display: 'inline-flex',
    alignItems: 'center',
    gap: "var(--lp-spacing-1)",
    marginLeft: '0.3rem',
    ...style
  }} {...rest}>
      {text && <span style={{
    marginRight: 8
  }}>{text}</span>}
      <Icon icon={iconLeft} color={iconLeftColor} />
      <a href={href} style={{
    color: {
      labelColor
    }
  }}>
        {label}
      </a>
      <div style={{
    marginRight: '0.3rem'
  }}>
        <Icon icon={iconRight} size={12} color={iconRightColor} />
      </div>
    </span>;
};

export const LinkArrow = ({href, label, description, newline = true, borderColor, className = '', style = {}, ...rest}) => {
  const linkArrowStyle = {
    display: 'inline-flex',
    alignItems: 'center',
    justifyContent: 'center',
    gap: "var(--lp-spacing-1)",
    width: 'fit-content',
    ...borderColor && ({
      borderColor
    })
  };
  return <span className={className} style={style} {...rest}>
      {newline && <br />}
      <span style={linkArrowStyle}>
        <a href={href} target="_blank" rel="noopener noreferrer">
          {label}
        </a>
        <Icon icon="arrow-up-right" size={14} color="var(--lp-color-accent)" />
      </span>
      {description && description}
      {description && <div style={{
    height: "var(--lp-spacing-3)"
  }} />}
    </span>;
};

export const InlineDivider = ({margin = "0.75rem 0", padding = "0", color = "var(--lp-color-border-default)", opacity = 0.4, height = "1px", className = "", style = {}, ...rest}) => <hr role="separator" className={className} style={{
  border: "none",
  margin,
  padding,
  height,
  backgroundColor: color,
  opacity,
  ...style
}} {...rest} />;

export const CustomDivider = ({color = "var(--lp-color-border-default)", middleText = "", spacing = "default", style = {}, className = "", ...rest}) => {
  const spacingPresets = {
    default: {
      margin: "24px 0"
    },
    overlap: {
      margin: "-1rem 0 -1rem 0"
    },
    tight: {
      margin: "0 0 -1rem 0"
    },
    section: {
      margin: "0 0 -2rem 0"
    },
    sectionOverlap: {
      margin: "-1rem 0 -2rem 0"
    },
    deepOverlap: {
      margin: "-1rem 0 -1.5rem 0"
    }
  };
  const spacingStyle = spacingPresets[spacing] || spacingPresets.default;
  return <div role="separator" aria-orientation="horizontal" className={className} style={{
    display: "flex",
    alignItems: "center",
    ...spacingStyle,
    fontSize: style?.fontSize || "16px",
    height: "fit-content",
    ...style
  }} {...rest}>
      <span style={{
    marginRight: "var(--lp-spacing-px-8)",
    opacity: 0.2
  }}>
        <Icon icon="/snippets/assets/logos/Livepeer-Logo-Symbol-Theme.svg" />
      </span>
      <div style={{
    flex: 1,
    height: "1px",
    background: "var(--lp-color-border-default)",
    opacity: 0.4
  }}></div>
      {middleText && <>
          <Icon icon="circle" size={2} />
          <span style={{
    margin: "0 8px",
    fontWeight: "bold",
    color: color,
    opacity: 0.7
  }}>
            {middleText}
          </span>
          <Icon icon="circle" size={2} />
        </>}
      <div style={{
    flex: 1,
    height: "1px",
    background: "var(--lp-color-border-default)",
    opacity: 0.4
  }}></div>
      <span style={{
    marginLeft: "var(--lp-spacing-px-8)",
    opacity: 0.2
  }}>
        <span style={{
    display: "inline-block",
    transform: "scaleX(-1)"
  }}>
          <Icon icon="/snippets/assets/logos/Livepeer-Logo-Symbol-Theme.svg" />
        </span>
      </span>
    </div>;
};

<Tip>
  This page is an automated workflow.
  <Subtitle variant="changelog" style={{fontSize: "0.95rem", marginTop: "0.25rem"}}>Subscribe to this changelog's <LinkArrow label="RSS Feed" href="/v2/resources/changelog/lips/rss.xml" newline={false} /></Subtitle>
</Tip>

<CustomDivider style={{margin: "-0.5rem 0 -1.5rem 0"}} />

Track changes to <LinkArrow label="Livepeer Improvement Proposals" href="https://github.com/livepeer/LIPs" newline={false} /> on GitHub.

<CustomDivider />

<Update label="Change status of LIP-107 from Draft to Proposed" tags={["Commit"]} rss={{ title: "LIPs: Change status of LIP-107 from Draft to Proposed", description: "Change status of LIP-107 from Draft to Proposed" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Change status of LIP-107 from Draft to Proposed

  Change status of LIP-107 from Draft to Proposed

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/4057063892111ac41c24e1ae36a9b9a3f4cca453" iconLeft="github" />
</Update>

<Update label="Merge pull request #111 from shtukaresearch/LIP-491d6a" tags={["Commit"]} rss={{ title: "LIPs: Merge pull request #111 from shtukaresearch/LIP-491d6a", description: "Merge pull request #111 from shtukaresearch/LIP-491d6a" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Merge pull request #111 from shtukaresearch/LIP-491d6a

  <ScrollBox maxHeight="150px" showHint={false}>
    Merge pull request #111 from shtukaresearch/LIP-491d6a
    LIP-107.md: update status to "Proposed"
  </ScrollBox>

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/7ca02b7d0f6d02e7bfec486875f54bb894c75e2f" iconLeft="github" />
</Update>

<Update label="LIP-107.md: update status to 'Proposed'" tags={["Commit"]} rss={{ title: "LIPs: LIP-107.md: update status to 'Proposed'", description: "LIP-107.md: update status to \"Proposed\"" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## LIP-107.md: update status to 'Proposed'

  LIP-107.md: update status to "Proposed"

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/787d1bccfb5c249d4733fd52fff05a42c4d0ca86" iconLeft="github" />
</Update>

<Update label="Merge pull request #109 from shtukaresearch/LIP-491d6a" tags={["Commit"]} rss={{ title: "LIPs: Merge pull request #109 from shtukaresearch/LIP-491d6a", description: "Merge pull request #109 from shtukaresearch/LIP-491d6a" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Merge pull request #109 from shtukaresearch/LIP-491d6a

  <ScrollBox maxHeight="150px" showHint={false}>
    Merge pull request #109 from shtukaresearch/LIP-491d6a
    LIP-107.md: change status to Last Call
  </ScrollBox>

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/97ffc34d59dab744609ea680a6dec7d8f9d03b17" iconLeft="github" />
</Update>

<Update label="Merge pull request #110 from livepeer/b3nnnp-patch-2" tags={["Commit"]} rss={{ title: "LIPs: Merge pull request #110 from livepeer/b3nnnp-patch-2", description: "Merge pull request #110 from livepeer/b3nnnp-patch-2" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Merge pull request #110 from livepeer/b3nnnp-patch-2

  <ScrollBox maxHeight="150px" showHint={false}>
    Merge pull request #110 from livepeer/b3nnnp-patch-2
    Update README.md
  </ScrollBox>

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/c7561033b7078b90675203f49f97c5a0a962152e" iconLeft="github" />
</Update>

<Update label="Update README.md" tags={["Commit"]} rss={{ title: "LIPs: Update README.md", description: "Update README.md" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Update README.md

  Update README.md

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/e89791e9ccce87c5f9c429bc9d84f5f55bfaccff" iconLeft="github" />
</Update>

<Update label="LIP-107.md: change status to Last Call" tags={["Commit"]} rss={{ title: "LIPs: LIP-107.md: change status to Last Call", description: "LIP-107.md: change status to Last Call" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## LIP-107.md: change status to Last Call

  LIP-107.md: change status to Last Call

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/bdf4949dc1c56b86a6a4db19fd2e37bb89da138a" iconLeft="github" />
</Update>

<Update label="Merge pull request #107 from shtukaresearch/LIP-491d6a" tags={["Commit"]} rss={{ title: "LIPs: Merge pull request #107 from shtukaresearch/LIP-491d6a", description: "Merge pull request #107 from shtukaresearch/LIP-491d6a" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Merge pull request #107 from shtukaresearch/LIP-491d6a

  <ScrollBox maxHeight="150px" showHint={false}>
    Merge pull request #107 from shtukaresearch/LIP-491d6a
    Add LIP "Put the brakes on LPT emissions"
  </ScrollBox>

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/a3de4613746626164a211ace2b5bc47af70f6d3f" iconLeft="github" />
</Update>

<Update label="Assign LIP number 107 and retitle" tags={["Commit"]} rss={{ title: "LIPs: Assign LIP number 107 and retitle", description: "Assign LIP number 107 and retitle" }} description={<Subtitle variant="changelog">February 2026</Subtitle>}>
  ## Assign LIP number 107 and retitle

  Assign LIP number 107 and retitle

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/27b538671962b56cb0ed7403052f6a56f9a6001a" iconLeft="github" />
</Update>

<Update label="Add discussion link." tags={["Commit"]} rss={{ title: "LIPs: Add discussion link.", description: "Add discussion link." }} description={<Subtitle variant="changelog">January 2026</Subtitle>}>
  ## Add discussion link.

  Add discussion link.

  <DoubleIconLink label="View commit on GitHub" href="https://github.com/livepeer/LIPs/commit/9959ff45a374f7716edab4fd72f0f66a6a55ea13" iconLeft="github" />
</Update>
