https://github.com/automattic/vip-workflow-plugin
  
  
    This WordPress plugin adds additional editorial workflow capabilities to WordPress. 
    https://github.com/automattic/vip-workflow-plugin
  
wordpress wordpress-plugin wpvip-plugin
        Last synced: 6 days ago 
        JSON representation
    
This WordPress plugin adds additional editorial workflow capabilities to WordPress.
- Host: GitHub
- URL: https://github.com/automattic/vip-workflow-plugin
- Owner: Automattic
- License: gpl-3.0
- Created: 2024-07-03T01:15:38.000Z (over 1 year ago)
- Default Branch: trunk
- Last Pushed: 2024-10-15T15:37:32.000Z (about 1 year ago)
- Last Synced: 2025-09-29T12:45:55.062Z (26 days ago)
- Topics: wordpress, wordpress-plugin, wpvip-plugin
- Language: PHP
- Homepage:
- Size: 4.72 MB
- Stars: 11
- Watchers: 6
- Forks: 2
- Open Issues: 7
- 
            Metadata Files:
            - Readme: README.md
- License: LICENSE
- Codeowners: .github/CODEOWNERS
- Security: SECURITY.md
 
Awesome Lists containing this project
README
          ---
### :warning: This plugin is currently in Beta. It is designed to run on [WordPress VIP](https://wpvip.com). This beta release is not intended for use on a production environment.
---
# WordPress VIP Workflow Plugin (Beta)
[](https://github.com/Automattic/vip-workflow-plugin/actions/workflows/php-tests.yml)[](https://github.com/Automattic/vip-workflow-plugin/actions/workflows/php-tests.yml)
The WordPress VIP Workflow Plugin allows you to control your editorial workflow. You can set up custom statuses for every phase of your workflow, and transitions between statuses generate actions, like notifications.
This plugin is currently developed for use on WordPress sites hosted on [WordPress VIP](https://wpvip.com).
- [Try it out](#try-it-out)
- [Installation](#installation)
  - [Install via `git subtree`](#install-via-git-subtree)
  - [Install via ZIP file](#install-via-zip-file)
  - [Plugin activation](#plugin-activation)
- [Usage](#usage)
  - [Admin](#admin)
  - [Notifications](#notifications)
  - [Publish Guard](#publish-guard)
  - [Editorial Experience](#editorial-experience)
    - [Guided Status Movements](#guided-status-movements)
    - [Preview Links](#preview-links)
    - [Editorial Metadata](#editorial-metadata)
- [Code Filters](#code-filters)
  - [`vw_notification_ignored_statuses`](#vw_notification_ignored_statuses)
  - [`vw_notification_email_recipients`](#vw_notification_email_recipients)
  - [`vw_notification_email_subject`](#vw_notification_email_subject)
  - [`vw_notification_email_message`](#vw_notification_email_message)
  - [`vw_notification_email_headers`](#vw_notification_email_headers)
  - [`vw_notification_send_to_webhook_payload`](#vw_notification_send_to_webhook_payload)
  - [`vw_preview_expiration_options`](#vw_preview_expiration_options)
- [Development](#development)
  - [Building the plugin](#building-the-plugin)
  - [Using Hot Module Replacement](#using-hot-module-replacement)
  - [Tests](#tests)
- [Credits](#credits)
## Try it out
Try out the VIP Workflow plugin in your browser [with WordPress Playground][playground-blueprint]:
[ ][playground-blueprint]
][playground-blueprint]
## Installation
The latest version of the plugin is available in the default `trunk` branch of this repository.
### Install via `git subtree`
We recommend installing the latest plugin version [via `git subtree`][wpvip-plugin-subtrees] within your site's repository:
```bash
# Enter your project's root directory:
cd my-site-repo/
# Add a subtree for the trunk branch:
git subtree add --prefix plugins/vip-workflow-plugin git@github.com:Automattic/vip-workflow-plugin.git trunk --squash
```
To deploy the plugin to a remote branch, `git push` the committed subtree.
The `trunk` branch will stay up to date with the latest version of the plugin. Use this command to pull the latest `trunk` branch changes:
```bash
git subtree pull --prefix plugins/vip-workflow-plugin git@github.com:Automattic/vip-workflow-plugin.git trunk --squash
```
Ensure that the plugin is up-to-date by pulling changes often.
Note: We **do not recommend** using `git submodule`. [Submodules on WPVIP that require authentication][wpvip-plugin-submodules] will fail to deploy.
### Install via ZIP file
The latest version of the plugin can be downloaded from the [repository's Releases page][repo-releases]. Unzip the downloaded plugin and add it to the `plugins/` directory of your site's GitHub repository.
### Plugin activation
We recommend [activating plugins with code][wpvip-plugin-activate].
## Usage
### Admin
Once the plugin is activated, go to the `VIP Workflow` menu in `wp-admin` to configure your workflow.
By default, the following post statuses are created:
1. Pitch
2. Assigned
3. In Progress
4. Draft
5. Pending Review
Note that, these statuses are also available in the quick edit experience on the posts page alongside the core post statuses.
The plugin doesn't expect any specific configuration, so your first step is to set up statuses that reflect your workflow. You may notice that the steps are listed in a linear order. The plugin assumes a linear workflow where content is moving from creation to publish.
You can also specify which types of content use custom statuses. If a post does not use custom statuses, it will use the standard WordPress publishing flow.
### Notifications
VIP Workflow has the ability to send email and/or webhook notifications when a post's status changes. By default, these are off as no email address or webhook is provided out of the box. You set up webhook and/or email notifications under Admin -> VIP Workflow -> Settings.
Additional filters are available here to customize the notifications that are sent.
### Publish Guard
By default, VIP Workflow prevents publishing a post or page (the supported post types are configurable within settings) unless it has reached the last status on your list. This feature can be turned off under settings under the `Publish Guard` option in Admin -> VIP Workflow -> Settings.
### Editorial Experience
#### Guided Status Movements
VIP Workflow adds an editor button to move posts between custom post statuses:
![Moving between custom statuses using a custom save process][media-guided-status-movements]
While editing a post with a custom status, click the "Move to ..." button to advance to the next state. When [Publish Guard](#publish-guard) is enabled, the regular "Publish" button will be hidden until a post reaches the final custom status state.
Custom statuses can also be modified by using the "Extended Post Status" row "Edit" button in the post sidebar.
#### Preview Links
VIP Workflow adds a "Preview" section to the post sidebar, which allows sharing previews of pre-published content:
![Generating a preview link on a pre-published post][media-generate-preview-link]
Anybody with a preview link (including not logged-in users) will be able to view a post with a preview link. Preview links can expire in three ways:
1. Via expiration. Preview links are generated with an expiration (1 hour, 8 hours, or 1 day by default). When this time has passed, the token URL will no longer be valid.
2. Via one-time usage. If the "One-time use" checkbox is selected, the preview URL will only work a single time, and then the token will become invalid. If this box is not selected, a preview URL can be used any number of times before expiration.
3. Via post status changes. If a post moves out of VIP Workflow's set of extended post statuses, tokens will no longer be valid. For example, a post moved to "Publish" will no longer have valid preview tokens.
#### Editorial Metadata
VIP Workflow adds a "Editorial Metadata" section to the post sidebar, which allows for additional data to be included with the post such as "Needs Legal Review". This can be managed under the plugin's settings, to get a visual for all of the configured editorial metadata fields.
## Code Filters
### `vw_notification_ignored_statuses`
When a post changes status, configured notifications are sent. You can use this filter to prevent notifications when a post transitions into specific statuses. By default, the plugin already filters the built-in `inherit` and `auto-draft` statuses.
```php
/**
* Filter the statuses that should be ignored when sending notifications
*
* @param array $ignored_statuses Array of statuses that should be ignored when sending notifications
* @param string $post_type The post type of the post
*/
apply_filters( 'vw_notification_ignored_statuses', [ $old_status, 'inherit', 'auto-draft' ], $post->post_type );
```
For example, this filter can be used to add `assigned` to the ignored statuses for a post of `post_type` page, so no notifications are sent for such posts:
```php
add_filter( 'vw_notification_ignored_statuses', function ( $ignored_statuses, $post_type ) {
    if ( $post_type === 'page' ) {
        $ignored_statuses[] = 'assigned';
    }
    return $ignored_statuses;
}, 10, 2 );
```
### `vw_notification_email_recipients`
Change the recipients that receive an email notification, when the status of a post changes. By default, it is set to the configured email address under Admin -> VIP Workflow -> Settings.
```php
/**
* Filter the email recipients
*
* @param array $email_recipients Array of email recipients
* @param string $action Action being taken, eg. status-change
* @param WP_Post $post Post object
*/
apply_filters( 'vw_notification_email_recipients', $email_recipients, $action, $post );
```
For example, this filter can be used to send email notifications to more than just 1 recipients especially for special statuses:
```php
add_filter( 'vw_notification_email_recipients', function ( $email_recipients, $action, $post  ) {
  if ( $post->post_status === 'legal-review' ) {
    $email_recipients[] = 'saul.goodman@sgoodmanassoc.com';
  }
  return $email_recipients;
}, 10, 2 );
```
### `vw_notification_email_subject`
Change the subject of the email that recipients receive, when the status of a post changes.
```php
/**
* Filter the email subject
*
* @param string $subject Subject of the email
* @param string $action Action being taken, eg. status-change
* @param WP_Post $post Post object
*/
apply_filters( 'vw_notification_email_subject', $subject, $action, $post );
```
For example, this filter can be used to set a standardized subject regardless of what the status is:
```php
add_filter( 'vw_notification_email_subject', function ( $subject, $action, $post  ) {
  return __( 'Content Status Update' );
}, 10, 2 );
```
### `vw_notification_email_message`
Change the message of the email that recipients receive, when the status of a post changes.
```php
/**
* Filter the email message
*
* @param string $message Body of the email
* @param string $action Action being taken, eg. status-change
* @param WP_Post $post Post object
*/
apply_filters( 'vw_notification_email_message', $message, $action, $post );
```
For example, this filter can be used to replace the signature that the plugin adds to the footer of the email with a company one instead:
```php
add_filter( 'vw_notification_email_message', function ( $message, $action, $post  ) {
  return str_replace( 'You are receiving this email because a notification was configured via the VIP Workflow Plugin.', 'You are receiving this email as part of ACME Corp.', $message);
}, 10, 2 );
```
### `vw_notification_email_headers`
Change the headers used for the email that recipients receive, when the status of a post changes. By default, they are the standard headers set by wp_mail.
```php
/**
* Filter the email recipients
*
* @param array $message_headers Message headers
* @param string $action Action being taken, eg. status-change
* @param WP_Post $post Post object
*/
apply_filters( 'vw_notification_email_headers', $message_headers, $post );
```
For example, this filter can be used to send HTML formatted email notifications instead of the default plain text formatted email notifications:
```php
add_filter( 'vw_notification_email_headers', function ( $message_headers, $action, $post  ) {
  return [ 'Content-Type: text/html; charset=UTF-8' ];
}, 10, 2 );
```
### `vw_notification_send_to_webhook_payload`
Change the payload sent to the webhook, when the status of a post changes. By default, it is as follows:
```
{
  "type": "plugin:vip-workflow:post-update",
  "timestamp": "2024-07-22 00:03:19",
  "data": "*vipgo* changed the status of *Post #85 - * from *In Progress* to *Draft*"
}
```
```php
/**
* Filter the payload before sending it to the webhook
*
* @param array $payload Payload to be sent to the webhook
*/
apply_filters( 'vw_notification_send_to_webhook_payload', $payload );
```
For example, this filter can be used to customize the payload so that it's compatible with Slack's [incoming webhooks](https://api.slack.com/messaging/webhooks):
```php
add_filter( 'vw_notification_send_to_webhook_payload', function ( $payload ) {
    return [
        'text' => $payload['data'],
    ];
}, 10, 1 );
```
### `vw_preview_expiration_options`
Change the default expiration options available in the preview URL feature.
```php
/**
 * Filter the expiration options available in the preview modal dropdown.
 *
 * @param array $expiration_options Array of expiration options. Each option uses keys:
 *     'label': The visible label for the option, e.g. "1 hour"
 *     'value': The value to be sent to the API, e.g. "1h". This value should be unique.
 *     'second_count': The number of seconds the this expiration should be valid for, e.g. 3600
 *     'default': Optional. Whether this option should be selected by default.
 */
return apply_filters( 'vw_preview_expiration_options', [ /* ... */ ]);
```
Here is an example using a shorter set of expiration lengths:
```php
add_filter( 'vw_preview_expiration_options', function () {
    return [
        [
            'label'        => '5 minutes',
            'value'        => '5m',
            'second_count' => 5 * 60,
            'default'      => true,
        ],
        [
            'label'        => '15 minutes',
            'value'        => '15m',
            'second_count' => 15 * 60,
        ],
        [
            'label'        => '1 hour',
            'value'        => '1h',
            'second_count' => 60 * 60,
        ],
    ];
} );
```
This generates these custom expiration values in the preiew link dialog:
![Preview link dialog with custom expiration times][media-generate-preview-link-custom-expiration]
## Development
This plugin uses `wp-env` for development, and `wp-env` to run the tests written for the plugin. `wp-env` requires Docker so please ensure you have that installed on your system first.
To install `wp-env`, use the following command:
```
npm -g i @wordpress/env
```
Read more about `wp-env` [here](https://www.npmjs.com/package/@wordpress/env).
Optionally, it's also possible to use `vip dev-env` instead for development. Installation instructions for the VIP cli can be found [here](https://docs.wpvip.com/vip-cli/installing-vip-cli/), and instructions on how to setup `vip dev-env` can be found [here](https://docs.wpvip.com/vip-cli/commands/dev-env/).
This plugin also uses Composer to manage PHP dependencies. Composer can be downloaded [here](https://getcomposer.org/download/).
### Building the plugin
1. Install PHP dependencies: `composer install`
2. Install NPM dependencies: `npm install`
3. Start dev environment: `wp-env start`
### Using Hot Module Replacement
React hot reloading is supported. A few configuration steps are required for the setup:
1. Set `define( 'SCRIPT_DEBUG', true );` in your `wp-config.php` or `vip-config.php`. At the time of writing, this is [a `wp-scripts` limitation](https://github.com/WordPress/gutenberg/blob/9e07a75/packages/scripts/README.md?plain=1#L390).
2. Run `npm run dev:hot`. If you're running WordPress on a non-localhost hostname, e.g. on `vip dev-env`, you may also need to specify the hostname:
   ```bash
   HOST=mysite.vipdev.lndo.site npm run dev:hot
   ```
   This can also be specified using a `.env` configuration file:
   ```
   HOST=mysite.vipdev.lndo.site
   ```
   If you use `wp-env`, you should be able to skip specifying `HOST` manually.
3. If HMR is not working and you're developing out of a new component tree, you may also need to opt-in to hot module reloading via [`module.hot.accept()`](https://github.com/Automattic/vip-workflow-plugin/blob/e058354/modules/custom-status/lib/custom-status-configure.js#L19-L21)
### Tests
We are currently in the process of writing unit tests to ensure thorough code coverage of the plugin. At the moment, these are just PHP tests which can be run locally using the following:
```
wp-env start
composer install
composer run test
```
## Credits
This plugin has been based on the wonderful [EditFlow](https://github.com/Automattic/Edit-Flow) plugin developed by Daniel Bachhuber, Scott Bressler, Mohammad Jangda, and others.
[media-generate-preview-link-custom-expiration]: https://github.com/Automattic/vip-workflow-plugin/blob/media/generate-preview-link-custom-expiration.gif
[media-generate-preview-link]: https://github.com/Automattic/vip-workflow-plugin/blob/media/generate-preview-link.gif
[media-guided-status-movements]: https://github.com/Automattic/vip-workflow-plugin/blob/media/guided-status-movements.gif
[playground-blueprint]: https://playground.wordpress.net/?blueprint-url=https://raw.githubusercontent.com/Automattic/vip-workflow-plugin/trunk/blueprint.json
[repo-releases]: https://github.com/Automattic/vip-workflow-plugin/releases
[wpvip-plugin-activate]: https://docs.wpvip.com/how-tos/activate-plugins-through-code/
[wpvip-plugin-submodules]: https://docs.wpvip.com/technical-references/plugins/installing-plugins-best-practices/#h-submodules
[wpvip-plugin-subtrees]: https://docs.wpvip.com/technical-references/plugins/installing-plugins-best-practices/#h-subtrees