Skip to content

Commit

Permalink
include xml in frontend attribution options
Browse files Browse the repository at this point in the history
  • Loading branch information
madewithkode committed Jun 17, 2024
1 parent f262be5 commit b446fab
Show file tree
Hide file tree
Showing 11 changed files with 66 additions and 12 deletions.
6 changes: 4 additions & 2 deletions frontend/src/assets/error_images.json
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,8 @@
"creator": "The British Library",
"license": "cc0",
"license_version": "1.0",
"license_url": "https://creativecommons.org/publicdomain/zero/1.0/"
"license_url": "https://creativecommons.org/publicdomain/zero/1.0/",
"frontendMediaType": "image"
}
},
{
Expand All @@ -23,7 +24,8 @@
"creator": "Edward Lagarde, American, 19th century",
"license": "cc0",
"license_version": "1.0",
"license_url": "https://creativecommons.org/publicdomain/zero/1.0/"
"license_url": "https://creativecommons.org/publicdomain/zero/1.0/",
"frontendMediaType": "image"
}
}
]
Expand Down
1 change: 1 addition & 0 deletions frontend/src/components/VErrorSection/VErrorImage.vue
Original file line number Diff line number Diff line change
Expand Up @@ -54,6 +54,7 @@ export default defineComponent({
alt: `errorImages.${image.id}`,
license: image.license as License,
license_version: image.license_version as LicenseVersion,
frontendMediaType: "image",
}
errorImage.attribution = getAttribution(errorImage, i18n)
return [errorItem.error, errorImage]
Expand Down
11 changes: 10 additions & 1 deletion frontend/src/components/VMediaInfo/VCopyLicense.vue
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,15 @@
>
<p>{{ getAttributionMarkup({ isPlaintext: true }) }}</p>
</VLicenseTabPanel>
<VLicenseTabPanel
:tab="tabs[3]"
:media-id="media.id"
:media-type="media.frontendMediaType"
>
<!-- Ignore reason: the interpolated string cannot have any whitespace around it when inside <p>, else there will be unwanted whitespace -->
<!-- prettier-ignore -->
<p id="attribution-xml" class="break-all font-mono" dir="ltr">{{ getAttributionMarkup({ isXml: true }) }}</p>
</VLicenseTabPanel>
</VTabs>
</div>
</template>
Expand All @@ -55,7 +64,7 @@ import VTabs from "~/components/VTabs/VTabs.vue"
import VTab from "~/components/VTabs/VTab.vue"
import VLicenseTabPanel from "~/components/VMediaInfo/VLicenseTabPanel.vue"
const tabs = ["rich", "html", "plain"] as const
const tabs = ["rich", "html", "plain", "xml"] as const
export default defineComponent({
name: "VCopyLicense",
Expand Down
2 changes: 1 addition & 1 deletion frontend/src/components/VMediaInfo/VLicenseTabPanel.vue
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ export default defineComponent({
* The kind of attribution shown in the tab.
*/
tab: {
type: String as PropType<"rich" | "html" | "plain">,
type: String as PropType<"rich" | "html" | "plain" | "xml">,
required: true,
},
/**
Expand Down
1 change: 1 addition & 0 deletions frontend/src/locales/scripts/en.json5
Original file line number Diff line number Diff line change
Expand Up @@ -801,6 +801,7 @@
plain: "Plain text",
copyText: "Copy text",
copied: "Copied!",
xml: "XML",
},
attribution: "This image was marked with a {link} license:",
},
Expand Down
2 changes: 1 addition & 1 deletion frontend/src/types/analytics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -145,7 +145,7 @@ export type Events = {
/** The unique ID of the media */
id: string
/** The format of the copied attribution */
format: "plain" | "rich" | "html"
format: "plain" | "rich" | "html" | "xml"
/** The media type being searched */
mediaType: MediaType
}
Expand Down
46 changes: 42 additions & 4 deletions frontend/src/utils/attribution-html.ts
Original file line number Diff line number Diff line change
Expand Up @@ -124,6 +124,21 @@ const licenseElementImg = (licenseElement: LicenseElement): string => {
const extLink = (href: string, text: string) =>
h("a", { target: "_blank", rel: "noopener noreferrer", href }, [text])

/**
* Generate a Dublin Core conforming XML attribution snippet.
*
* @param mediaItem - the media item being attributed
* @returns the XML markup of the attribution
*/
const xmlAttribution = (mediaItem: Record<string, string>) =>
`<attribution xmlns:dc="http://purl.org/dc/elements/1.1/">
<dc:creator>${mediaItem.creator}</dc:creator>
<dc:title>${mediaItem.title}</dc:title>
<dc:rights>"${mediaItem.title}" ${mediaItem.byCreator} ${mediaItem.markedLicensed} ${mediaItem.license}. ${mediaItem.viewLegal}</dc:rights>
<dc:identifier>${mediaItem.foreign_landing_url}</dc:identifier>
<dc:type>${mediaItem.type}</dc:type>
</attribution>`

/* Interfaces */

/**
Expand All @@ -139,6 +154,7 @@ export type AttributableMedia = Pick<
| "license"
| "license_version"
| "license_url"
| "frontendMediaType"
>

/**
Expand All @@ -148,6 +164,7 @@ export type AttributableMedia = Pick<
export interface AttributionOptions {
includeIcons?: boolean
isPlaintext?: boolean
isXml?: boolean
}

/* Actual util */
Expand All @@ -163,15 +180,20 @@ export interface AttributionOptions {
export const getAttribution = (
mediaItem: AttributableMedia,
i18n: VueI18n | null = null,
{ includeIcons, isPlaintext }: AttributionOptions = {
{ includeIcons, isPlaintext, isXml }: AttributionOptions = {
isPlaintext: false,
includeIcons: true,
isXml: false,
}
): string => {
if (!mediaItem) {
return ""
}

if (isXml) {
isPlaintext = true
}

const isPd = isPublicDomain(mediaItem.license)

const i18nBase = "mediaDetails.reuse.credit"
Expand Down Expand Up @@ -248,7 +270,23 @@ export const getAttribution = (
.replace(/\s{2}/g, " ")
.trim()

return isPlaintext
? attribution
: h("p", { class: "attribution" }, [attribution])
if (isXml) {
const { frontendMediaType, ...restMediaItem } = mediaItem
const { markedLicensed, license, viewLegal, creator } = attributionParts

const xmlAttributionParts = {
...restMediaItem,
markedLicensed,
license,
viewLegal,
type: frontendMediaType === "audio" ? "Sound" : "StillImage",
byCreator: creator,
}

return xmlAttribution(xmlAttributionParts)
} else if (isPlaintext) {
return attribution
} else {
return h("p", { class: "attribution" }, [attribution])
}
}
3 changes: 2 additions & 1 deletion frontend/test/locales/ar.json
Original file line number Diff line number Diff line change
Expand Up @@ -556,7 +556,8 @@
"html": "لغة البرمجة",
"plain": "نص عادي",
"copyText": "نسخ النص",
"copied": "نسخ!"
"copied": "نسخ!",
"xml": "XML"
},
"attribution": "تم تمييز هذه الصورة بترخيص {link}:"
},
Expand Down
3 changes: 2 additions & 1 deletion frontend/test/locales/es.json
Original file line number Diff line number Diff line change
Expand Up @@ -556,7 +556,8 @@
"html": "HTML",
"plain": "Texto sin formato",
"copyText": "Copiar el texto",
"copied": "¡Copiado!"
"copied": "¡Copiado!",
"xml": "XML"
},
"attribution": "Esta imagen se ha marcado con una licencia {link}:"
},
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -31,6 +31,6 @@ describe("VCopyLicense", () => {

it("should contain the correct contents", () => {
const { queryAllByText } = render(VCopyLicense, options)
expect(queryAllByText(/Copy text/i)).toHaveLength(3)
expect(queryAllByText(/Copy text/i)).toHaveLength(4)
})
})
1 change: 1 addition & 0 deletions frontend/test/unit/specs/utils/attribution-html.spec.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ const mediaItem: AttributableMedia = {
license: "pdm",
license_version: "1.0",
license_url: "https://license/url",
frontendMediaType: "image",
}

describe("getAttribution", () => {
Expand Down

0 comments on commit b446fab

Please sign in to comment.