Rosey is used to generate a multilingual site from a set of JSON files. As part of this, Rosey creates a redirect so that the site visitor is redirected to the language set in their browser settings.
How it works at a high level:
-
Html elements are tagged by a developer for translation using
data-rosey
tags. -
Rosey creates a JSON file called
base.json
from these tags by scanning your built static site. -
Rosey takes a different
locales/xx-XX.json
file, which contains the original phrase with a user entered translation and generates the finished translated site.
What the RCC connector does is create a way for non-technical editors to create the locales/xx-XX.json
files needed to generate the site. Using the base.json
file, YAML files are generated with the correct CloudCannon input configuration to enable translations via an interface in CloudCannon's CMS. These editor-friendly YAML files are then turned into the JSON files needed by Rosey to generate your final multilingual site.
All of this happens in your site's postbuild, meaning it automatically happens each build. The file generation and entry of translations happens on your staging site, while the multilingual site generation takes place on your production (main) site.
An easier-to-understand approach would be to maintain separate copies of each page for each language. This would mean creating a directory for each language, with content pages for each. This is sometimes referred to as split-by-directory. While it can be easier to understand, and debug, it can become tedious to have to replicate any non-text changes across all the separate copies of your languages.
This approach has you maintain one copy of a page. Inputs are generated for all the text content that is tagged for translation, meaning editors can focus on providing just the translations instead of replicating all changes made to a page. You can think of it as separating your content and your layouts - a concept well established in the SSG (and CMS) world. You can change the layout and styling in one place, and have those changes reflected across all the languages you translate to.
- A CloudCannon organisation with access to publishing workflows
- A static site
While the Rosey CloudCannon Connector is mostly agnostic to which SSG you use to generate your static site, the markdown processing for each SSG is slightly different. We need to extend whatever markdown processing the SSG natively uses so that we automatically tag our block-level html body content with data-rosey
tags, usually using some kind of custom plugin.
We have provided plugins for, and currently support:
- Astro
- Jekyll
- Eleventy (coming soon)
-
Create two sites using a staging -> production publishing workflow on CloudCannon, if you don't already have one.
-
On your staging site:
a. Add the env variable
SYNC_PATHS
, with the value/rosey/
.b. If you have a Smartling account set up for automatic translations, add the env variable
DEV_USER_SECRET
. Add your Smartling API key as the value ofDEV_USER_SECRET
. -
On your production site, add the env variable
ROSEYPROD
, with a value oftrue
. -
Copy the
rosey
androsey-connector
directories to your project. In yourrosey/config.yml
add at least one language code to thelocales
array, and add your staging cloudvent url to thebase_url
key. -
Add a
.cloudcannon
directory in the root of your project if you don't have one already. Add apostbuild
file to it, replacingdist
with the output directory of your project. Taking a default 11ty build as an example; you would replacedist
with_site
. If you already have a CloudCannon postbuild file, add this logic to your current one..cloudcannon/postbuild
#!/usr/bin/env bash if [[ $ROSEYPROD != "true" ]]; then npx rosey generate --source dist node rosey-connector/roseyCloudCannonConnector.js fi if [[ $ROSEYPROD == "true" ]]; then echo "Translating site with Rosey" # By default, Rosey will place the default language under a language code, e.g. /en/index.html, and will generate a redirect file at /index.html. # By setting the flag --default-language-at-root, Rosey will output the default language at the root path, e.g. /index.html. # By setting the flag --default-language-at-root, Rosey will not generate any redirect pages. # We only want this to run on our production site, as it can interfere with CloudCannon CMS's visual editor # There's a little bit of shuffling around here to ensure the translated site ends up where CloudCannon picks up your site mv ./dist ./untranslated_site npx rosey build --source untranslated_site --dest dist fi
-
Install the following packages to your project:
package.json
:"dependencies": { "markdown-it": "^13.0.1", "node-html-markdown": "^1.3.0", "rosey": "^2.3.3", "slugify": "^1.6.6", "yaml": "^2.4.2", "smartling-api-sdk-nodejs": "^2.11.0", "dotenv": "^16.4.5", }
Extras to install if you are using Astro:
"dependencies": { "unist-util-visit": "^5.0.0", "hast-util-from-html-isomorphic": "^2.0.0", }
-
Add a
translations
collection to yourcloudcannon.config.yml
. If you have the keycollection_groups:
defined, remember to addtranslations
to a collection group, so that it is visible in your sidebar.If your site is nested in a subdirectory you'll need to remove your
source
key, and manually add the subdirectory to each path that needs it. The translations collection's pathrosey
does not need the prefix of the subdirectory since it lives in the root of our project. Schema paths in CloudCannon are not affected by thesource
key, so do not need updating.cloudcannon.config.yml
collections_config: translations: path: rosey icon: translate disable_url: true disable_add: true disable_add_folder: true disable_file_actions: false glob: - config.yaml - 'translations/**' _inputs: urlTranslation: type: text comment: Provide a translation for the URL that Rosey will build this page at.
-
This project is written in ESM syntax. If your project is in CJS, you may need to update your project, or the
rosey-connector
files.To change your project to ESM, make sure your package.json is
"type": "module"
, and either change any CJS files to.cjs
extension, or refactor to ESM syntax.Alternatively it may be easier to change the
.js
files inrosey-connector
to.mjs
, and update any.js
imports in those files. -
After your next build in CC, you should see nearly empty translations files. A url input will be generated for you to translate the page's url if need be, without anything in your site needing to be tagged. To add text content to translate, start tagging your layouts and components with data-rosey tags.
An example tag in 11ty may look like:
data-rosey="{{ heading.heading_text | slugify }}"
Or in a more complete example:
<h1 class="heading" data-rosey="{{ heading.heading_text | slugify }}">{{ heading.heading_text }}</h1>
11ty has the slugify global filter, which means you can slugify the content and use that as the translation key. If you are using an SSG that doesn't have a slugify filter built in (like Astro), you can roll your own. One has been provided in
rosey-connector/helpers/component-helper.js
.For markdown body content, you need to extend the SSG's built in markdown processing. Plugins are used to tag markdown that is turned into block level html elements, with an html attribute
data-rosey="an-example-phrase-for-translation"
. Content that is processed through the SSGs native markdown processing in templating (eg. Jekyllsmarkdownify
) will also need the same treatment, where the larger (perhaps many paragraph) phrase is broken into individual block level elements.In the case of an SSG like Jekyll, where a
markdownify
filter is built in, extending the markdown processing will also affect templating with that filter on it. In the case of an SSG like Astro a component (rosey-connector/ssgs/astroMarkdownComponent.astro
), with markdown rendering on the content it receives, is used to parse any markdown content that needs processed through your templating. This accomplishes the same thing as extending themarkdownify
filter in Jekyll - it removes the need to tag the whole piece of markdown content as one phrase, because it's automatically being tagged on all block level elements. -
To add automatic AI-powered translations - which your editors can then QA in the app - enable Smartling in your
rosey/config.yaml
file, by settingsmartling_enabled: true
. Make sure to fill in yourdev_project_id
, anddev_user_identifier
, with the credentials in your Smartling account. Ensure you have added you secret API key to your environment variables in CloudCannon, asDEV_USER_SECRET
. You can set this locally in a.env
file if you want to test it in your development environment.
Important
Make sure to not push any secret API keys to your source control. The .env
file should already be in your .gitignore.
Important
Be aware these translations have some cost involved, so make sure you understand the pricing around Smartling machine-translations before enabling this.
See a demonstration of this workflow here.
When tagging content for translation, the slugified contents of that translation should be used as the data-rosey
id.
An example in Jekyll:
<h1 class="{{c}}__title" data-rosey="{{ include.title | slugify }}">{{ include.title }}</h1>
The built in slugify
filter makes it easy to slugify the text contents for use as the data-rosey
tag. Templating with the markdownify
filter does not need tagged like this, as it will automatically be tagged with plugins.
Create a prebuild in your .cloudcannon
folder.
#!/usr/bin/env bash
echo "Moving jekyllMarkdownTaggerPlugin.rb to _plugins"
mv rosey-connector/ssgs/jekyllMarkdownTaggerPlugin.rb site/_plugins/jekyllMarkdownTaggerPlugin.rb
echo "Moved jekyllMarkdownTaggerPlugin.rb to _plugins!"
echo "Moving jekyllImageUnwrapPlugin.rb to _plugins"
mv rosey-connector/ssgs/jekyllImageUnwrapPlugin.rb site/_plugins/jekyllImageUnwrapPlugin.rb
echo "Moved jekyllImageUnwrapPlugin.rb to _plugins!"
This prebuild moves two plugins two our sites _plugins
folder. Both plugins customise the markdown processing of Jekyll; by extending how Jekyll uses Kramdown to parse the markdown. This affects page body content, and templating with the markdownify
filter. This means neither body content, nor templating with the markdownify
filter need to be tagged manually.
jekyllMarkdownPlugin.rb
tags all block level elements with data-rosey
tags. It uses the slugified text contents of the element for the value.
jekyllImagePlugin.rb
removes the wrapping paragraph element from an image. This is important so that we don't have image links mistakenly appear in our translations.
Change your postbuild to use _site
instead of dist
.
#!/usr/bin/env bash
npx @bookshop/generate
if [[ $ROSEYPROD != "true" ]];
then
npx rosey generate --source _site
node rosey-connector/roseyCloudCannonConnector.js
fi
if [[ $ROSEYPROD == "true" ]];
then
echo "Translating site with Rosey"
# By default, Rosey will place the default language under a language code, e.g. /en/index.html, and will generate a redirect file at /index.html.
# By setting the flag --default-language-at-root, Rosey will output the default language at the root path, e.g. /index.html.
# By setting the flag --default-language-at-root, Rosey will not generate any redirect pages.
# We only want this to run on our production site, as it can interfere with CloudCannon CMS's visual editor
# There's a little bit of shuffling around here to ensure the translated site ends up where CloudCannon picks up your site
mv ./_site ./untranslated_site
npx rosey build --source untranslated_site --dest _site
fi
See a demonstration of this workflow here.
To use the provided markdown plugin, and markdown component for Astro, these extra dependencies need to be installed:
"dependencies": {
"unist-util-visit": "^5.0.0",
"hast-util-from-html-isomorphic": "^2.0.0",
}
Your astro.config.mjs
should have the following configuration, or add this to yours.
import mdx from "@astrojs/mdx";
import { autoAddRoseyTags } from "./rosey-connector/ssgs/astroMarkdownTaggerPlugin.ts";
// https://astro.build/config
export default defineConfig({
site: "https://adjective-noun.cloudvent.net/", // Replace this with your own
integrations: [bookshop(), mdx()],
markdown: {
rehypePlugins: [autoAddRoseyTags],
remarkRehype: {
// https://github.com/syntax-tree/mdast-util-to-hast?tab=readme-ov-file#options
handlers: {
mdxJsxTextElement(state, node) {
return {
type: "element",
tagName: node.name,
properties: {},
children: state.all(node),
};
},
},
},
},
});
MDX allows you to use components throughout your markdown content, to allow for more complex things than traditional markdown syntax could represent. Bookshop handles the import of any Bookshop components into each file, to allow for any snippets to be added to the page. CloudCannon configuration then defines which snippets an editor can see and their editing experience for editing, or adding them to the page.
A rehype plugin has been provided to automatically tag block level markdown elements for translation. A handler has been added so that our plugin's AST parser knows what to do with any JSX elements it comes across in our mdx content.
The ./rosey-connector/ssgs/astroMarkdownTaggerPlugin.ts
plugin is used to extend Astro's parsing of markdown content into HTML. As the name suggests, it tags block level content in your markdown. This means you don't need to manually tag any content that will be processed as part of your page's body content - it should happen as part of the build.
Sometimes a component needs to contain markdown content. A type: markdown
input in CloudCannon will allow an editor to add markdown as a component's content.
Some SSGs come with a markdownify
filter out of the box that processes content from markdown to html. In such an SSG we would simply add this filter to the templating our component. In Astro, we need to roll our own with one of the many markdown processing libraries out there. A component has been provided rosey-connector/ssgs/astroMarkdownComponent.astro
to add wherever you need to parse markdown that isn't going to be automatically parsed by Astro.
Drag it into your project's components folder, and update the import import { generateRoseyMarkdownID } from "../helpers/component-helper";
to reflect it's new relative address. It can then be used throughout your components and layouts like:
<div class="mb-4" style={`color: ${block.text.color};`}>
<Markdown content={block.text.markdown_content} />
</div>
You can style the content like:
.markdown-text h1 {
font-size: 3rem;
line-height: 3rem;
}
.markdown-text h2 {
font-size: 2.5rem;
line-height: 2.5rem;
}
When tagging content for translation, the slugified contents of that translation should be used as the data-rosey
id.
A helper function has been provided. Add this to the top of your component, or layout, adjusting the import address as needed.
import { generateRoseyId } from "../../../rosey-connector/helpers/component-helper.js";
Add it to your html templating like:
<h1
class="heading"
data-rosey={generateRoseyId(block.heading.heading_text)}>
{block.heading.heading_text}
</h1>
To add a single folder as an upstream dependency, we can use a git subtree.
Initial setup of fetching the rosey-connector
directory from https://github.com/CloudCannon/rcc, for use in a downstream repository. This allows us to maintain the RCC logic in one place.
# Add remote to upstream repo, create new tracking branch, fetch immediately
# An alias may need to be set if using multiple SSH keys
git remote add -f rcc-upstream [email protected]:CloudCannon/rcc.git
git checkout -b upstream/rcc rcc-upstream/main
# Split off subdir of tracking branch into separate branch
git subtree split -q --squash --prefix=rosey-connector --annotate="[rcc] " --rejoin -b merging/rcc
# Add the split subdir on separate branch as a subdirectory on staging
git checkout staging
git subtree add --prefix=rosey-connector --squash merging/rcc
Pulling changes to the rosey-connector
directory from https://github.com/CloudCannon/rcc.
# switch back to tracking branch, fetch & rebase.
git checkout upstream/rcc
git pull rcc-upstream/main
# update the separate branch with changes from upstream
git subtree split -q --prefix=rosey-connector --annotate="[rcc] " --rejoin -b merging/rcc
# switch back to staging and use subtree merge to update the subdirectory
git checkout staging
git subtree merge -q --prefix=rosey-connector --squash merging/rcc