Skip to content

Custom Permalinks is a powerful WordPress plugin that grants you complete control over your site's URLs. Easily set unique, custom permalinks for any post, page, tag, or category, allowing you to design an ideal site structure.

License

Notifications You must be signed in to change notification settings

samiahmedsiddiqui/custom-permalinks

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Custom Permalinks

You want to take control of your WordPress site's URLs? The Custom Permalinks plugin gives you the power to set unique, custom URLs for any post, page, tag, or category. This means you can design your site's structure exactly how you envision it, rather than being limited by WordPress's default settings. When you set a custom permalink, the original post URL will be automatically redirected to your new, customized URL.

Key Features

  • Individual Permalink Control: Assign unique URLs to any post, page, tag, or category.
  • Site Structure Control: Gain ultimate control over how your site's URLs are organized.
  • Post Type Permalink Structures (v3.0.0+): Define custom permalink structures for each public Post Type using predefined tags, automatically generating URLs upon content creation. You can still manually edit any permalink. If left empty, default settings will apply.

Getting Started: Plugin Settings

You can configure Custom Permalinks by navigating to Settings > Custom Permalinks in your WordPress Dashboard.

Available Tags for Permalink Structures

When setting up your custom permalink structures, you can use a variety of tags that will dynamically populate the URL. Here's a breakdown of what's available:

Tag Name Description
%year% The year of the post in four digits, eg: 2025
%monthnum% Month the post was published, in two digits, eg: 01
%day% Day the post was published in two digits, eg: 02
%hour% Hour of the day, the post was published, eg: 15
%minute% Minute of the hour, the post was published, eg: 43
%second% Second of the minute, the post was published, eg: 33
%post_id% The unique ID of the post, eg: 123
%category% A clean version of the category name (its slug). Nested sub-categories will appear as nested directories in the URL.
%author% A sanitized version of the post author’s name.
%postname% A clean version of the post or page title (its slug). For example, "This Is A Great Post!" becomes this-is-a-great-post in the URL.
%parent_postname% Similar to %postname%, but uses the immediate parent page's slug if a parent is selected.
%parents_postnames% Similar to %postname%, but includes all parent page slugs if parents are selected.
%title% The title of the post, converted to a slug. For example, "This Is A Great Post!" becomes this-is-a-great-post. Unlike %postname% which is set once, %title% automatically updates in the permalink if the post title changes (unless the post is published or the permalink is manually edited).
%ctax_TAXONOMY_NAME% A clean version of a custom taxonomy's name. Replace TAXONOMY_NAME with the actual taxonomy name. You can also provide a default slug for when no category/taxonomy is selected by using ?? (e.g., %ctax_type??sales% will use "sales" as a default).
%ctax_parent_TAXONOMY_NAME% Similar to %ctax_TAXONOMY_NAME%, but includes the immediate parent category/tag slug in the URL if a parent is selected.
%ctax_parents_TAXONOMY_NAME% Similar to %ctax_TAXONOMY_NAME%, but includes all parent category/tag slugs in the URL if parents are selected.
%custom_permalinks_TAG_NAME% Developers have the flexibility to define their own custom tags(replace _TAG_NAME with your desired name). To ensure these tags resolve to the correct permalinks, simply apply the custom_permalinks_post_permalink_tag filter.

Important Note: For new posts, Custom Permalinks will keep updating the permalink while the post is in draft mode, assuming a structure is defined in the plugin settings. Once the post is published or its permalink is manually updated, the plugin will stop automatic updates for that specific post.

Advanced Customization and Filters

Custom Permalinks offers developers a robust set of filters and actions to precisely control its behavior. This section outlines how to leverage these features for tasks like generating permalinks programmatically and fine-tuning URL structures.


Setting a Custom Value in Your Post Type Permalink

Let's say you have a custom post type named "Press" and you want to include the year and month from an ACF (Advanced Custom Fields) date field directly in its permalink.

If your post type's permalink structure is about/newsroom/press-releases/%custom_permalinks_year%/%custom_permalinks_month%/%postname%/, here's how you can achieve that with a custom value:

/**
 * Add ACF field year and month in the permalink of the "Press" post type.
 *
 * @param string $custom_tag Custom tag name.
 * @param string $post_type  Post type from where it is called.
 * @param object $post       The post object.
 *
 * @return string Custom tag value which needs to be used.
 */
function custom_permalinks_post_permalink_tag( $custom_tag, $post_type, $post ) {
	$custom_tag_value = $custom_tag;
	if ( 'press' === $post_type &&
		( 'year' === $custom_tag || 'month' === $custom_tag )
	) {
		// Replace the field name 'press_date'.
		$press_date = get_field( 'press_date', $post->ID );
		if ( ! empty( $press_date ) ) {
			$date = new DateTime( $press_date );
		} else {
			$date = new DateTime( $post->post_date );
		}

		if ( 'year' === $custom_tag ) {
			$custom_tag_value = $date->format( 'Y' );
		} else {
			$custom_tag_value = $date->format( 'm' );
		}
	}

	return $custom_tag_value;
}
add_filter( 'custom_permalinks_post_permalink_tag', 'custom_permalinks_post_permalink_tag', 10, 3 );

This custom_permalinks_post_permalink_tag filter allows you to dynamically insert values into your permalink structure. The example retrieves the year and month from your ACF date field named press_date and inserts them where %custom_permalinks_year% and %custom_permalinks_month% are defined in your permalink structure. You can integrate this code into your theme's functions.php file or a custom plugin to retrieve those values and return them for the respective permalink tags.


Manipulate Permalink Before Saving

Make changes to a permalink string just before it's saved to the database. This is useful for enforcing specific formatting, like ensuring a trailing slash:

function custom_permalinks_permalink_before_saving( $permalink, $post_id, $language_code ) {
	// Check for a trailing slash in the permalink.
	if ( '/' !== substr( $permalink, -1 ) ) {
		// If the permalink doesn't have a trailing slash, add one.
		$permalink .= '/';
	}

	return $permalink;
}
add_filter( 'custom_permalink_before_saving', 'custom_permalinks_permalink_before_saving', 10, 3 );

Allow Accented Letters in Permalinks

Enable the use of accented characters in permalinks:

add_filter( 'custom_permalinks_allow_accents', '__return_true' );

Allow Uppercase Letters in Permalinks

Enable the use of uppercase characters in permalinks:

add_filter( 'custom_permalinks_allow_caps', '__return_true' );

Allow Redundant Hyphens in Permalinks

Allow permalinks to contain redundant hyphens (e.g., --):

add_filter( 'custom_permalinks_redundant_hyphens', '__return_true' );

Generating Custom Permalinks

This section outlines how to generate custom permalinks for your WordPress posts, either individually or for an entire post type. This functionality allows site owners and developers to programmatically create or regenerate permalinks without manual updates.

Generate Permalink for a Single Post ID

To generate a custom permalink for a specific post, you can use the following action hook, replacing $post_id with the actual ID of your post:

do_action( 'custom_permalinks_generate_post_permalink', $post_id );

This simple line of code triggers the permalink generation process for the specified post.

Generate Permalinks for an Entire Post Type

For bulk updates or after structural changes, you might need to regenerate permalinks for all posts within a specific post type. Here's an example demonstrating how to do this for a custom post type called product.

/**
 * Generates custom permalinks for all posts within a specified post type.
 *
 * @param string $post_type The post type to generate permalinks for.
 */
function custom_permalinks_generate_permalinks_for_post_type( $post_type ) {
	$args = array(
		'post_type'      => $post_type,
		'posts_per_page' => -1, // Get all posts of this type.
		'post_status'    => 'publish', // Only published posts.
		'fields'         => 'ids', // Only retrieve post IDs for efficiency.
	);

	$posts = get_posts( $args );
	if ( $posts ) {
		foreach ( $posts as $post_id ) {
			do_action( 'custom_permalinks_generate_post_permalink', $post_id );
		}
		echo "Permalinks for all '{$post_type}' posts have been regenerated.";
	} else {
		echo "No '{$post_type}' posts found to regenerate permalinks for.";
	}
}

// Example usage: Call the function to regenerate permalinks for the 'product' post type.
custom_permalinks_generate_permalinks_for_post_type( 'product' );

This PHP function iterates through all published posts of the specified $post_type and applies the do_action hook to each, ensuring their permalinks are regenerated. You can integrate this code into your theme's functions.php file or a custom plugin, typically running it as a one-time process or via an administrative trigger.


Exclude Post Type from Custom Permalink Form

Remove the custom permalink settings form from the edit screen of a specific post type:

function custom_permalinks_exclude_post_types( $post_type ) {
	// Replace 'custompost' with the name of the post type you want to exclude.
	if ( 'custompost' === $post_type ) {
		return '__true';
	}

	return '__false';
}
add_filter( 'custom_permalinks_exclude_post_type', 'custom_permalinks_exclude_post_types' );

Exclude Specific Posts from Custom Permalink Form

Remove the custom permalink settings form from individual posts based on criteria like their ID:

function custom_permalinks_exclude_posts( $post ) {
	// Replace '1557' with the ID of the post you want to exclude.
	if ( 1557 === $post->ID ) {
		return true;
	}

	return false;
}
add_filter( 'custom_permalinks_exclude_posts', 'custom_permalinks_exclude_posts' );

Exclude Permalink from Processing

Skip processing for specific permalinks, which can be useful for URLs like XML sitemaps:

function custom_permalinks_xml_sitemap_url( $permalink ) {
	if ( false !== strpos( $permalink, 'sitemap.xml' ) ) {
		// Use '__true' to specifically ignore this permalink.
		return '__true';
	}

	// Return null (or nothing) to allow default processing.
	return;
}
add_filter( 'custom_permalinks_request_ignore', 'custom_permalinks_xml_sitemap_url' );

Add PATH_INFO to $_SERVER Variable

Enable PATH_INFO to be added to the $_SERVER superglobal variable:

add_filter( 'custom_permalinks_path_info', '__return_true' );

Disable All Redirects

To prevent Custom Permalinks from performing any redirects:

function custom_permalinks_avoid_redirect( $permalink ) {
	// Always return true to disable all redirects.
	return true;
}
add_filter( 'custom_permalinks_avoid_redirect', 'custom_permalinks_avoid_redirect' );

Disable Specific Redirects

Prevent a particular permalink from being redirected by checking its value:

function custom_permalinks_avoid_redirect( $permalink ) {
	// Replace 'testing-hello-world/' with the permalink you want to exclude from redirection.
	if ( 'testing-hello-world/' === $permalink ) {
		return true;
	}
	return false;
}
add_filter( 'custom_permalinks_avoid_redirect', 'custom_permalinks_avoid_redirect' );

Disable like Query

Disable the like query functionality, which can impact URL matching in some specific scenarios:

add_filter( 'cp_remove_like_query', '__return_false' );

Note: Use custom_permalinks_like_query if URLs don't work after upgrading to v1.2.9.

Note: If you experience issues with URLs after upgrading to v1.2.9, consider using custom_permalinks_like_query instead.


For Assistance:

  • Premium Users: If you need assistance implementing these filters, please don't hesitate to reach out to us via our Premium contact support.
  • Other Users: You can also directly reach out to the plugin author via LinkedIn.

Need Help or Found a Bug?

  • Support: For one-on-one email support, consider purchasing Custom Permalinks Premium. While some basic support may be provided on the WordPress.org forums, email support is prioritized for premium users.
  • Bug Reports: If you encounter a bug, please report it on GitHub. Make sure to provide complete information to reproduce the issue. GitHub is for bug reports, not general support questions.

If you experience any site-breaking issues after upgrading, please report them on the WordPress Forum or GitHub with detailed information. You can always revert to an older version by downloading it from https://wordpress.org/plugins/custom-permalinks/advanced/.

Installation

You have two ways to install Custom Permalinks:

From within WordPress

  1. Go to Plugins > Add New in your WordPress dashboard.
  2. Search for "Custom Permalinks".
  3. Click "Install Now" and then "Activate" the plugin from your Plugins page.

Manually via FTP

  1. Download the custom-permalinks folder.
  2. Upload the custom-permalinks folder to your /wp-content/plugins/ directory.
  3. Activate Custom Permalinks through the "Plugins" menu in your WordPress dashboard.

About

Custom Permalinks is a powerful WordPress plugin that grants you complete control over your site's URLs. Easily set unique, custom permalinks for any post, page, tag, or category, allowing you to design an ideal site structure.

Topics

Resources

License

Stars

Watchers

Forks

Contributors 6