
Product
Introducing Socket Firewall Enterprise: Flexible, Configurable Protection for Modern Package Ecosystems
Socket Firewall Enterprise is now available with flexible deployment, configurable policies, and expanded language support.
@bubblydoo/angular-react
Advanced tools
This is a small Angular library that lets you use React components inside Angular projects.
<react-wrapper [component]="Button" [props]="{ children: 'Hello world!' }">
function ReactComponent({ text }) {
  return <AngularWrapper component={TextComponent} inputs={{ text }}>
}
npm i @bubblydoo/angular-react
import { AngularReactModule } from '@bubblydoo/angular-react'
@NgModule({
  ...,
  imports: [
    ...,
    AngularReactModule
  ]
})
ReactWrapperComponentUse this component when you want to use React in Angular.
It takes two inputs:
component: A React componentprops?: The props you want to pass to the React componentThe React component will be first rendered on ngAfterViewInit and rerendered on every ngOnChanges call.
import Button from './button.tsx';
@Component({
  template: `<react-wrapper [component]="Button" [props]="{ children: 'Hello world!' }">`
})
class AppComponent {
  Button = Button
}
AngularWrapperUse this component when you want to use Angular in React.
It takes a few inputs:
component: An Angular componentinputs?: The inputs you want to pass to the Angular component, in an objectoutputs?: The outputs you want to pass to the Angular component, in an objectevents?: The events from the Angular component to listen to, using addEventListener. Event handlers are wrapped in NgZone.runref?: The ref to the rendered DOM element (uses React.forwardRef)import { TextComponent } from './text/text.component.ts'
function Text(props) {
  return (
    <AngularWrapper
      component={TextComponent}
      inputs={{ text: props.text }}
      events={{ click: () => console.log('clicked') }}/>
  )
}
useInjectedThe Angular Injector is provided on each React component by default using React Context. You can use Angular services and other injectables with it:
import { useInjected } from '@bubblydoo/angular-react'
const authService = useInjected(AuthService)
useObservableBecause consuming observables is so common, we added a helper hook for it:
import { useObservable, useInjected } from '@bubblydoo/angular-react'
function LoginStatus() {
  const authService = useInjected(AuthService)
  const [value, error, completed] = useObservable(authService.isAuthenticated$)
  if (error) return <>Something went wrong!<>
  return <>{value ? "Logged in!" : "Not logged in"}</>
}
If you want to have a global React Context, you can register it as follows:
// app.component.ts
constructor(angularReact: AngularReactService) {
  const client = new ApolloClient();
  // equivalent to ({ children }) => <ApolloProvider client={client}>{children}</ApolloProvider>
  angularReact.wrappers.push(({ children }) => React.createElement(ApolloProvider, { client, children }));
}
In this example, we use ApolloProvider to provide a client to each React element. We can then use useQuery in all React components.
This is only needed when your host app is an Angular app. If you're using Angular-in-React, the context will be bridged.
You can get a ref to the Angular component instance as follows:
import { ComponentRef } from '@angular/core';
const ref = useRef<ComponentRef<any>>();
<AngularWrapper ref={ref} />;
To get a reference to the Angular component's HTML element, use ref.location.nativeElement.
To forward a ref to a React component, you can simply use the props:
const Message = forwardRef((props, ref) => {
  return <div ref={ref}>{props.message}</div>;
});
@Component({
  template: `<react-wrapper [component]="Message" [props]="{ ref, message }">`
})
export class MessageComponent {
  Message = Message;
  message = 'hi!';
  ref(div: HTMLElement) {
    div.innerHTML = 'hi from the callback ref!';
  }
}
@Component({
  selector: "inner",
  template: `number: {{ number$ | async }}`,
  changeDetection: ChangeDetectionStrategy.OnPush,
})
class InnerComponent {
  number$ = this.contexts.read(NumberContext);
  constructor(@Inject(PassedReactContextToken) public contexts: PassedReactContext) {}
}
function App() {
  const [number, setNumber] = useState(42);
  return (
    <NumberContext.Provider value={number}>
      <button onClick={() => setNumber(number + 1)}>increment</button>
      <AngularWrapper component={InnerComponent} />
    </NumberContext.Provider>
  );
}
useToAngularTemplateRef: to convert a React component into a TemplateRefimport { useToAngularTemplateRef } from "@bubblydoo/angular-react";
@Component({
  selector: 'message',
  template: `
    <div>
      <ng-template [ngTemplateOutlet]="tmpl" [ngTemplateOutletContext]="{ message }"></ng-template>
    </div>
  `
})
class MessageComponent {
  @Input() tmpl: TemplateRef<{ message: string }>;
  @Input() message: string;
}
function Text(props: { message: string }) {
  return <>{props.message}</>
}
function Message(props: { message: string }) {
  const tmpl = useToAngularTemplateRef(Text);
  const inputs = useMemo(() => ({
    message: props.message,
    tmpl
  }), [props.message, tmpl]);
  return <AngularWrapper component={MessageComponent} inputs={inputs} />
}
useFromAngularTemplateRef: to convert a TemplateRef into a React componentfunction Message(props: {
  message: string;
  tmpl: TemplateRef<{ message: string }>;
}) {
  const Template = useFromAngularTemplateRef(props.tmpl);
  return <Template message={props.message.toUpperCase()} />;
}
@Component({
  selector: "outer",
  template: `
    <ng-template #tmpl let-message="message">{{ message }}</ng-template>
    <div>
      <react-wrapper
        [component]="Message"
        [props]="{ tmpl, message }"
      ></react-wrapper>
    </div>
  `,
})
class MessageComponent {
  Message = Message;
  @Input() message!: string;
}
Contexts are bridged automatically using this library, using its-fine.
You can test the functionality of the components inside a local Storybook:
yarn storybook
If you want to use your local build in an Angular project, you'll need to build it:
yarn build
Then, use yarn link:
cd dist/angular-react
yarn link # this will link @bubblydoo/angular-react to dist/angular-react
# or `npm link`
In your Angular project:
yarn link @bubblydoo/angular-react
# or `npm link @bubblydoo/angular-react`
node_modules/@bubblydoo/angular-react will then be symlinked to dist/angular-react.
You might want to use resolutions or overrides if you run into NG0203 errors.
"resolutions": {
  "@bubblydoo/angular-react": "file:../angular-react/dist/angular-react"
}
See this blog post for the motivation and more details: Transitioning from Angular to React, without starting from scratch
FAQs
Use React in Angular and Angular in React, easily
The npm package @bubblydoo/angular-react receives a total of 1,637 weekly downloads. As such, @bubblydoo/angular-react popularity was classified as popular.
We found that @bubblydoo/angular-react 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.

Product
Socket Firewall Enterprise is now available with flexible deployment, configurable policies, and expanded language support.

Security News
Open source dashboard CNAPulse tracks CVE Numbering Authorities’ publishing activity, highlighting trends and transparency across the CVE ecosystem.

Product
Detect malware, unsafe data flows, and license issues in GitHub Actions with Socket’s new workflow scanning support.