Utilizzo dei connettori Plugin Outlet da un tema o plugin

Discourse include centinaia di Plugin Outlet che possono essere utilizzati per iniettare nuovo contenuto o sostituire contenuto esistente nell’interfaccia utente di Discourse. Gli ‘Outlet arguments’ sono resi disponibili per personalizzare il contenuto in base al contesto.

Scegliere un outlet

Per trovare il nome di un plugin outlet, cerca nel codice sorgente di Discourse “<PluginOutlet”, oppure utilizza il componente del tema plugin outlet locations. (ad esempio topic-above-posts).

Outlet di tipo wrapper

Alcuni outlet nel codice sorgente hanno la forma <PluginOutlet @name="foo" />. Questi ti permettono di iniettare nuovo contenuto. Altri outlet ‘avvolgono’ un’implementazione core esistente in questo modo:

<PluginOutlet @name="foo">
  implementazione core
</PluginOutlet>

Definire un connettore per questo tipo di outlet ‘wrapper’ sostituirà l’implementazione core. Solo un tema/plugin attivo può contribuire con un connettore per un plugin outlet di tipo wrapper.

Per i plugin outlet di tipo wrapper, puoi renderizzare l’implementazione core originale utilizzando la parola chiave {{yield}}. Questo può essere utile se vuoi sostituire l’implementazione core solo in determinate condizioni, o se desideri avvolgerla in qualcosa.

Definire il connettore

Una volta scelto un outlet, decidi un nome per il tuo connettore. Questo deve essere unico tra tutti i temi/plugin installati in una data community. Ad esempio: brand-official-topics

Nel tuo tema/plugin, definisci un nuovo connettore .gjs con un percorso formattato in questo modo:

:art: {theme}/javascripts/discourse/connectors/{outlet-name}/{connector-name}.gjs

:electric_plug: {plugin}/assets/javascripts/discourse/connectors/{outlet-name}/{connector-name}.gjs

Il contenuto di questi file verrà renderizzato come un componente Ember. Per informazioni generali su Ember e sul formato .gjs, consulta le guide di Ember.

Per il nostro ipotetico connettore “brand official topics”, il file potrebbe apparire così:

<template>
  <div class="alert alert-info">
    Questo argomento è stato creato da un membro del
    <a href="https://discourse.org/team">Discourse Team</a>
  </div>
</template>

Utilizzare gli outlet arguments

I Plugin Outlets forniscono informazioni sul contesto circostante tramite @outletArgs. Gli argomenti passati a ogni outlet variano. Un modo semplice per visualizzare gli argomenti è aggiungere questo al tuo template:

{{log @outletArgs}}

Questo registrerà gli argomenti nella console degli sviluppatori del tuo browser. Appariranno come un oggetto Proxy: per esplorare l’elenco degli argomenti, espandi [[Target]] del proxy.

Nel nostro esempio topic-above-posts, l’argomento renderizzato è disponibile sotto @outletArgs.model. Quindi possiamo aggiungere il nome utente del membro del team in questo modo:

<template>
  <div class="alert alert-info">
    Questo argomento è stato creato da
    {{@outletArgs.model.details.created_by.username}}
    (un membro del
    <a href="https://discourse.org/team">Discourse Team</a>)
  </div>
</template>

Aggiungere logica più complessa

A volte, un template semplice non è sufficiente. Per aggiungere logica Javascript al tuo connettore, aggiorna il tuo file .gjs per esportare un componente basato su classe. Questo funziona esattamente come qualsiasi altra definizione di componente e può includere iniezioni di servizi.

Nel nostro esempio topic-above-posts, potremmo voler renderizzare l’utente in modo diverso in base all’impostazione del sito ‘prioritize username in ux’. Il file .gjs potrebbe assomigliare a questo:

.../connectors/topic-above-posts/brand-official-topic.gjs:

import Component from "@glimmer/component";
import { service } from "@ember/service";

export default class BrandOfficialTopics extends Component {
  @service siteSettings;

  get displayName() {
    const user = this.args.outletArgs.model.details.created_by;
    if (this.siteSettings.prioritize_username_in_ux) {
      return user.username;
    } else {
      return user.name;
    }
  }

  <template>
    <div class="alert alert-info">
      Questo argomento è stato creato da
      {{this.displayName}}
      (un membro del
      <a href="https://discourse.org/team">Discourse Team</a>)
    </div>
  </template>
}

Rendering condizionale

Se vuoi che il tuo contenuto venga renderizzato solo in determinate condizioni, spesso è sufficiente avvolgere il tuo template con un blocco {{#if}} di Handlebars. Se ciò non basta, potresti voler utilizzare il hook shouldRender per controllare se il template del tuo connettore deve essere renderizzato affatto.

Innanzitutto, assicurati di avere un connettore .gjs basato su classe come descritto sopra. Quindi, aggiungi una funzione static shouldRender(). Estendendo il nostro esempio:

import Component from "@glimmer/component";

export default class BrandOfficialTopics extends Component {
  static shouldRender(outletArgs, helper) {
    const firstPost = outletArgs.model.postStream.posts[0];
    return firstPost.primary_group_name === "team";
  }
  // ... (qualsiasi altra logica)

  <template>{{! ... }}</template>
}

Ora il connettore verrà renderizzato solo quando il primo post dell’argomento è stato creato da un membro del team.

shouldRender viene valutato in un contesto di autotracking di Glimmer. Le future modifiche a qualsiasi proprietà referenziata (ad esempio outletArgs) faranno sì che la funzione venga rieseguita.

Introdurre nuovi outlet

Se hai bisogno di un outlet che non esiste ancora, sentiti libero di fare una pull request o aprire un argomento in Development.


Questo documento è controllato tramite versioning: suggerisci modifiche su github.

39 Mi Piace
Using discourse's plugin outlets
What is the best way to integrate member applications?
Add HTML (Link) Next To Logo
Group Semantics
Can I put the search form at the top of our 404 page?
How to show user total post count beside name
Developing Discourse Plugins - Part 2 - Connect to a plugin outlet
Native theme support
Feedback on "on-discourse" javascript for setting up custom JS for each page?
Topic-timeline api.decorateWidget call has stopped working
Developing Discourse Plugins - Part 2 - Connect to a plugin outlet
Developing Discourse Themes & Theme Components
How to add btn before "sign in"
Minimizing Maintenance on Theme Customizations
How to add custom fields to models
Tags at the top of the topic list in a Category
I want to insert images (banner) between the topic answers. How do I start?
How to add a link shortcut to the area under the title
Baidu Search
Add Banner/HTML (Widget) before reply button
Upcoming Header Changes - Preparing Themes and Plugins
Upgrading Discourse to Ember 4
Converting modals from legacy controllers to new DModal component API
Need help integrating code wrote on Edittext to the Discourse
Settings not appearing
Add link to external SSO profile to profile page
How to add a custom button in user profile card?
(not recommended) Overriding Discourse templates from a Theme or Plugin
How to override the site-header.hbs file from custom theme?
Upcoming topic-list changes - how to prepare themes and plugins
Working with .erb templates in a plugin
How to Integrate a Custom Plugin in discourse UI
Templating of my "component" broke. How do I fix it?
Templating of my "component" broke. How do I fix it?
Modernizing inline script tags for templates & JS API
Custom Components -- add button or text at any plugin outlet
(not recommended) Overriding Discourse templates from a Theme or Plugin
Adding "latest topics" header in the main interface
Display Tags inline with thread title, instead of being on the bottom line
How to add a custom button in user profile card?
Discourse view file update does not reflect in browser
Discourse view file update does not reflect in browser
Add likes and views to search display
Using template hbs to add HTML content to a plugin outlet
Adding a billing section in the member section
Adding a billing section in the member section
How to modify the header HTML, but still remaining the default founctions
Removing support for "template overrides" and mobile-specific templates
How can i add image in login and register box
Most “traditional” or classic forum Category listing
Newbie help accessing code
How to add custom html next to logo using discourse plugin methods
Using the DModal API to render Modal windows (aka popups/dialogs) in Discourse
Air Theme
How to create a plugin with backend API calls to populate composer while drafting?
How to add a custom url text link on the login page
Add Text In Header Beside Logo
Split up theme Javascript into multiple files

Questo è ora deprecato, giusto?
Avviso di deprecazione: La definizione delle classi connettore tramite registerConnectorClass è deprecata. Vedere https://meta.discourse.org/t/32727 per pattern più moderni. [deprecation id: discourse.register-connector-class-legacy]

Esatto; puoi usare api.renderInOutlet invece. :slight_smile:

https://github.com/discourse/discourse/blob/main/app/assets/javascripts/discourse/app/lib/plugin-api.js#L982-L1008

1 Mi Piace

Penso sia un po’ più complicato di così :sweat_smile:
https://github.com/Firepup6500/discourse-custom-profile-link/blob/master/common/head_tag.html

Non preoccuparti; vedrò cosa posso fare per aiutarti più tardi (devo dormire adesso). :smile:

1 Mi Piace

Grazie! (Mi dispiace per il codice disordinato, volevo solo che funzionasse l’ultima volta che l’ho toccato :sweat_smile:)

1 Mi Piace

Sono un po’ in difficoltà qui, ho ricevuto una notifica per il mio componente utilizzato nel file HEAD del mio tema. Non sono sicuro di come riscriverlo con api.renderInOutlet.

  const ajax = require('discourse/lib/ajax').ajax;
  const Topic = require('discourse/models/topic').default;
  // Usiamo ajax e il modello Topic da Discourse

  api.registerConnectorClass('above-main-container', 'featured-topics', {
    // above-main-container è l'outlet del plugin,
    // featured-topics è il nome del tuo componente personalizzato

    setupComponent(args, component) {

   // il resto del codice segue

Credo di aver provato a sostituire api.registerConnectorClass con api.renderInOutlet ma si è rotto. Non sono un esperto di codifica di temi qui. Grazie per qualsiasi aiuto.

Puoi vedere un esempio qui:

StatBanner è una classe nativa definita nella directory components:

Nel tuo caso, sarebbe api.renderInOutlet("above-main-container", YourClass)

Non credo che tu possa farlo nel file HEAD. Dovresti suddividere il tuo codice in più file.

Ti incoraggio a usare Discourse Theme CLI poiché sarà molto più facile sviluppare un componente del tema!

Il tuo componente del tema è pubblico?

3 Mi Piace

C’è un modo per ottenere la presa corrente nel mio file .js?

Non credo sia possibile.

In passato era possibile ispezionare parentView nei componenti classici.

Ma quella proprietà è stata deprecata.

1 Mi Piace

Cosa intendi con “ottenere l’output corrente”? Vuoi il nome dell’output? O qualcos’altro?

Ho trovato una soluzione al mio problema (qui), ma era qualcosa del tipo:

  • Creare file .hbs e .js per 3 diversi outlet.
  • In ogni file JS, verificare se l’outlet che sta utilizzando è il valore dell’impostazione banner_location.
  • Se lo è, mostrare il banner. Altrimenti, nascondere il banner.
1 Mi Piace

Fantastico! Sembra che tu abbia deciso di usare api.renderInOutlet, con un valore dinamico per il nome dell’outlet :chefs_kiss:

1 Mi Piace