Bug 68841. Underline xul:radio and xul:checkbox accesskeys. r=mconnor, sr=neil

This commit is contained in:
aaronleventhal%moonset.net 2004-09-20 11:29:07 +00:00
Родитель be1ab09a7b
Коммит 5219ca9a11
8 изменённых файлов: 312 добавлений и 23 удалений

Просмотреть файл

@ -39,9 +39,9 @@
#include "nsIDOMXULDescriptionElement.idl"
[scriptable, uuid(f68136d6-1dd1-11b2-a184-a55a337e8507)]
[scriptable, uuid(c987629e-6370-45f5-86ec-aa765fa861cd)]
interface nsIDOMXULLabelElement : nsIDOMXULDescriptionElement {
attribute boolean accessKey;
attribute DOMString accessKey;
attribute DOMString control;
};

Просмотреть файл

@ -63,6 +63,7 @@
#include "nsIDocument.h"
#include "nsIDOMDocument.h"
#include "nsIDOMElement.h"
#include "nsIDOMXULLabelElement.h"
#include "nsIEventStateManager.h"
#include "nsITheme.h"
#include "nsUnicharUtils.h"
@ -247,7 +248,13 @@ nsTextBoxFrame::UpdateAttributes(nsPresContext* aPresContext,
if (aAttribute == nsnull || aAttribute == nsXULAtoms::accesskey) {
nsAutoString accesskey;
mContent->GetAttr(kNameSpaceID_None, nsXULAtoms::accesskey, accesskey);
nsCOMPtr<nsIDOMXULLabelElement> labelElement = do_QueryInterface(mContent);
if (labelElement) {
labelElement->GetAccessKey(accesskey); // Accesskey may be stored on control
}
else {
mContent->GetAttr(kNameSpaceID_None, nsXULAtoms::accesskey, accesskey);
}
if (!accesskey.Equals(mAccessKey)) {
if (!doUpdateTitle) {
// Need to get clean mTitle and didn't already

Просмотреть файл

@ -18,8 +18,27 @@
onget="return this.getAttribute('disabled') == 'true';"/>
<property name="image" onset="return this.setAttribute('image',val);"
onget="return this.getAttribute('image');"/>
<property name="accessKey" onset="return this.setAttribute('accesskey',val);"
onget="return this.getAttribute('accesskey');"/>
<property name="accessKey">
<getter>
<![CDATA[
return this.labelElement ? this.labelElement.accessKey : this.getAttribute('accesskey');
]]>
</getter>
<setter>
<![CDATA[
// If there is a label, store the accesskey on the labelElement
if (this.labelElement) {
this.labelElement.accessKey = val;
}
else {
this.setAttribute('accesskey', val);
}
return val;
]]>
</setter>
</property>
<field name="labelElement"/>
</implementation>
</binding>

Просмотреть файл

@ -2,8 +2,9 @@
<bindings id="textBindings"
xmlns="http://www.mozilla.org/xbl"
xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">
xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"
xmlns:html="http://www.w3.org/1999/xhtml">
<!-- bound to <description>s -->
<binding id="text-base">
<implementation implements="nsIDOMXULDescriptionElement, nsIAccessibleProvider">
@ -17,7 +18,7 @@
</property>
<property name="disabled" onget="return this.hasAttribute('disabled');"
onset="if (val) this.setAttribute('disabled', 'true');
else this.removeAttribute('disabled');
else this.removeAttribute('disabled');
return val;"/>
<property name="value" onget="return this.getAttribute('value');"
onset="this.setAttribute('value', val); return val;"/>
@ -27,11 +28,128 @@
</binding>
<binding id="text-label" extends="chrome://global/content/bindings/text.xml#text-base">
<content>
<html:span anonid="accessKeyParens"><children/></html:span>
</content>
<implementation implements="nsIDOMXULLabelElement">
<property name="accessKey" onget="return this.getAttribute('accesskey');"
onset="this.setAttribute('accesskey', val); return val;"/>
<property name="control" onget="return this.getAttribute('control');"
onset="this.setAttribute('control', val); return val;"/>
<constructor>
<![CDATA[
this.formatAccessKey();
]]>
</constructor>
<method name="formatAccessKey">
<body>
<![CDATA[
var control = this.labeledControlElement;
if (!control) {
var bindingParent = document.getBindingParent(this);
if (bindingParent instanceof Components.interfaces.nsIDOMXULLabeledControlElement) {
control = bindingParent; // For controls that make the <label> an anon child
}
}
if (control) {
control.labelElement = this;
}
var afterLabel = document.getAnonymousElementByAttribute(this, "anonid", "accessKeyParens");
afterLabel.textContent = ""; // This does not clear real nodes!
var oldAccessKey = this.getElementsByAttribute('class', 'accesskey').item(0);
if (oldAccessKey) { // Clear old accesskey
if (oldAccessKey.previousSibling instanceof Text) {
oldAccessKey.previousSibling.appendData(oldAccessKey.textContent)
}
else {
oldAccessKey.parentNode.insertBefore(oldAccessKey.firstChild, oldAccessKey);
}
oldAccessKey.parentNode.removeChild(oldAccessKey);
}
var accessKey = this.accessKey;
var labelText = this.textContent;
if (!accessKey || !labelText) {
return;
}
var accessKeyIndex = labelText.indexOf(accessKey);
if (accessKeyIndex < 0) { // Try again in upper case
accessKeyIndex = labelText.toUpperCase().indexOf(accessKey.toUpperCase());
}
var span = document.createElementNS("http://www.w3.org/1999/xhtml", "span");
span.className = "accesskey";
if (accessKeyIndex < 0) {
// If accesskey is not in string, append in parentheses
afterLabel.textContent = " (";
span.textContent = accessKey.toUpperCase();
afterLabel.appendChild(span);
afterLabel.appendChild(document.createTextNode(")"));
return;
}
var treeWalker = document.createTreeWalker(this, NodeFilter.SHOW_TEXT, null, true);
var accessKeyNode = treeWalker.nextNode();
while (accessKeyIndex >= accessKeyNode.length) {
accessKeyIndex -= accessKeyNode.length;
accessKeyNode = treeWalker.nextNode();
}
// Now wrap the access key in a <span class="accesskey"/>
if (accessKeyIndex) {
accessKeyNode = accessKeyNode.splitText(accessKeyIndex); // returns 2nd node from split
}
accessKeyNode.parentNode.insertBefore(span, accessKeyNode);
if (accessKeyNode.length > 1) {
accessKeyNode.splitText(1);
}
span.appendChild(accessKeyNode);
]]>
</body>
</method>
<property name="accessKey">
<getter>
<![CDATA[
var accessKey = this.getAttribute('accesskey');
if (!accessKey) {
var labeledEl = this.labeledControlElement;
if (labeledEl) {
accessKey = labeledEl.getAttribute('accesskey');
}
}
return accessKey ? accessKey[0] : null;
]]>
</getter>
<setter>
<![CDATA[
// If the control already has an accesskey store it there as well
var control = this.labeledControlElement;
if (control && control.hasAttribute('accesskey')) {
control.setAttribute('accesskey', val);
}
this.setAttribute('accesskey', val);
this.formatAccessKey();
return val;
]]>
</setter>
</property>
<property name="labeledControlElement" readonly="true"
onget="var control = this.control; return control ? document.getElementById(control) : null;" />
<property name="control" onget="return this.getAttribute('control');">
<setter>
<![CDATA[
var control = this.labeledControlElement;
if (control) {
control.labelElement = null; // No longer pointed to be this label
}
this.setAttribute('control', val);
this.formatAccessKey();
return val;
]]>
</setter>
</property>
</implementation>
</binding>
@ -46,7 +164,7 @@
controlElement.focus();
"/>
</handlers>
</binding>
</binding>
<binding id="text-link" extends="chrome://global/content/bindings/text.xml#text-label">
<handlers>

Просмотреть файл

@ -638,6 +638,10 @@ label[control] {
-moz-binding: url("chrome://global/content/bindings/text.xml#label-control");
}
html|span.accesskey {
text-decoration: underline;
}
/********** textbox **********/
textbox {

Просмотреть файл

@ -18,8 +18,27 @@
onget="return this.getAttribute('disabled') == 'true';"/>
<property name="image" onset="return this.setAttribute('image',val);"
onget="return this.getAttribute('image');"/>
<property name="accessKey" onset="return this.setAttribute('accesskey',val);"
onget="return this.getAttribute('accesskey');"/>
<property name="accessKey">
<getter>
<![CDATA[
return this.labelElement ? this.labelElement.accessKey : this.getAttribute('accesskey');
]]>
</getter>
<setter>
<![CDATA[
// If there is a label, store the accesskey on the labelElement
if (this.labelElement) {
this.labelElement.accessKey = val;
}
else {
this.setAttribute('accesskey', val);
}
return val;
]]>
</setter>
</property>
<field name="labelElement"/>
</implementation>
</binding>

Просмотреть файл

@ -2,8 +2,9 @@
<bindings id="textBindings"
xmlns="http://www.mozilla.org/xbl"
xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul">
xmlns:xul="http://www.mozilla.org/keymaster/gatekeeper/there.is.only.xul"
xmlns:html="http://www.w3.org/1999/xhtml">
<!-- bound to <description>s -->
<binding id="text-base">
<implementation implements="nsIDOMXULDescriptionElement, nsIAccessibleProvider">
@ -17,7 +18,7 @@
</property>
<property name="disabled" onget="return this.hasAttribute('disabled');"
onset="if (val) this.setAttribute('disabled', 'true');
else this.removeAttribute('disabled');
else this.removeAttribute('disabled');
return val;"/>
<property name="value" onget="return this.getAttribute('value');"
onset="this.setAttribute('value', val); return val;"/>
@ -27,11 +28,128 @@
</binding>
<binding id="text-label" extends="chrome://global/content/bindings/text.xml#text-base">
<content>
<html:span anonid="accessKeyParens"><children/></html:span>
</content>
<implementation implements="nsIDOMXULLabelElement">
<property name="accessKey" onget="return this.getAttribute('accesskey');"
onset="this.setAttribute('accesskey', val); return val;"/>
<property name="control" onget="return this.getAttribute('control');"
onset="this.setAttribute('control', val); return val;"/>
<constructor>
<![CDATA[
this.formatAccessKey();
]]>
</constructor>
<method name="formatAccessKey">
<body>
<![CDATA[
var control = this.labeledControlElement;
if (!control) {
var bindingParent = document.getBindingParent(this);
if (bindingParent instanceof Components.interfaces.nsIDOMXULLabeledControlElement) {
control = bindingParent; // For controls that make the <label> an anon child
}
}
if (control) {
control.labelElement = this;
}
var afterLabel = document.getAnonymousElementByAttribute(this, "anonid", "accessKeyParens");
afterLabel.textContent = ""; // This does not clear real nodes!
var oldAccessKey = this.getElementsByAttribute('class', 'accesskey').item(0);
if (oldAccessKey) { // Clear old accesskey
if (oldAccessKey.previousSibling instanceof Text) {
oldAccessKey.previousSibling.appendData(oldAccessKey.textContent)
}
else {
oldAccessKey.parentNode.insertBefore(oldAccessKey.firstChild, oldAccessKey);
}
oldAccessKey.parentNode.removeChild(oldAccessKey);
}
var accessKey = this.accessKey;
var labelText = this.textContent;
if (!accessKey || !labelText) {
return;
}
var accessKeyIndex = labelText.indexOf(accessKey);
if (accessKeyIndex < 0) { // Try again in upper case
accessKeyIndex = labelText.toUpperCase().indexOf(accessKey.toUpperCase());
}
var span = document.createElementNS("http://www.w3.org/1999/xhtml", "span");
span.className = "accesskey";
if (accessKeyIndex < 0) {
// If accesskey is not in string, append in parentheses
afterLabel.textContent = " (";
span.textContent = accessKey.toUpperCase();
afterLabel.appendChild(span);
afterLabel.appendChild(document.createTextNode(")"));
return;
}
var treeWalker = document.createTreeWalker(this, NodeFilter.SHOW_TEXT, null, true);
var accessKeyNode = treeWalker.nextNode();
while (accessKeyIndex >= accessKeyNode.length) {
accessKeyIndex -= accessKeyNode.length;
accessKeyNode = treeWalker.nextNode();
}
// Now wrap the access key in a <span class="accesskey"/>
if (accessKeyIndex) {
accessKeyNode = accessKeyNode.splitText(accessKeyIndex); // returns 2nd node from split
}
accessKeyNode.parentNode.insertBefore(span, accessKeyNode);
if (accessKeyNode.length > 1) {
accessKeyNode.splitText(1);
}
span.appendChild(accessKeyNode);
]]>
</body>
</method>
<property name="accessKey">
<getter>
<![CDATA[
var accessKey = this.getAttribute('accesskey');
if (!accessKey) {
var labeledEl = this.labeledControlElement;
if (labeledEl) {
accessKey = labeledEl.getAttribute('accesskey');
}
}
return accessKey ? accessKey[0] : null;
]]>
</getter>
<setter>
<![CDATA[
// If the control already has an accesskey store it there as well
var control = this.labeledControlElement;
if (control && control.hasAttribute('accesskey')) {
control.setAttribute('accesskey', val);
}
this.setAttribute('accesskey', val);
this.formatAccessKey();
return val;
]]>
</setter>
</property>
<property name="labeledControlElement" readonly="true"
onget="var control = this.control; return control ? document.getElementById(control) : null;" />
<property name="control" onget="return this.getAttribute('control');">
<setter>
<![CDATA[
var control = this.labeledControlElement;
if (control) {
control.labelElement = null; // No longer pointed to be this label
}
this.setAttribute('control', val);
this.formatAccessKey();
return val;
]]>
</setter>
</property>
</implementation>
</binding>
@ -46,7 +164,7 @@
controlElement.focus();
"/>
</handlers>
</binding>
</binding>
<binding id="text-link" extends="chrome://global/content/bindings/text.xml#text-label">
<handlers>

Просмотреть файл

@ -597,6 +597,10 @@ label[control] {
-moz-binding: url("chrome://global/content/bindings/text.xml#label-control");
}
html|span.accesskey {
text-decoration: underline;
}
/********** textbox **********/
textbox {