-
Notifications
You must be signed in to change notification settings - Fork 399
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
# Description <!-- Please include a summary of the changes and the related issue. Please also include relevant motivation and context. List any dependencies that are required for this change. --> Closes NA **Type of change** <!-- Please delete options that are not relevant. Remember to title the PR according to the type of change --> - Documentation update **How Has This Been Tested** <!-- Please add some reference about how your feature has been tested. --> **Checklist** <!-- Please go over the list and make sure you've taken everything into account --> - I added relevant documentation --------- Co-authored-by: José Francisco Calvo <[email protected]> Co-authored-by: José Francisco Calvo <[email protected]> Co-authored-by: Damián Pumar <[email protected]> Co-authored-by: Paco Aranda <[email protected]> Co-authored-by: pre-commit-ci[bot] <66853113+pre-commit-ci[bot]@users.noreply.github.com> Co-authored-by: Francisco Aranda <[email protected]> Co-authored-by: burtenshaw <[email protected]> Co-authored-by: Ben Burtenshaw <[email protected]> Co-authored-by: Leire Aguirre <[email protected]>
- Loading branch information
1 parent
0b4683b
commit 1e54a48
Showing
11 changed files
with
395 additions
and
24 deletions.
There are no files selected for viewing
Binary file added
BIN
+621 KB
argilla/docs/assets/images/how_to_guides/custom_field/3d_object_viewer.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added
BIN
+98 KB
argilla/docs/assets/images/how_to_guides/custom_field/images_in_two_columns.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
Binary file added
BIN
+84.8 KB
argilla/docs/assets/images/how_to_guides/custom_field/metadata_table.png
Loading
Sorry, something went wrong. Reload?
Sorry, we cannot display this file.
Sorry, this file is invalid so it cannot be displayed.
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,328 @@ | ||
--- | ||
description: Learn how to create custom fields using HTML, CSS, and JavaScript templates in Argilla. | ||
--- | ||
|
||
# Custom fields with layout templates | ||
|
||
This guide demonstrates how to create custom fields in Argilla using HTML, CSS, and JavaScript templates. | ||
|
||
!!! info "Main Class" | ||
|
||
```python | ||
rg.CustomField( | ||
name="custom", | ||
title="Custom", | ||
template="<div>{{record.fields.custom.key}}</div>", | ||
advanced_mode=False, | ||
required=True, | ||
description="Field description", | ||
) | ||
``` | ||
|
||
> Check the [CustomField - Python Reference](../reference/argilla/settings/fields.md#src.argilla.settings._field.CustomField) to see the attributes, arguments, and methods of the `CustomField` class in detail. | ||
|
||
## Understanding the Record Object | ||
|
||
The `record` object is the main JavaScript object that contains all the information about the Argilla `record` object in the UI, like `fields`, `metadata`, etc. Your template can use this object to display record information within the custom field. You can for example access the fields of the record by navigating to `record.fields.<field_name>` and this generally works the same for `metadata`, `responses`, etc. | ||
|
||
## Using Handlebars in your template | ||
|
||
By default, custom fields will use the [handlebars syntax engine](https://handlebarsjs.com/) to render templates with `record` information. This engine will convert the content inside the brackets `{{}}` to the values of record's field's object that you reference within your template. As described in the [Understanding the Record Object](#understanding-the-record-object) section, you can access the fields of the record by navigating to `{{record.fields.<field_name>}}`. For more complex use cases, handlebars has various [expressions, partials, and helpers](https://handlebarsjs.com/guide/) that you can use to render your data. You can deactivate the `handlebars` engine with the `advanced_mode=True` parameter in `CustomField`, then you will need to define custom javascript to access the record attributes, like described in the [Advanced Mode](#advanced-mode) section. | ||
|
||
### Usage example | ||
|
||
Because of the handlebars syntax engine, we only need to pass the HTML and potentially some CSS in between the `<style>` tags. | ||
|
||
```python | ||
css_template = """ | ||
<style> | ||
#container { | ||
display: flex; | ||
gap: 10px; | ||
} | ||
.column { | ||
flex: 1; | ||
} | ||
</style> | ||
""" # (1) | ||
|
||
html_template = """ | ||
<div id="container"> | ||
<div class="column"> | ||
<h3>Original</h3> | ||
<img src="{{record.fields.image.original}}" /> | ||
</div> | ||
<div class="column"> | ||
<h3>Revision</h3> | ||
<img src="{{record.fields.image.revision}}" /> | ||
</div> | ||
</div> | ||
""" # (2) | ||
``` | ||
|
||
1. This is a CSS template, which ensures that the container and columns are styled. | ||
2. This is an HTML template, which creates a `container` with two columns and injects the value corresponding to the `key` of the `image` field into it. | ||
|
||
We can now pass these templates to the `CustomField` class. | ||
|
||
```python | ||
import argilla as rg | ||
|
||
custom_field = rg.CustomField( | ||
name="image", | ||
template=css_template + html_template, | ||
) | ||
|
||
settings = rg.Settings( | ||
fields=[custom_field], | ||
questions=[rg.TextQuestion(name="response")], | ||
) | ||
|
||
dataset = rg.Dataset( | ||
name="custom_field_dataset", | ||
settings=settings, | ||
).create() | ||
|
||
dataset.records.log([ | ||
rg.Record( | ||
fields={ | ||
"image": { | ||
"original": "https://argilla.io/brand-assets/argilla/argilla-logo-color-black.png", | ||
"revision": "https://argilla.io/brand-assets/argilla/argilla-logo-black.png", | ||
} | ||
} | ||
)] | ||
) | ||
``` | ||
|
||
The result will be the following: | ||
|
||
![example-gallery-end](../assets/images/how_to_guides/custom_field/images_in_two_columns.png) | ||
|
||
### Example Gallery | ||
|
||
??? "Metadata in a table" | ||
|
||
You can make it easier to read metadata by displaying it in a table. This uses handlebars to iterate over the metadata object and display each key-value pair in a row. | ||
|
||
```python | ||
template = """ | ||
<style> | ||
.container { | ||
border: 1px solid #ddd; | ||
font-family: sans-serif; | ||
} | ||
.row { | ||
display: flex; | ||
border-bottom: 1px solid #ddd; | ||
} | ||
.row:last-child { | ||
border-bottom: none; | ||
} | ||
.column { | ||
flex: 1; | ||
padding: 8px; | ||
} | ||
.column:first-child { | ||
border-right: 1px solid #ddd; | ||
} | ||
</style> | ||
<div class="container"> | ||
<div class="header"> | ||
<div class="column">Metadata</div> | ||
<div class="column">Value</div> | ||
</div> | ||
{{#each record.metadata}} | ||
<div class="row"> | ||
<div class="column">{{@key}}</div> | ||
<div class="column">{{this}}</div> | ||
</div> | ||
{{/each}} | ||
</div> | ||
""" | ||
record = rg.Record( | ||
fields={"text": "hello"}, | ||
metadata={ | ||
"name": "John Doe", | ||
"age": 25, | ||
} | ||
) | ||
``` | ||
![example-gallery-end](../assets/images/how_to_guides/custom_field/metadata_table.png) | ||
|
||
??? "JSON viewer" | ||
|
||
The value of a custom field is a dictionary in Python and a JavaScript object in the browser. You can render this object as a JSON string using the `json` helper. This is implemented in Argilla's frontend for convenience. If you want to learn more about handlebars helpers, you can check the [handlebars documentation](https://handlebarsjs.com/guide/builtin-helpers.html). | ||
|
||
```python | ||
template = "{{ json record.fields.user_profile }}" | ||
|
||
record = rg.Record( | ||
fields={ | ||
"user_profile": { | ||
"name": "John Doe", | ||
"age": 30, | ||
"address": "123 Main St", | ||
"email": "[email protected]", | ||
} | ||
}, | ||
) | ||
``` | ||
|
||
## Advanced Mode | ||
|
||
When `advanced_mode=True`, you can use the `template` argument to pass a full HTML page. This allows for more complex customizations, including the use of JavaScript. The record object will be available in the global scope, so you can access it in your JavaScript code as described in the [Understanding the Record Object](#understanding-the-record-object) section. | ||
|
||
### Usage example | ||
|
||
Let's reproduce example from the [Without advanced mode](#without-advanced-mode) section but this time we will insert the [handlebars syntax engine](https://handlebarsjs.com/) into the template ourselves. | ||
|
||
```python | ||
render_template = """ | ||
<script id="template" type="text/x-handlebars-template"> | ||
<div id="container"> | ||
<div class="column"> | ||
<h3>Original</h3> | ||
<img src="{{record.fields.image.original}}" /> | ||
</div> | ||
<div class="column"> | ||
<h3>Revision</h3> | ||
<img src="{{record.fields.image.revision}}" /> | ||
</div> | ||
</div> | ||
</script> | ||
""" # (1) | ||
|
||
script = """ | ||
<script src="https://cdn.jsdelivr.net/npm/handlebars@latest/dist/handlebars.js"></script> | ||
<script> | ||
const template = document.getElementById("template").innerHTML; | ||
const compiledTemplate = Handlebars.compile(template); | ||
const html = compiledTemplate({ record }); | ||
document.body.innerHTML = html; | ||
</script> | ||
""" # (2) | ||
``` | ||
|
||
1. This is a JavaScript template script. We set `id` to `template` to use it later in our JavaScript code and `type` to `text/x-handlebars-template` to indicate that this is a Handlebars template. | ||
2. This is a JavaScript template script. We load the Handlebars library and then use it to compile the template and render the record. | ||
|
||
We can now pass these templates to the `CustomField` class, ensuring that the `advanced_mode` is set to `True`. | ||
|
||
```python | ||
import argilla as rg | ||
|
||
custom_field = rg.CustomField( | ||
name="image", | ||
template=render_template + script, | ||
advanced_mode=True | ||
) | ||
``` | ||
|
||
Besides the new `CustomField` code above, reusing the same approach as in the [Using handlebars in your template](#using-handlebars-in-your-template) section, will create a dataset and log a record to it, yielding the same result. | ||
|
||
![example-gallery-end](../assets/images/how_to_guides/custom_field/images_in_two_columns.png) | ||
|
||
### Example Gallery | ||
|
||
??? "3D object viewer" | ||
|
||
We will now use native javascript and [three.js](https://threejs.org/) to create a 3D object viewer. We will then use the `record` object directly to insert URLs from the record's fields. | ||
|
||
```python | ||
template = """ | ||
<script src="https://cdnjs.cloudflare.com/ajax/libs/three.js/r128/three.min.js"></script> | ||
<script src="https://cdn.jsdelivr.net/npm/[email protected]/examples/js/loaders/GLTFLoader.js"></script> | ||
<script src="https://cdn.jsdelivr.net/npm/[email protected]/examples/js/controls/OrbitControls.js"></script> | ||
|
||
|
||
<div style="display: flex;"> | ||
<div> | ||
<h3>Option A</h3> | ||
<canvas id="canvas1" width="400" height="400"></canvas> | ||
</div> | ||
<div> | ||
<h3>Option B</h3> | ||
<canvas id="canvas2" width="400" height="400"></canvas> | ||
</div> | ||
</div> | ||
|
||
<script> | ||
function init(canvasId, modelUrl) { | ||
let scene, camera, renderer, controls; | ||
|
||
const canvas = document.getElementById(canvasId); | ||
scene = new THREE.Scene(); | ||
camera = new THREE.PerspectiveCamera(75, 1, 0.1, 1000); | ||
renderer = new THREE.WebGLRenderer({ canvas, alpha: true }); | ||
|
||
renderer.setSize(canvas.clientWidth, canvas.clientHeight); | ||
|
||
const directionalLight = new THREE.DirectionalLight(0xffffff, 1); | ||
directionalLight.position.set(2, 2, 5); | ||
scene.add(directionalLight); | ||
|
||
const ambientLight = new THREE.AmbientLight(0x404040, 7); | ||
scene.add(ambientLight); | ||
|
||
controls = new THREE.OrbitControls(camera, renderer.domElement); | ||
controls.maxPolarAngle = Math.PI / 2; | ||
|
||
const loader = new THREE.GLTFLoader(); | ||
loader.load( | ||
modelUrl, | ||
function (gltf) { | ||
const model = gltf.scene; | ||
scene.add(model); | ||
model.position.set(0, 0, 0); | ||
|
||
const box = new THREE.Box3().setFromObject(model); | ||
const center = box.getCenter(new THREE.Vector3()); | ||
model.position.sub(center); | ||
camera.position.set(center.x, center.y, center.z + 1.2); | ||
|
||
animate(); | ||
}, | ||
undefined, | ||
function (error) { | ||
console.error(error); | ||
} | ||
); | ||
|
||
function animate() { | ||
requestAnimationFrame(animate); | ||
controls.update(); | ||
renderer.render(scene, camera); | ||
} | ||
} | ||
|
||
init("canvas1", record.fields.object.option_a); | ||
init("canvas2", record.fields.object.option_b); | ||
</script> | ||
|
||
""" | ||
``` | ||
|
||
Next, we will create a record with two URLs to 3D objects from [the 3d-arena dataset](https://huggingface.co/datasets/dylanebert/3d-arena). | ||
|
||
```python | ||
record = rg.Record( | ||
fields={ | ||
"object": { | ||
"option_a": "https://huggingface.co/datasets/dylanebert/3d-arena/resolve/main/outputs/Strawb3rry/a_bookshelf_with_ten_books_stacked_vertically.glb", | ||
"option_b": "https://huggingface.co/datasets/dylanebert/3d-arena/resolve/main/outputs/MeshFormer/a_bookshelf_with_ten_books_stacked_vertically.glb", | ||
} | ||
} | ||
) | ||
``` | ||
|
||
![example-gallery-end](../assets/images/how_to_guides/custom_field/3d_object_viewer.png) | ||
|
||
## Updating templates | ||
|
||
As described in the [dataset guide](./dataset.md), you can update certain setting attributes for a published dataset. This includes the custom fields templates, which is a usefule feature when you want to iterate on the template of a custom field without the need to create a new dataset. The following example shows how to update the template of a custom field. | ||
|
||
```python | ||
dataset.settings.fields["custom"].template = "<new-template>" | ||
dataset.update() | ||
``` |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.