Utilisation des connecteurs d'emplacement de plugin depuis un thème ou un plugin

Discourse comprend des centaines de Plugin Outlet (points de sortie de plugin) qui peuvent être utilisés pour injecter du nouveau contenu ou remplacer le contenu existant dans l’interface utilisateur de Discourse. Les « arguments de l’outlet » (Outlet arguments) sont mis à disposition afin que le contenu puisse être personnalisé en fonction du contexte.

Choisir un outlet

Pour trouver le nom d’un plugin outlet, recherchez dans le cœur de Discourse « <PluginOutlet », ou utilisez le composant de thème plugin outlet locations. (par exemple, topic-above-posts).

Wrapper outlets

Certains outlets dans le cœur ressemblent à ceci : <PluginOutlet @name="foo" />. Ceux-ci vous permettent d’injecter du nouveau contenu. D’autres outlets « envelopperont » (wrap) une implémentation de base existante comme ceci :

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

Définir un connecteur pour ce type d’« outlet enveloppant » remplacera l’implémentation de base. Un seul thème/plugin actif peut fournir un connecteur pour un plugin outlet enveloppant.

Pour les plugin outlets enveloppants, vous pouvez rendre l’implémentation de base originale en utilisant le mot-clé {{yield}}. Ceci peut être utile si vous souhaitez uniquement remplacer l’implémentation de base sous certaines conditions, ou si vous souhaitez l’envelopper dans quelque chose.

Définir le template

Une fois que vous avez choisi un outlet, décidez d’un nom pour votre connecteur. Celui-ci doit être unique parmi tous les thèmes/plugins installés sur une communauté donnée. Par exemple : brand-official-topics

Dans votre thème/plugin, définissez un nouveau template Handlebars avec un chemin formaté comme ceci :

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

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

Le contenu de ces fichiers sera rendu comme un Composant Ember. Pour des informations générales sur Ember/Handlebars, consultez les guides Ember.

Pour notre connecteur hypothétique « brand official topics », le template pourrait ressembler à ceci :

<div class="alert alert-info">
  This topic was created by a member of the
  <a href="https://discourse.org/team">Discourse Team</a>
</div>

Certains plugin outlets envelopperont automatiquement votre contenu dans un élément HTML. Le type d’élément est défini par @connectorTagName sur le <PluginOutlet>.

Utilisation des arguments de l’outlet

Les Plugin Outlets fournissent des informations sur le contexte environnant via @outletArgs. Les arguments passés à chaque outlet varient. Un moyen simple de visualiser les arguments est d’ajouter ceci à votre template :

{{log @outletArgs}}

Ceci affichera les arguments dans la console de développement de votre navigateur. Ils apparaîtront comme un objet Proxy — pour explorer la liste des arguments, développez le [[Target]] du proxy.

Dans notre exemple topic-above-posts, le sujet rendu est disponible sous @outletArgs.model. Nous pouvons donc ajouter le nom d’utilisateur du membre de l’équipe comme ceci :

<div class="alert alert-info">
  This topic was created by
  {{@outletArgs.model.details.created_by.username}}
  (a member of the
  <a href="https://discourse.org/team">Discourse Team</a>)
</div>

Ajout d’une logique plus complexe

Parfois, un simple template Handlebars ne suffit pas. Pour ajouter une logique Javascript à votre connecteur, vous pouvez définir un fichier Javascript adjacent à votre template Handlebars. Ce fichier doit exporter une définition de composant. Cela fonctionne exactement comme toute autre définition de composant, et peut inclure des injections de service.

Définir un composant de cette manière supprimera l’élément enveloppant automatique connectorTagName, vous voudrez donc peut-être réintroduire un élément du même type dans votre fichier hbs.

Dans notre exemple topic-above-posts, nous pourrions vouloir afficher l’utilisateur différemment en fonction du paramètre du site « prioritize username in ux ». Une définition de composant pour cela pourrait ressembler à ceci :

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

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;
    }
  }
}

Nous pouvons ensuite mettre à jour le template pour faire référence au nouveau getter :

<div class="alert alert-info">
  This topic was created by
  {{this.displayName}}
  (a member of the
  <a href="https://discourse.org/team">Discourse Team</a>)
</div>

Rendu conditionnel

Si vous souhaitez que votre contenu ne soit rendu que sous certaines conditions, il suffit souvent d’envelopper votre template avec un bloc {{#if}} Handlebars. Si cela ne suffit pas, vous voudrez peut-être utiliser le hook shouldRender pour contrôler si votre template de connecteur doit être rendu ou non.

Tout d’abord, assurez-vous d’avoir une définition de connecteur .js comme décrit ci-dessus. Ensuite, ajoutez une fonction statique static shouldRender(). En étendant notre exemple :

import Component from "@glimmer/component";
import { getOwner } from "discourse-common/lib/get-owner";

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

Maintenant, le connecteur ne sera rendu que lorsque le premier message du sujet a été créé par un membre de l’équipe.

shouldRender est évalué dans un contexte d’autotrace (autotracking) Glimmer. Les changements futurs sur toute propriété référencée (par exemple, outletArgs) provoqueront la réévaluation de la fonction.

Introduction de nouveaux outlets

Si vous avez besoin d’un outlet qui n’existe pas encore, n’hésitez pas à créer une pull request, ou à ouvrir un sujet dans Dev.


Ce document est contrôlé par version — suggérez des modifications sur github.

39 « J'aime »
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
Native theme support
Developing Discourse Plugins - Part 2 - Connect to a plugin outlet
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
How to add a custom button in user profile card?
How to add a custom button in user profile card?
Customizing the topic list
(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
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
Display Tags inline with thread title, instead of being on the bottom line
Add link to external SSO profile to profile page
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

Ceci est maintenant déprécié, n’est-ce pas ?
Avis de dépréciation : La définition des classes de connecteur via registerConnectorClass est dépréciée. Voir https://meta.discourse.org/t/32727 pour des modèles plus modernes. [deprecation id: discourse.register-connector-class-legacy]

C’est exact ; vous pouvez utiliser api.renderInOutlet à la place. :slight_smile:

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

1 « J'aime »

Je pense que c’est un peu plus compliqué que ça :sweat_smile:
https://github.com/Firepup6500/discourse-custom-profile-link/blob/master/common/head_tag.html

Ne t’inquiète pas ; je verrai ce que je peux faire pour t’aider plus tard (je dois dormir maintenant). :smile:

1 « J'aime »

Merci ! (Désolé pour le code qui est un désordre, je voulais juste qu’il fonctionne la dernière fois que j’y ai touché :sweat_smile:)

1 « J'aime »

Je suis un peu dépassé, j’ai reçu une notification concernant mon composant utilisé dans le fichier HEAD de mon thème. Je ne suis pas sûr de la manière de réécrire avec api.renderInOutlet.

  const ajax = require('discourse/lib/ajax').ajax;
  const Topic = require('discourse/models/topic').default;
  // Nous utilisons ajax et le modèle Topic de Discourse

  api.registerConnectorClass('above-main-container', 'featured-topics', {
    // above-main-container est la sortie du plugin,
    // featured-topics est le nom de votre composant personnalisé

    setupComponent(args, component) {

   // le reste du code suit

J’imagine que j’ai essayé de remplacer api.registerConnectorClass par api.renderInOutlet, mais cela a échoué. Je ne suis pas vraiment un expert en codage de thèmes ici. Merci pour toute aide.

Vous pouvez voir un exemple ici :

StatBanner est une classe native définie dans le répertoire components :

Dans votre cas, ce serait api.renderInOutlet("above-main-container", YourClass)

Je ne pense pas que vous puissiez le faire dans le fichier HEAD. Vous devriez diviser votre code en plusieurs fichiers.

Je vous encourage à utiliser Discourse Theme CLI car il sera beaucoup plus facile de développer un composant de thème !

Votre composant de thème est-il public ?

3 « J'aime »

Existe-t-il un moyen d’obtenir la sortie actuelle dans mon fichier .js ?

Je ne pense pas que ce soit possible.

Il était possible d’inspecter parentView dans les composants classiques.

Mais cette propriété a été dépréciée.

1 « J'aime »

Que voulez-vous dire par « obtenir le point de vente actuel » ? Vous voulez le nom du point de vente ? Ou autre chose ?

J’ai trouvé une solution à mon problème (ici), mais c’était quelque chose comme ceci :

  • Créer des fichiers .hbs et .js pour 3 sorties différentes.
  • Dans chaque fichier JS, vérifier si la sortie utilisée correspond à la valeur du paramètre banner_location.
  • Si c’est le cas, afficher la bannière. Sinon, masquer la bannière.
1 « J'aime »

Génial ! Il semble que vous ayez décidé d’utiliser api.renderInOutlet, avec une valeur dynamique pour le nom de la sortie :chefs_kiss:

1 « J'aime »